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[ ...
随机推荐
- Tkinter单选框及滚动条
界面:左侧是单选框,右侧是信息显示框,下方是按扭 功能:点击开始爬取按扭,则会自动执行函数,显示在文本框中 indicatoron = 0 改变单选框按扭样式 效果图一: 效果图二: 效果图三: 示例 ...
- mysql 10进制与35进制之间的转换 注意Power处理bigint的问题
35进制的目的是防止0和O造成的视觉误差 BEGIN DECLARE m_StrHex35 VARCHAR(100); -- 返回35进制表示的结果 DECLARE m_Remainder B ...
- phpredis中文手册——《redis中文手册》 php版--引用他人
出处: http://www.cnblogs.com/zcy_soft/archive/2012/09/21/2697006.html 目录(使用CTRL+F快速查找命令): Key String H ...
- Bouncycastle中的RSA技术以及解决之道
一个使用bouncycastle进行安全操作的实用类 2007-04-13 12:54 import java.io.*;import java.security.*;import java.secu ...
- linux 解压,源码包
从网络上下载到的源码包, 最常见的是 .tar.gz 包, 还有一部分是 .tar.bz2包 要解压很简单 : .tar.gz 格式解压为 tar -zxvf ...
- Redis服务停止报错解决方案[NOAUTH Authentication required]
Redis服务器设置密码后,使用service redis stop 会出现以下信息: service redis stop Stopping ... OK (error) NOAUTH Authen ...
- IntelliJ IDEA修改Output输出缓存区大小【应对:too much output to process】
IntelliJ IDEA默认的Output输出缓存区大小只有1024KB,超过大小限制的就会被清除,而且还会显示[too much output to process],可通过如下配置界面进行修改O ...
- Makefile选项CFLAGS,LDFLAGS,LIBS
CFLAGS 表示用于 C 编译器的选项, CXXFLAGS 表示用于 C++ 编译器的选项.这两个变量实际上涵盖了编译和汇编两个步骤. CFLAGS: 指定头文件(.h文件)的路径,如:CFLAGS ...
- Linux系统默认服务建议开启关闭说明列表
服务名称 功能简介 建议 acpid 电源管理接口.如果是笔记本用户建议开启,可以监听内核层的相关电源事件. 开启 anacron 系统的定时任务程序.cron的一个子系统,如果定时任务错过了执行时间 ...
- (C#) 设定时间格式
private string GetCurrentDateTime() { return DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss ") ...