题意:

给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 【贪心】【线段树】的更多相关文章

  1. Codeforces 626G Raffles(贪心+线段树)

    G. Raffles time limit per test:5 seconds memory limit per test:256 megabytes input:standard input ou ...

  2. BZOJ4391 High Card Low Card [Usaco2015 dec](贪心+线段树/set库

    正解:贪心+线段树/set库 解题报告: 算辣直接甩链接qwq 恩这题就贪心?从前往后从后往前各推一次然后找一遍哪个地方最大就欧克了,正确性很容易证明 (这里有个,很妙的想法,就是,从后往前推从前往后 ...

  3. Buses and People CodeForces 160E 三维偏序+线段树

    Buses and People CodeForces 160E 三维偏序+线段树 题意 给定 N 个三元组 (a,b,c),现有 M 个询问,每个询问给定一个三元组 (a',b',c'),求满足 a ...

  4. CodeForces 877E DFS序+线段树

    CodeForces 877E DFS序+线段树 题意 就是树上有n个点,然后每个点都有一盏灯,给出初始的状态,1表示亮,0表示不亮,然后有两种操作,第一种是get x,表示你需要输出x的子树和x本身 ...

  5. [Codeforces 1197E]Culture Code(线段树优化建图+DAG上最短路)

    [Codeforces 1197E]Culture Code(线段树优化建图+DAG上最短路) 题面 有n个空心物品,每个物品有外部体积\(out_i\)和内部体积\(in_i\),如果\(in_i& ...

  6. [Codeforces 1199D]Welfare State(线段树)

    [Codeforces 1199D]Welfare State(线段树) 题面 给出一个长度为n的序列,有q次操作,操作有2种 1.单点修改,把\(a_x\)修改成y 2.区间修改,把序列中值< ...

  7. [Codeforces 316E3]Summer Homework(线段树+斐波那契数列)

    [Codeforces 316E3]Summer Homework(线段树+斐波那契数列) 顺便安利一下这个博客,给了我很大启发(https://gaisaiyuno.github.io/) 题面 有 ...

  8. 【题解】P1712 [NOI2016]区间(贪心+线段树)

    [题解]P1712 [NOI2016]区间(贪心+线段树) 一个observe是,对于一个合法的方案,将其线段长度按照从大到小排序后,他极差的来源是第一个和最后一个.或者说,读入的线段按照长度分类后, ...

  9. codeforces 626 G. Raffles(线段树+思维+贪心)

    题目链接:http://codeforces.com/contest/626/problem/G 题解:这题很明显买彩票肯定要买贡献最大的也就是说买p[i]*(num[i]+1)/(num[i]+a[ ...

随机推荐

  1. Maven最佳实践:版本管理

    什么是版本管理 首先,这里说的版本管理(version management)不是指版本控制(version control),但是本文假设你拥有基本的版本控制的知识,了解subversion的基本用 ...

  2. 彻底解决ASP.NET MVC 3 404错误码返回302的问题

    转自:http://blog.csdn.net/mycloudke/article/details/9746333 404状态码:,意味着当在页面上显示用户点击不存在,提高用户体验度,搜索引擎会放弃这 ...

  3. oracle 自定义异常处理

    --第一种方式:使用raise_application_error抛出自定义异常declare i number:=-1;begin if i=-1 then raise_application_er ...

  4. du 命令,对文件和目录磁盘使用的空间的查看

    Linux du命令也是查看使用空间的,但是与df命令不同的是Linux du命令是对文件和目录磁盘使用的空间的查看,还是和df命令有一些区别的. 1.命令格式: du [选项][文件] 2.命令功能 ...

  5. SPOJ #10657. LOGIC (riddle)

    1 line in Ruby(2.0), 36B: p (1..n=gets.to_i).reduce(:*)+2**n-n

  6. 【KVM】Ubuntu14.04 安装KVM

    1. 首先检查系统是否支持CPU虚拟化 # egrep -o "svm|vmx" /proc/cpuinfo 若显示如下类似信息,则说明支持CPU虚拟化 vmx vmx ... v ...

  7. bzoj4702: 装箱游戏

    Description Alice和Bob正在玩一个非常无聊的游戏以打发时间.游戏是这样的.初始的时候,有n个箱子和m个物品.箱子.物 品都是不同的.因此有nm种方法把这些物品放到箱子里.两个人轮流操 ...

  8. bzoj4642: 泡泡

    Description "OI真的像是一条奇趣横生的路啊,也许它是绕过了高考的大山,也许确实有通往大学的捷径.但我,真的,真的只在 乎那路上美丽的泡泡." --TB   TB喜欢所 ...

  9. android学习笔记45——android的数据存储和IO

    android的数据存储和IO SharedPreferences与Editor简介 SharedPreferences保存的数据主要是类似于配置信息格式的数据,因此其保存的数据主要是简单的类型的ke ...

  10. C语言中字符串常量到底存在哪了?

    常量存储总结局部变量.静态局部变量.全局变量.全局静态变量.字符串常量以及动态申请的内存区 1.局部变量存储在栈中2.全局变量.静态变量(全局和局部静态变量)存储在静态存储区3.new申请的内存是在堆 ...