题意

你有 \(n\) 个任务,初始收益为 \(a\) ,共 \(t\) 轮游戏,每轮可以选择完成一个任务(可以做多次),完成之后可以给任意任务升级,升级之后的任务收益为 \(b\) ,每个任务还有完成的概率 \(p\) ,问期望最大收益是多少。

\(n\leq 10^5,1\leq a< b\leq 10^8,t\leq 10^9\)

分析

  • 一个重要而显然的结论是如果我们有了一次升级的机会,一定升级 \(b*p\) 最大的那一个,之后一直选择完成这个任务,记 \(M\) 表示最大的 \(b*p\) 。然后只关注没有一次完成时的最大期望收益。

  • 定义状态 \(f_t​\) 表示还剩 \(t​\) 轮游戏,还没有完成一个任务的期望最大收益。

  • 转移: \(f_{t+1}=\max\{p_i*(tM+a_i)+(1-p_i)*f_t\}\).

  • 可以变形成斜率优化的形式(注意这里的 \(f_t\) 要看成 \(k\) 的一部分) ,也可以写成这种形式:

\[f_{t+1}=\max\{p_i*(tM-f_t)+p_ia_i\}+f_t
\]

  • 将 \(p_i\) 看成 \(k\) , \(p_ia_i\) 看成 \(b\) ,然后每个转移都是形如直线 \(y=kx+b\) 的形式。

  • 我们可以维护一个下凸壳,每次可以二分当前的 \(x=tM-f_t\) 在哪一段,可以做到 \(Tlogn\) 的时间。

  • 发现 \(tM-f_t\)是单调不降的,证明如下:

设 \(x_{t+1} \geq x_t\)

有:\(tM-f_t\geq (t-1)M-f_{t-1}\)

\(M\geq f_t-f_{t-1}\)

考虑 \(f​\) 的实际意义可以发现上式一定成立,因为一轮的期望收益不会超过 \(M\) 。

可以考虑构造,对于 \(t+1\) 轮的最优决策, \(t\) 轮可以复制过来(除开第一轮),而在 \(t+1\) 轮产生的每种情况,\(t\) 轮都可以效仿(发生的概率是相同的),发现期望收益只有最后一轮不一样,最多差为 \(M\)。

  • 然后在每一段倍增矩乘即可。

  • 总时间复杂度为 \(O(n\log T)\)。

代码

#include<bits/stdc++.h>
using namespace std;
#define go(u) for(int i=head[u],v=e[i].to;i;i=e[i].lst,v=e[i].to)
#define rep(i,a,b) for(int i=a;i<=b;++i)
#define pb push_back
typedef long long LL;
inline int gi(){
int x=0,f=1;char ch=getchar();
while(!isdigit(ch)) {if(ch=='-') f=-1;ch=getchar();}
while(isdigit(ch)){x=(x<<3)+(x<<1)+ch-48;ch=getchar();}
return x*f;
}
template<typename T>inline bool Max(T &a,T b){return a<b?a=b,1:0;}
template<typename T>inline bool Min(T &a,T b){return b<a?a=b,1:0;}
const int N=1e5 + 7;
typedef double db;
const db eps=1e-12;
int n,len,tp,st[N];
LL t;
db a[N],b[N],p[N],M;
int dcmp(db x){
if(fabs(x)<eps) return 0;
return x<0?-1:1;
}
struct line{
db k,b;int id;
line(){}line(db k,db b,int id):k(k),b(b),id(id){}
bool operator <(const line &rhs)const{
if(dcmp(k-rhs.k)!=0) return dcmp(k-rhs.k)<0;
return dcmp(b-rhs.b)<0;
}
}l[N];
db X(int a,int b){
return (l[b].b-l[a].b)/(l[a].k-l[b].k);
}
struct mat{
db v[5][5];
mat(){memset(v,0,sizeof v);}
void init(){memset(v,0,sizeof v);}
mat operator *(const mat &rhs)const{
mat res;
rep(i,0,4)
rep(j,0,4)
rep(k,0,4)
res.v[i][j]+=v[i][k]*rhs.v[k][j];
return res;
}
}A,B[34],C;
mat Pow(mat a,LL b){
mat res;
rep(i,1,4) res.v[i][i]=1;
for(;b;b>>=1,a=a*a) if(b&1) res=res*a;
return res;
}
int main(){
scanf("%d%I64d\n",&n,&t);
rep(i,1,n) {
scanf("%lf%lf%lf",&a[i],&b[i],&p[i]);
l[i]=line(p[i],p[i]*a[i],i);
Max(M,b[i]*p[i]);
}
sort(l+1,l+1+n);
rep(i,1,n) {
if(dcmp(l[i].k-l[i+1].k)==0) continue;
l[++len]=l[i];
}
rep(i,1,len){
while(tp>1&&dcmp(X(st[tp-1],i)-X(st[tp],st[tp-1]))<=0) --tp;
st[++tp]=i;
}
int now=1;LL cnt=0;
for(int now=1;now<=tp&&cnt^t;++now){
double R=cnt*M-A.v[1][1];
while(now<tp&&X(st[now],st[now+1])<=R) ++now; int i=st[now];R=X(st[now],st[now+1]);
B[0].init();
B[0].v[1][1]=1-l[i].k;
B[0].v[2][1]=l[i].k;
B[0].v[2][2]=B[0].v[3][1]=B[0].v[3][3]=B[0].v[4][2]=B[0].v[4][4]=1;
A.v[1][3]=l[i].b,A.v[1][4]=M; rep(i,1,33) B[i]=B[i-1]*B[i-1]; for(int i=33;~i;--i)if(t-cnt>(1ll<<i)){
C=A*B[i];
if(now==tp||dcmp((cnt+(1ll<<i))*M-C.v[1][1]-R)<0) A=C,cnt+=(1ll<<i);
}
cnt++,A=A*B[0];
}
printf("%.13lf\n",A.v[1][1]);
return 0;
}

[CF1067D]Computer Game[凸包/斜率优化+倍增+矩阵乘法]的更多相关文章

  1. CF1067D. Computer Game(斜率优化+倍增+矩阵乘法)

    题目链接 https://codeforces.com/contest/1067/problem/D 题解 首先,如果我们获得了一次升级机会,我们一定希望升级 \(b_i \times p_i\) 最 ...

  2. CF781D Axel and Marston in Bitland [倍增 矩阵乘法 bitset]

    Axel and Marston in Bitland 好开心第一次补$F$题虽然是$Div.2$ 题意: 一个有向图,每条边是$0$或$1$,要求按如下规则构造一个序列然后走: 第一个是$0$,每次 ...

  3. 【loj2325】「清华集训 2017」小Y和恐怖的奴隶主 概率dp+倍增+矩阵乘法

    题目描述 你有一个m点生命值的奴隶主,奴隶主受伤未死且当前随从数目不超过k则再召唤一个m点生命值的奴隶主. T次询问,每次询问如果如果对面下出一个n点攻击力的克苏恩,你的英雄期望会受到到多少伤害. 输 ...

  4. 倍增&矩阵乘法 专题复习

    倍增&矩阵乘法 专题复习 PreWords 这两个基础算法我就不多说啦,但是还是要介绍一下" 广义矩阵 "乘法 其实就是把矩阵换成取\(max\),然后都一样... 据神仙 ...

  5. BZOJ 3672[NOI2014]购票(树链剖分+线段树维护凸包+斜率优化) + BZOJ 2402 陶陶的难题II (树链剖分+线段树维护凸包+分数规划+斜率优化)

    前言 刚开始看着两道题感觉头皮发麻,后来看看题解,发现挺好理解,只是代码有点长. BZOJ 3672[NOI2014]购票 中文题面,题意略: BZOJ 3672[NOI2014]购票 设f(i)f( ...

  6. 4.28 省选模拟赛 负环 倍增 矩阵乘法 dp

    容易想到 这个环一定是简单环. 考虑如果是复杂环 那么显然对于其中的第一个简单环来说 要么其权值为负 如果为正没必要走一圈 走一部分即可. 对于前者 显然可以找到更小的 对于第二部分是递归定义的. 综 ...

  7. Codeforces 576D - Flights for Regular Customers(bitset 优化广义矩阵乘法)

    题面传送门 题意: 有一张 \(n\) 个点 \(m\) 条边的有向图,你初始在 \(1\) 号点,边上有边权 \(c_i\) 表示只有当你经过至少 \(c_i\) 条边的时候你才能经过第 \(i\) ...

  8. 【POJ3613】Cow Relays 离散化+倍增+矩阵乘法

    题目大意:给定一个 N 个顶点,M 条边的无向图,求从起点到终点恰好经过 K 个点的最短路. 题解:设 \(d[1][i][j]\) 表示恰好经过一条边 i,j 两点的最短路,那么有 \(d[r+m] ...

  9. dp斜率优化

    算法-dp斜率优化 前置知识: 凸包 斜率优化很玄学,凭空讲怎么也讲不好,所以放例题. [APIO2014]序列分割 [APIO2014]序列分割 给你一个长度为 \(n\) 的序列 \(a_1,a_ ...

随机推荐

  1. python turtle 绘制图像

    def _circle(): t = turtle.Turtle() t.pencolor("yellow") t.pensize(5) t.speed(200) t.circle ...

  2. IHttpModule 和 IHttpHandler 配置方法

    <?xml version="1.0" encoding="utf-8"?> <configuration> <appSettin ...

  3. MySQL 支持utf8mb4

    utf8mb4 utf8mb3 utf8 Refer to The utf8mb4 Character Set The utf8 Character Set (Alias for utf8mb3) M ...

  4. 看代码网备份|利用WebClient|eKing.CmdDownLoadDbBakOper|实现定时拷贝数据库备份文件到文件服务器

    摘要: 1.有两台服务器 (1)看代码网(记为A):内网IP:10.186.73.30 (2)文件服务器(记为B):内网IP:10.135.87.157 2.在A架设一个网站,端口8088(防火强设置 ...

  5. mysql数据导入导出与数据表优化

    一.数据导入 mysqlimport -uroot oa d:/aa.txt --fields-terminated-by=, --fields-optionally-enclosed-by= --l ...

  6. Hyper-v虚拟化

    1.打开Hyper-V管理器,新建虚拟机 2.点击下一步,继续设置 3.设置虚拟机名称和存储位置,点击选择打钩,自定义路径 4.设置虚拟机运行内存 5.设置虚拟机设置好的网络 6.创建虚拟磁盘 7创建 ...

  7. python3: 文件与IO

    1.读写文本数据 # Write chunks of text data with open('somefile.txt', 'wt') as f: f.write(text1) # Redirect ...

  8. [ML学习笔记] 朴素贝叶斯算法(Naive Bayesian)

    [ML学习笔记] 朴素贝叶斯算法(Naive Bayesian) 贝叶斯公式 \[P(A\mid B) = \frac{P(B\mid A)P(A)}{P(B)}\] 我们把P(A)称为"先 ...

  9. ansible(一)

    Ansible简介 Ansible:Ansible的核心程序Host Lnventory:记录了每一个由Ansible管理的主机信息,信息包括ssh端口,root帐号密码,ip地址等等.可以通过fil ...

  10. postgresql----几何类型和函数

    postgresql支持的几何类型如下表: 名字 存储空间 描述 表现形式 point 16字节 平面上的点 (x,y) line 32字节 直线 {A,B,C} lseg 32字节 线段 ((x1, ...