主席树+二分 p4602

题意:给出每一种果汁的美味度,价格,升数;

m个询问,每个询问给出最高上限的钱g,以及给出最少的w

意思是,最多用g的钱去买最少l的果汁,问能得到的最大美味度;

美味度是取所有果汁中美味度的最小值;

所以这道题有:美味度,价格,升数,

一开始想的时候,因为多了一个条件不知道怎么操作,看了题解之后才发现,将其中的美味度拿来二分了;

也就是说,题目中的美味度取值,是二分出来的;

那么如何建树呢?

因为美味度二分,自然建树的时候是拿美味度作为主体;

也就是说,按美味度从小到大建树(小的美味度可以包括大的美味度)

然后价格作为权值,维护 花费和升数;

那么   在最后提问的时候,我们给定一个l,r; 二分他的mid;

如果他的mid二分出来的值符合,则更新;

那么   如何二分呢,我们将钱最为一个条件放入数中去深搜,如果在这个钱的范围内能找到大于等于升数的,就ans=mid,r=mid-1;

否则则反;

 1 #include<cstdio>
 2 #include<algorithm>
 3 #include<string.h>
 4 #include<queue>
 5 using namespace std;
 6 typedef long long ll;
 7 const ll maxn=1e5+10;
 8 ll sum[maxn],b[maxn],root[maxn];
 9 struct node
10 {
11     ll d,p,l;
12 }G[maxn];
13 struct Node
14 {
15     ll ln,rn;
16     ll sum,val;
17 }tree[maxn<<5]; ll cnt;
18 bool cmp(node a,node b)
19 {
20     return a.d>b.d;
21 }
22 void update(ll pos,ll &x,ll y,ll l,ll r,ll p,ll k)
23 {
24     tree[++cnt]=tree[y];x=cnt;
25     tree[x].sum+=p*k;
26     tree[x].val+=k;
27     if(l==r) return;
28     ll mid=l+r>>1;
29     if(pos<=mid) update(pos,tree[x].ln,tree[y].ln,l,mid,p,k);
30     else update(pos,tree[x].rn,tree[y].rn,mid+1,r,p,k);
31 }
32 ll query(ll root,ll l,ll r,ll limit)
33 {
34     if(l==r){
35         if(tree[root].val*b[l]>=limit) return limit/b[l];
36         else return tree[root].val;
37     }
38     ll mid=l+r>>1;
39     ll left=tree[root].ln,right=tree[root].rn;
40     if(tree[left].sum>limit) return query(left,l,mid,limit);
41     else return tree[left].val+query(right,mid+1,r,limit-tree[left].sum);
42 }
43 int main()
44 {
45     ll n,m;
46     scanf("%lld%lld",&n,&m);
47     for(ll i=1;i<=n;i++){
48         scanf("%lld%lld%lld",&G[i].d,&G[i].p,&G[i].l);
49         b[i]=G[i].p;
50     }
51     sort(b+1,b+1+n);
52     ll limit=unique(b+1,b+1+n)-b-1;
53     sort(G+1,G+1+n,cmp);
54     for(ll i=1;i<=n;i++){
55         sum[i]=sum[i-1]+G[i].l;
56         ll pos=lower_bound(b+1,b+1+limit,G[i].p)-b;
57         update(pos,root[i],root[i-1],1,limit,G[i].p,G[i].l);
58     }
59     while(m--){
60         ll g,w;
61         scanf("%lld%lld",&g,&w);
62         if(sum[n]<w){
63             printf("-1\n");
64             continue;
65         }
66         ll left=lower_bound(sum+1,sum+1+n,w)-sum;
67         ll right=n,ans=-1;
68         while(left<=right){
69             ll mid=left+right>>1;
70             if(query(root[mid],1,limit,g)>=w){
71                 ans=mid;right=mid-1;
72             }
73             else left=mid+1;
74         }
75         if(ans==-1) printf("-1\n");
76         else printf("%lld\n",G[ans].d);
77     }
78     return 0;
79 }

原文地址:https://www.cnblogs.com/pangbi/p/11663975.html

时间: 2024-10-09 18:50:57

主席树+二分 p4602的相关文章

HDU6621 K-th Closest Distance HDU2019多校训练第四场 1008(主席树+二分)

HDU6621 K-th Closest Distance HDU2019多校训练第四场 1008(主席树+二分) 传送门:http://acm.hdu.edu.cn/showproblem.php?pid=6621 题意: 给你n个数,有m次询问 每次问你在区间[l,r]内 第k小的|\(a_i-p\)|是多少 题解: 主席树+二分 每次二分答案 如果p+mid到p-mid的值的个数大于k个的话,mid值就是可行了,然后缩小区间往左找即可 因为保证有解,所以二分出来的mid值就是答案了 que

13年山东省赛 Boring Counting(离线树状数组or主席树+二分or划分树+二分)

转载请注明出处: http://www.cnblogs.com/fraud/          ——by fraud 2224: Boring Counting Time Limit: 3 Sec  Memory Limit: 128 MB Description In this problem you are given a number sequence P consisting of N integer and Pi is the ith element in the sequence.

bzoj 2653 middle (主席树+二分)

版权声明:本文为博主原创文章,未经博主允许不得转载. bzoj 2653 题意: 一个长度为n的序列a,设其排过序之后为b,其中位数定义为b[n/2],其中a,b从0开始标号,除法取下整. 给你一个长度为n的序列s. 回答Q个这样的询问:s的左端点在[a,b]之间,右端点在[c,d]之间的子序列中,最大的中位数. 其中a<b<c<d. 位置也从0开始标号. 强制在线. 解法: 首先可以想到的是二分答案,再判断是否满足条件 . 对于答案x,我们将原数组中大于等于x的数记1,小于的记为-1,

BZOJ1926 [Sdoi2010]粟粟的书架 【主席树 + 二分 + 前缀和】

题目 幸福幼儿园 B29 班的粟粟是一个聪明机灵.乖巧可爱的小朋友,她的爱好是画画和读书,尤其喜欢 Thomas H. Co rmen 的文章.粟粟家中有一个 R行C 列的巨型书架,书架的每一个位置都摆有一本书,上数第i 行.左数第j 列 摆放的书有Pi,j页厚.粟粟每天除了读书之外,还有一件必不可少的工作就是摘苹果,她每天必须摘取一个指定的 苹果.粟粟家果树上的苹果有的高.有的低,但无论如何凭粟粟自己的个头都难以摘到.不过她发现, 如果在脚 下放上几本书,就可以够着苹果:她同时注意到,对于第

HDU - 6621 K-th Closest Distance 主席树+二分答案

K-th Closest Distance 主席树第二波~ 题意 给你\(n\)个数\(m\)个询问,问\(i\in [l,r]\)计算每一个\(|a_{i}-p|\)求出第\(k\)小 题目要求强制在线\(l = l \oplus ans.r = r \oplus ans.p = p \oplus ans.k = k \oplus ans\)(ans为上次询问的答案) 思路 二分答案\(ans\),找区间\(i\in[l,r], a_{i} \in [p-ans, p+ans]\)里的数量\(

【BZOJ2653】【主席树+二分】middle

Description 一个长度为n的序列a,设其排过序之后为b,其中位数定义为b[n/2],其中a,b从0开始标号,除法取下整. 给你一个长度为n的序列s. 回答Q个这样的询问:s的左端点在[a,b]之间,右端点在[c,d]之间的子序列中,最大的中位数. 其中a<b<c<d. 位置也从0开始标号. 我会使用一些方式强制你在线. Input 第一行序列长度n. 接下来n行按顺序给出a中的数. 接下来一行Q. 然后Q行每行a,b,c,d,我们令上个询问的答案是x(如果这是第一个询问则x=0

[hdu-6621]K-th Closest Distance 主席树 线段树 2019 多校4

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6621 题意:给一个数组,多次询问 l,r,k,p,问 l~r区间 第k小的|p-ai|是多少1<=p.ai<1e6 .1 <= K <= 169, R - L + 1 >= K,1 <= n, m <= 10^5 题解:二分答案+主席树二分绝对值答案r,然后用主席树查[p-r , p+r]区间有多少个数,如果大于等于k个减小r并记录答案,小于k个增大r有一种情况是r满

【BZOJ-1926】粟粟的书架 二分 + 前缀和 + 主席树

1926: [Sdoi2010]粟粟的书架 Time Limit: 30 Sec  Memory Limit: 552 MBSubmit: 616  Solved: 238[Submit][Status][Discuss] Description 幸福幼儿园 B29 班的粟粟是一个聪明机灵.乖巧可爱的小朋友,她的爱好是画画和读书,尤其喜欢 Thomas H. Cormen 的文章.粟粟家中有一个 R行C 列的巨型书架,书架的每一个位置都摆有一本书,上数第i 行.左数第j 列摆放的书有Pi,j页厚

【BZOJ】1146: [CTSC2008]网络管理Network(树链剖分+线段树套平衡树+二分 / dfs序+树状数组+主席树)

第一种做法(时间太感人): 这题我真的逗了,调了一下午,疯狂造数据,始终找不到错. 后来发现自己sb了,更新那里没有打id,直接套上u了.我.... 调了一下午啊!一下午的时光啊!本来说好中午A掉去学习第二种做法,噗 好吧,现在第一种做法是hld+seg+bst+二分,常数巨大,log^4级别,目前只会这种. 树剖后仍然用线段树维护dfs序区间,然后在每个区间建一颗平衡树,我用treap,(这题找最大啊,,,囧,并且要注意,这里的rank是比他大的数量,so,我们在二分时判断要判断一个范围,即要