题意

你有 \(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. LeetCode题解之Longest Continuous Increasing Subsequence

    1.题目描述 2.问题分析 从每一个num[i]往前扫描即可. 3.代码 int findLengthOfLCIS(vector<int>& nums) { ){ return n ...

  2. 从ibd文件获取表空间id

    xtrabackup恢复过程中出现如下错误 InnoDB: Doing recovery: scanned up to log sequence number ( %) InnoDB: Doing r ...

  3. 服务器重启可能会导致SQL Server中部分数据库变为single user mode

    今天检查公司生产服务器的SQL Server数据库,惊讶的发现有三个生产数据库变为了single user mode.奇怪的是没有任何人和程序执行过SQL语句将这三个数据库设置为single user ...

  4. centOS7中Mariadb数据库安装与基本管理

    一.Mariadb数据库安装 1. 直接yum源安装 yum -y install mariadb mariadb-serversystemctl start mariadb /启动Mariadb服务 ...

  5. November 11th, 2017 Week 45th Saturday

    Happiness is a direction, not a place. 快乐是一个方向,不是一个目的. Do you remember those moments in your life wh ...

  6. 关于服务器端的Json文件的接收,踩了一早上的坑的问题

    JSON文件的发送和接收 服务器端接收的JSON文件也是String型的文件,因此不可以直接写成如下的格式,此错误格式下无法找到发送的{}内的数据,服务器会报错提示无法找到你需要的类型数据,也就是根本 ...

  7. Ecstore Linux服务器环境基本配置

    Nginx基本配置(另存为nginx.conf直接可以使用): #user nobody; worker_processes 1; error_log logs/error.log; #error_l ...

  8. CentOs7 编译安装PHP7.1.5

    1 创建php用户和用户组,并在github下载php7源码 #######新建php用户和php组 [root@typecodes ~]# groupadd -r www && us ...

  9. BZOJ5368:[PKUSC2018]真实排名(组合数学)

    Description 小C是某知名比赛的组织者,该比赛一共有n名选手参加,每个选手的成绩是一个非负整数,定义一个选手的排名是:成绩不小于他的选手的数量(包括他自己). 例如如果333位选手的成绩分别 ...

  10. 天地图api地址

    天地图地址 http://lbs.tianditu.com/api-new/examples.html 参考资料 http://lbs.tianditu.com/api-new/class.html