zoukankan      html  css  js  c++  java
  • 非常简单地理解带权二分(wqs二分)

    非常感性简单地理解带权二分(又名 wqs 二分),尽管不是很严谨,如有错误请指出

    (Large atural) Gosha is hunting / 原题链接 / 更好阅读体验

    解法

    设我们有 (A) 个红球和 (B) 个蓝球,用红球抓 (i) 号胖可丁的概率是 (a_i),用蓝球抓 (i) 号胖可丁的概率是 (b_i)

    首先我们有一个暴力 ( ext{DP}):设 (f_{i,j,k})(1sim i) 这些胖可丁中,用了 (j) 个红球和 (k) 个蓝球的最大期望。

    [f_{i,j,k}=max(f_{i-1,j,k},f_{i-1,j-1,k}+a_i,f_{i-1,j,k-1}+b_i,f_{i-1,j-1,k-1}+ 1-(1-a_i)(1-b_i) ) ]

    (max) 中依次代表着对于第 (i) 个胖可丁:不用精灵球;只用红球;只用蓝球;两种球都用。

    答案是 (f_{n,A,B})。当然这是 (O(n^3)) 的,显然会超时。


    我们发现,用了 ((x+1)) 个精灵球肯定会比用 (x) 个精灵球得到的期望更大。所以它是一个单调的东西。事实上,如果 (f_{i,j,k})(i,k) 都是不变常数,即它是关于 (j) 的函数,那么它是一个凸函数。因为你第一个红球肯定会抓期望最大的胖可丁,第二个红球肯定会抓期望次大的胖可丁……这样增长的速度就会越来越慢。当然, 如果 (f_{i,j,k})(i,j) 都是不变常数,即它是关于 (k) 的函数也同理。所以这个凸函数的性质意味着我们可以用带权二分。

    根据带权二分的思想,我们需要将 (f_{i,j,k}) 简化为 (f_{i,j})。我们可以假设现在蓝球是免费的,想用多少就用多少(好耶!),那这样就会有转移方程:

    [f_{i,j}=max(f_{i-1,j},f_{i-1,j}+a_i,f_{i,j-1}+b_i,f_{i-1,j-1}+1-(1-a_i)(1-b_i)) ]

    时间复杂度 (O(n^2))

    但是,毫无疑问,这样的话所有的转移肯定都会加上蓝球的贡献,毕竟免费的肯定越用多越好。所以这样想:我们给蓝球套上一个价格,即,你需要 (k) 元才能买到一个蓝球。这样的话, ( ext{DP}) 方程就是这样的了:

    [f_{i,j}=max(f_{i-1,j},f_{i-1,j-1}+a_i,f_{i,j}+b_i-k,f_{i-1,j-1}+1-(1-a_i)(1-b_i)-k) ]

    这样肯定就会有一些贫穷的 (f_{i,j}) 不能加上蓝球的贡献。

    于是,我们在 ( ext{DP}) 过程中同时记录一下使用蓝球的个数 (cnt)。如果 (cnt>B) 那么就是蓝球供不应求了。我们肯定会让 (k) 更大一点,也就是让蓝球更昂贵,更奢侈,让更多可怜的 (f_{i,j}) 买不到蓝球,以减少 (cnt)

    相反地,如果 (cnt<B),那么就是蓝球生产过剩、供大于求。我们必须要让 (k) 更小一点,也就是让蓝球更便宜,以增加 (cnt)

    如果 (cnt=B) ,恭喜,我们恰好用了 (B) 个蓝球,这应该就是合适的价格了。

    答案就是 (f_{n,A})……不对,蓝球事实上是免费的。所以我们要把那些钱还回去,答案就是 (f_{n,A}+cnt imes k)

    于是我们可以二分这个价钱 (k)。一开始二分的边界是 (l=0,r=1),因为概率都是大于等于零、小于等于一的。

    时间复杂度变为 (O(n^2log V))。其中 (log V) 是二分中产生的。


    所以你也可以继续给红球也依法炮制,给它安一个价格。而且 ( ext{DP}) 也不复存在,变成了一个贪心。更令人喜出望外的是,这样时间复杂度甚至可以变成 (O(nlog^2 V)),可以非常充裕地通过本题!

    注意最后答案是 (f_{n,A,B}+A imes k_a +B imes k_b),而不是(f_{n,A,B}+cnt_a imes k_a +cnt_b imes k_b)。原因的话需要去以斜率之类的方式去说明。由于本篇题解只是感性理解,而非用斜率去证明,所以大家就把这个东西记下来吧。

    代码

    注意精度……这个题目很在乎精度。我这里是 eps=1e-8

    而且从这个题目我知道了:人傻常数大,人傻常数低……(同校大佬一次过,时间还少)

    码风毒瘤见谅。

    #include<bits/stdc++.h>
    #define rep(i,x,y) for(int i=x;i<=y;++i)
    #define lod long double
    using namespace std;
    const int n7=2021;const lod eps=1e-8;
    int n,A,B,cnt1,cnt2;
    lod a[n7],b[n7],c[n7],tot,val1,val2,ans;
     
    void check(lod rmb1,lod rmb2){
    	cnt1=cnt2=0,tot=0;
    	rep(i,1,n){
    		bool flag1=0,flag2=0;lod tmp=0;
    		if(a[i]-rmb1>tmp+eps)tmp=a[i]-rmb1,flag1=1,flag2=0;
    		if(b[i]-rmb2>tmp+eps)tmp=b[i]-rmb2,flag1=0,flag2=1;
    		if(c[i]-rmb1-rmb2>tmp+eps)tmp=c[i]-rmb1-rmb2,flag1=1,flag2=1;
    		if(flag1)cnt1++;
            if(flag2)cnt2++;
            tot+=tmp;
    	}
    }
     
    int main(){
    	cin>>n>>A>>B;
    	rep(i,1,n)cin>>a[i];
    	rep(i,1,n)cin>>b[i];
    	rep(i,1,n)c[i]=1-(1-a[i])*(1-b[i]);
    	lod l1=0,r1=1;
    	while(l1+eps<r1){
    		lod mid1=(l1+r1)/2;
    		lod l2=0,r2=1;
    		while(l2+eps<r2){
    			lod mid2=(l2+r2)/2;
    			check(mid1,mid2);
    			val2=mid2;
    			if(cnt2==B)break;
    			if(cnt2>B)l2=mid2;
    			else r2=mid2;
    		}
    		val1=mid1;
    		if(cnt1==A)break;
    		if(cnt1>A)l1=mid1;
    		else r1=mid1;
    	}
    	ans=tot+val1*A+val2*B;
    	cout<<ans;
    	return 0;
    }
    
  • 相关阅读:
    风讯DotNetCMS sp5安装笔记
    datalist中实现自动编号写法
    windows server 2008 自动登录设置
    C#软件监控外部程序运行状态
    WPF启动屏幕SplashScreen
    Windows Server 2019 在桌面上显示“我的电脑”
    使用jquery的load方法加载html页面,html引入的js无效
    sql查询文章上一篇下一篇
    C#调用user32.dll Win32的API函数
    C#调用dll提示"试图加载格式不正确的程序
  • 原文地址:https://www.cnblogs.com/BlankAo/p/14294523.html
Copyright © 2011-2022 走看看