Codeforces 626G Raffles 【贪心】【线段树】
题意:
给n个奖池,t张彩票,q次操作。
每个奖池的奖金为pi。
每个奖池现有的彩票的数量为ai,保证ai>=1;
q次操作,每次有两种,第i个奖池的现有彩票数量加一,或减一。
不允许投票的数量多于奖池数量的二分之一。
保证:
n,t,q<=2e5
ai<=1000 pi<=1000
求在采用最佳策略的前提下获得奖金的期望。
思路:
首先要证明贪心的正确性,即把某张票投入某奖池之后其下一张票给期望做出的贡献要小于上一张彩票...
把式子写一下,求导,发现导数是单调递减的...
然后是对于每次操作的处理。
一开始一直纠结如何处理从某奖池拿出的亏损。因为按照贡献差来说第一个和后来的是有区别的,而且还要处理是否超票的问题。
但是看了卿学姐的思路...
其实思路是很简洁的,大概的内容是维护一个亏损的线段树一个盈利的线段树,亏损的意思是从某一奖池拿出一张票我们期望的减少,盈利的意思是往某一奖池投入一张票期望的增加。其实奖池的投递数量不用限制的,只要把盈利控制为0就可以了。而对于减少某奖池现有彩票的数量,直接对上限和投递数量的数组进行处理,然后更新维护这个奖池的盈利和亏损就可以了。因为亏损和盈利是可以直接根据这两个数据确定的。
线段树几天没写手有点生...
#include<bits/stdc++.h>
using namespace std;
double power[];
int num[];
int have[];
struct tr{
int get_id,lose_id,s,e;
double ans,get,lose;
void updatte(){
ans=power[s]*min(1.0*have[s]/(have[s]+num[s]),0.5);
get_id=lose_id=s;
if(have[s]>=num[s])get=;
else{
get=power[s]*(have[s]+)/(have[s]+num[s]+);
get-=power[s]*have[s]/(have[s]+num[s]);
}
if(have[s]>num[s])lose=;
else if(have[s]){
lose=power[s]*have[s]/(have[s]+num[s]);
lose-=power[s]*(have[s]-)/(have[s]+num[s]-);
}
else lose=1e18;
}
};
tr tree[<<];
void update(int k){
tree[k].ans=tree[k<<].ans+tree[k<<|].ans;
tree[k].get=max(tree[k<<].get,tree[k<<|].get);
tree[k].lose=min(tree[k<<].lose,tree[k<<|].lose);
if(tree[k<<].get<tree[k<<|].get)tree[k].get_id=tree[k<<|].get_id;
else tree[k].get_id=tree[k<<].get_id;
if(tree[k<<].lose<tree[k<<|].lose)tree[k].lose_id=tree[k<<].lose_id;
else tree[k].lose_id=tree[k<<|].lose_id;
}
void build(int s,int e,int k){
tree[k].s=s;
tree[k].e=e;
if(s==e){
tree[k].updatte();
return;
}
int mid=(s+e)>>;
build(s,mid,k<<);
build(mid+,e,k<<|);
update(k);
}
int ttmp=;
void nnn(int pos,int k){
int s=tree[k].s;
int e=tree[k].e;
if(s==e){
tree[k].updatte();
return;
}
int mid=(s+e)>>;
if(pos<=mid)nnn(pos,k<<);
else nnn(pos,k<<|);
update(k);
}
int main()
{
int n,t,q;
scanf("%d%d%d",&n,&t,&q);
for(int i=;i<=n;i++){
scanf("%lf",&power[i]);
}
for(int i=;i<=n;i++){
scanf("%d",&num[i]);
}
build(,n,);
while(t>){
int get_id=tree[].get_id;
have[get_id]++;
nnn(get_id,);
t--;
}
for(int i=;i<q;i++){
int typ,id;
scanf("%d%d",&typ,&id);
if(typ==){
num[id]++;
nnn(id,);
while(tree[].get>tree[].lose){
int get_id=tree[].get_id;
int lose_id=tree[].lose_id;
have[get_id]++;
have[lose_id]--;
nnn(get_id,);
nnn(lose_id,);
}
}
else{
num[id]--;
nnn(id,);
while(tree[].get>tree[].lose){
int get_id=tree[].get_id;
int lose_id=tree[].lose_id;
have[get_id]++;
have[lose_id]--;
nnn(get_id,);
nnn(lose_id,);
}
}
printf("%.12lf\n",tree[].ans);
}
return ;
}
Codeforces 626G Raffles 【贪心】【线段树】的更多相关文章
- Codeforces 626G Raffles(贪心+线段树)
G. Raffles time limit per test:5 seconds memory limit per test:256 megabytes input:standard input ou ...
- BZOJ4391 High Card Low Card [Usaco2015 dec](贪心+线段树/set库
正解:贪心+线段树/set库 解题报告: 算辣直接甩链接qwq 恩这题就贪心?从前往后从后往前各推一次然后找一遍哪个地方最大就欧克了,正确性很容易证明 (这里有个,很妙的想法,就是,从后往前推从前往后 ...
- Buses and People CodeForces 160E 三维偏序+线段树
Buses and People CodeForces 160E 三维偏序+线段树 题意 给定 N 个三元组 (a,b,c),现有 M 个询问,每个询问给定一个三元组 (a',b',c'),求满足 a ...
- CodeForces 877E DFS序+线段树
CodeForces 877E DFS序+线段树 题意 就是树上有n个点,然后每个点都有一盏灯,给出初始的状态,1表示亮,0表示不亮,然后有两种操作,第一种是get x,表示你需要输出x的子树和x本身 ...
- [Codeforces 1197E]Culture Code(线段树优化建图+DAG上最短路)
[Codeforces 1197E]Culture Code(线段树优化建图+DAG上最短路) 题面 有n个空心物品,每个物品有外部体积\(out_i\)和内部体积\(in_i\),如果\(in_i& ...
- [Codeforces 1199D]Welfare State(线段树)
[Codeforces 1199D]Welfare State(线段树) 题面 给出一个长度为n的序列,有q次操作,操作有2种 1.单点修改,把\(a_x\)修改成y 2.区间修改,把序列中值< ...
- [Codeforces 316E3]Summer Homework(线段树+斐波那契数列)
[Codeforces 316E3]Summer Homework(线段树+斐波那契数列) 顺便安利一下这个博客,给了我很大启发(https://gaisaiyuno.github.io/) 题面 有 ...
- 【题解】P1712 [NOI2016]区间(贪心+线段树)
[题解]P1712 [NOI2016]区间(贪心+线段树) 一个observe是,对于一个合法的方案,将其线段长度按照从大到小排序后,他极差的来源是第一个和最后一个.或者说,读入的线段按照长度分类后, ...
- codeforces 626 G. Raffles(线段树+思维+贪心)
题目链接:http://codeforces.com/contest/626/problem/G 题解:这题很明显买彩票肯定要买贡献最大的也就是说买p[i]*(num[i]+1)/(num[i]+a[ ...
随机推荐
- 1 TKinter小窗口及标题
说明 :本博客户关于tkinter的知识参考:2014 年辛星 Python 界面编程教程第二版 创建一个GUI程序的步骤: 创建一个GUI程序 1.导入Tkinter模块 2.创建控件 3.指定这个 ...
- DBA常用SQL之表空间与数据文件
)), ) ts_size from dba_tablespaces t, dba_data_files d where t.tablespace_name = d.tablespace_name g ...
- MockMultipartFile
org.springframework.mock.webClass MockMultipartFile java.lang.Object org.springframework.mock.web.Mo ...
- VBA 打开一个string指定的文件
Open csvFileName For Input As #1 Dim lineChanger as String lineChanger = Chr(13) fileData = Split(St ...
- 微信公众平台Php版php开发(转)
http://www.1990c.com/?p=932 近在做微信公众平台开发,一口气写了二十几个功能,挺有意思的- 今天来分享一下开发经验~微信公众平台提供的接口很简单,先看看消息交互流程: 说 ...
- HackerRank "The Indian Job"
A sly knapsack problem in disguise! Thanks to https://github.com/bhajunsingh/programming-challanges/ ...
- 域控制器中的FSMO角色
FSMO是Flexible single master operation的缩写,意思就是灵活单主机操作.营运主机(Operation Masters,又称为Flexible Single Maste ...
- Hadoop学习2--Linux准备及环境准备
1.环境安装: 虚拟机:VMware Player 系统:Ubuntu12 注意事项:注意位数,包括系统,java,Hadoop 2.切换账号 当前登录账号是自己的账号,如果想切换到root,且是第一 ...
- 黄聪:阿里云Windows2012服务器IIS8实现wordpress完美伪静态(ISAPIRewritev)
1.下载64位URL重写组件:http://www.iis.net/downloads/microsoft/url-rewrite (可以直接下载:urlrewrite2.rar) 2.暂停IIS ...
- vim 学习日志(1):剪切,复制,粘贴,删除,撤销
一:光标命令 左h 上j 下k 右l :goto n 表示跳转到文本的第n个字符 :n 表示跳到第n行 nG n为行数,该命令立即使光标跳到指定行:n为空,光标跳到文件最后一行. Ctrl+G ...