Luogu 1314 【NOIP2011】聪明的质检员 (二分)

mac2022-06-30  22

Luogu 1314 【NOIP2011】聪明的质检员 (二分)

Description

小 T 是一名质量监督员,最近负责检验一批矿产的质量。这批矿产共有n个矿石,从 1 到n逐一编号,每个矿石都有自己的重量wi以及价值vi。检验矿产的流程是:

给定 m个区间[Li,Ri];选出一个参数W;对于一个区间[Li,Ri],计算矿石在这个区间上的检验值Yi:

\[Y_i= \sum_{j} 1×\sum _{j}v_j,j \in [L_i,R_i],W_j>=W\] 这批矿产的检验结果Y为各个区间的检验值之和。即:\[Y=\sum_{i=1}^{m}Y_i\] 若这批矿产的检验结果与所给标准值 S相差太多,就需要再去检验另一批矿产。小 T 不想费时间去检验另一批矿产,所以他想通过调整参数W的值,让检验结果尽可能的靠近标准值 S,即使得S−Y的绝对值最小。请你帮忙求出这个最小值。

Input

第一行包含三个整数n,m,S,分别表示矿石的个数、区间的个数和标准值。 接下来的n 行,每行2 个整数,中间用空格隔开,第i+1 行表示i 号矿石的重量wi 和价值vi 。 接下来的m 行,表示区间,每行2 个整数,中间用空格隔开,第i+n+1 行表示区间[Li,Ri]的两个端点Li 和Ri。注意:不同区间可能重合或相互重叠。

Output

输出只有一行,包含一个整数,表示所求的最小值。

Sample Input

5 3 15 1 5 2 5 3 5 4 5 5 5 1 5 2 4 3 3

Sample Output

10

Http

Luogu:https://www.luogu.org/problem/show?pid=1314

Source

二分

题目大意

给出n组二元组和m个区间,现在定义一个区间上的检验结果为这个区间上所有二元组中,第一个数大于W的二元组的第二个数*这些二元组的个数。定义整个的检验结果为给定的m个区间的检验结果之和。将整个的检验结果与一给定的标准值相比,两者之差的绝对值即为这个W对应的答案。现在求W让这个答案最小,求出最小值。

解决思路

讲题目的思路理清后,我们可以想到二分W的值。因为这个整个的检验结果与W是保持单调的,若W增大,则检验结果变小,反之变大。 所以我们可以二分W,每次二分出W后,计算一下其检验结果,若检验结果大于给定的S,则将左端点右移,否则将右端点左移。注意,最后输出的答案是每一次的检验结果与S作差的绝对值的最小值。 至于计算检验结果,因为题中给定的都是区间,所以我们可以每找出一个W后\(O(n)\)计算一下前缀和,然后\(O(1)\)地计算区间和。 注意,所有的变量都要开long long。

代码

#include<iostream> #include<cstdio> #include<cstdlib> #include<cstring> #include<algorithm> using namespace std; #define ll long long const int maxN=300000; const int inf=2147483647; ll n,m,S; ll Ans=1e13; ll Weight[maxN]; ll Value[maxN]; ll Rangel[maxN]; ll Ranger[maxN]; ll Sum[maxN]; ll Cnt[maxN]; ll read(); bool Solve(ll nowW); int main() { n=read(); m=read(); S=read(); ll l=0,r=0; for (int i=1;i<=n;i++) { Weight[i]=read(); Value[i]=read(); r=max(r,Weight[i]);//r取最大值 } for (int i=1;i<=m;i++) { Rangel[i]=read(); Ranger[i]=read(); } r=r+100;//为了防止出错,扩大上限 while (l<=r)//二分W { ll mid=(l+r)>>1; if (Solve(mid)) l=mid+1; else r=mid-1; //cout<<l<<" "<<r<<endl; } cout<<Ans<<endl; return 0; } ll read() { ll x=0; char ch=getchar(); while ((ch<'0')||(ch>'9')) ch=getchar(); while ((ch>='0')&&(ch<='9')) { x=x*10+ch-48; ch=getchar(); } return x; } bool Solve(ll nowW) { Sum[0]=0; Cnt[0]=0; for (int i=1;i<=n;i++)//计算前缀和,Sum是价值之和,Cnt是人数之和 { Sum[i]=Sum[i-1]+((Weight[i]>=nowW)?(Value[i]):0); Cnt[i]=Cnt[i-1]+((Weight[i]>=nowW)?1:0); } ll tot=0; for (int i=1;i<=m;i++)//计算区间贡献之和 tot+=(Sum[Ranger[i]]-Sum[Rangel[i]-1])*(Cnt[Ranger[i]]-Cnt[Rangel[i]-1]); Ans=min(Ans,abs(tot-S));//取最优值 if (tot>=S) return 1; return 0; }

转载于:https://www.cnblogs.com/SYCstudio/p/7499449.html

相关资源:JAVA上百实例源码以及开源项目
最新回复(0)