[BJOI2019]奥术神杖
https://www.luogu.org/problemnew/show/P5319
题解
首先观察我们要求的答案的形式:
\]
这个东西貌似还不能最优化,根据套路论,把这个东西整体取个\(ln\),于是就变成了:
\]
然后这个东西就可以分数规划了,验证的话跑个dp就好了。
有一点就是必须匹配至少一个串,题解好像在状态里加了一维0/1表示有没有匹配到一个串,我是偷懒直接加了个\(eps\)。
代码
#include<bits/stdc++.h>
#define N 1509
using namespace std;
typedef long long ll;
const double eps=1e-5;
queue<int>q;
double dp[N][N],cnt[N],mx[N],w;
int tot,ch[N][10],f[N],n,m,tag1[N][N],tag2[N][N];
char s[N],s1[N];
inline ll rd(){
ll x=0;char c=getchar();bool f=0;
while(!isdigit(c)){if(c=='-')f=1;c=getchar();}
while(isdigit(c)){x=(x<<1)+(x<<3)+(c^48);c=getchar();}
return f?-x:x;
}
inline void ins(int len){
int now=0;
for(int i=1;i<=len;++i){
if(!ch[now][s1[i]-'0'])ch[now][s1[i]-'0']=++tot;
now=ch[now][s1[i]-'0'];
}
mx[now]+=w;cnt[now]++;
}
inline void build(){
for(int i=0;i<10;++i)if(ch[0][i])q.push(ch[0][i]);
while(!q.empty()){
int u=q.front();q.pop();
for(int i=0;i<10;++i){
if(ch[u][i])f[ch[u][i]]=ch[f[u]][i],mx[ch[u][i]]+=mx[ch[f[u]][i]],
cnt[ch[u][i]]+=cnt[ch[f[u]][i]],q.push(ch[u][i]);
else ch[u][i]=ch[f[u]][i];
}
}
}
inline bool check(double mid){
memset(dp,-0x3f,sizeof(dp));
dp[0][0]=0;
// printf("%.2lf ",mid);
for(int i=1;i<=n;++i){
// printf(" ");
for(int j=0;j<=tot;++j){
if(s[i]=='.')
for(int k=0;k<10;++k){
int x=ch[j][k];
double xx=dp[i-1][j]+mx[x]-mid*cnt[x];
// printf("%.2lf ",xx);
if(xx>dp[i][x]){
dp[i][x]=xx;
tag1[i][x]=k;tag2[i][x]=j;
}
}
else{
int x=ch[j][s[i]-'0'];
double xx=dp[i-1][j]+mx[x]-mid*cnt[x];
if(xx>dp[i][x]){
dp[i][x]=xx;
tag2[i][x]=j;
}
}
}
}
// puts("");
for(int j=0;j<=tot;++j)if(dp[n][j]>eps)return 1;
return 0;
}
void dfs(int len,int now){
if(!len)return;
if(s[len]=='.')s[len]=tag1[len][now]+'0';
dfs(len-1,tag2[len][now]);
// cout<<now<<" ";printf("%.2lf ",dp[len][now]);
}
int main(){
n=rd();m=rd();
double l=0,r=0;
scanf("%s",s+1);
for(int i=1;i<=m;++i){
scanf("%s",s1+1);w=log((double)rd());
r=max(r,w);
ins(strlen(s1+1));
}
r+=1000;
build();
while(l+eps<r){
double mid=(l+r)/2;
if(check(mid))l=mid;
else r=mid;
}
// printf("%.2lf\n",l);
check(l);
for(int j=0;j<=tot;++j)if(dp[n][j]>eps){
dfs(n,j);break;
}
printf("%s",s+1);
return 0;
}
[BJOI2019]奥术神杖的更多相关文章
- [BJOI2019]奥术神杖(分数规划,动态规划,AC自动机)
[BJOI2019]奥术神杖(分数规划,动态规划,AC自动机) 题面 洛谷 题解 首先乘法取\(log\)变加法,开\(c\)次根变成除\(c\). 于是问题等价于最大化\(\displaystyle ...
- [BJOI2019]奥术神杖——AC自动机+DP+分数规划+二分答案
题目链接: [BJOI2019]奥术神杖 答案是$ans=\sqrt[c]{\prod_{i=1}^{c}v_{i}}=(\prod_{i=1}^{c}v_{i})^{\frac{1}{c}}$. 这 ...
- luoguP5319 [BJOI2019]奥术神杖(分数规划,AC自动机DP)
luoguP5319 [BJOI2019]奥术神杖(分数规划,AC自动机DP) Luogu 题解时间 难点在于式子转化,设有c个满足的子串,即求最大的 $ ans = \sqrt[c]{\prod_{ ...
- 【题解】Luogu P5319 [BJOI2019]奥术神杖
原题传送门 题目让我们最大化\(val=\sqrt[k]{\prod_{i=1}^k w_i}\),其中\(k\)是咒语的个数,\(w_i\)是第\(i\)个咒语的神力 看着根号和累乘不爽,我们两边同 ...
- [BJOI2019]奥术神杖(分数规划+AC自动机+DP)
题解:很显然可以对权值取对数,然后把几何平均值转为算术平均值,然后很显然是分数规划.先对每个模式串建立AC自动机,每个节点w[i],sz[i]分别表示以其为前缀的字符串,然后再二分最优解k,然后w[i ...
- luogu P5319 [BJOI2019]奥术神杖
传送门 要求的东西带个根号,这玩意叫几何平均数,说到平均数,我们就能想到算术平均数(就是一般意义下的平均数),而这个东西是一堆数之积开根号,所以如果每个数取对数,那么乘法会变成加法,开根号变成除法,所 ...
- #loj3089 [BJOI2019]奥术神杖
卡精度好题 最关键的一步是几何平均数的\(ln\)等于所有数字取\(ln\)后的算术平均值 那么现在就变成了一个很裸的01分数规划问题,一个通用的思路就是二分答案 现在来考虑二分答案的底层怎么写 把所 ...
- [BJOI2019]奥术神杖(AC自动机,DP,分数规划)
题目大意: 给出一个长度 $n$ 的字符串 $T$,只由数字和点组成.你可以把每个点替换成一个任意的数字.再给出 $m$ 个数字串 $S_i$,第 $i$ 个权值为 $t_i$. 对于一个替换方案,这 ...
- [BJOI2019] 奥术神杖 [取log+AC自动机+dp]
题面 传送门 思路 首先,看到这个乘起来开根号的形式,应该能想到用取$\log$的方式做一个转化: $\sqrt[n]{\prod_i a_i}=\frac{1}{n}\sum_i \log_b a_ ...
随机推荐
- 简述C#中IO的应用
在.NET Framework 中. System.IO 命名空间主要包含基于文件(和基于内存)的输入输出(I/O)服务的相关基础类库.和其他命名空间一样. System.IO 定义了一系列类.接口. ...
- 10分钟 在linux里创建.net core helloworld控制台程序
官方教程 安装linux https://www.cnblogs.com/LittleFeiHu/p/9749455.html 第一步 :选择和你本机适用的Linux版本,我这里用的是18.04. 第 ...
- jQuery(五)、筛选
1 过滤 1.eq(index | -index) 获取第N个元素,index为元素索引,-index值基于最后一个元素的位置(从 1 开始) 2.first() 获取第一个元素 3.last() 获 ...
- netty的好处
netty作为一个高性能的异步通信框架,它到底有哪些好处了,又用到哪些基础技术呢? 1.使用ServerBootstrap 作为netty服务端的启动辅助类,并且在创建ServerBootstrap时 ...
- MyBatis学习---整合SpringMVC
[目录]
- Android远程桌面助手(B1371)
Android远程桌面助手(B1371),下载:https://files.cnblogs.com/files/we-hjb/ARDC%28B1371%29.7z 1.增加了对超大分辨率4320*21 ...
- 第五篇Scrum冲刺博客
一.Daily Scrum Meeting照片 二.每个人的工作 成员 ItemID 已完成工作 明天计划完成的工作 遇到的困难 张鸿 o1 整合界面至游戏中 将其他剩余功能进行整合 游戏状态的切换 ...
- SQLServer创建用户登录
创建用户登录注意事项 密码是区分大小写的. 只有创建SQL Server登录时,才支持对密码预先进行哈希运算. 如果指定MUST_CHANGE,则CHECK_EXPIRATION和 CHECK_POL ...
- Scala操作Hbase空指针异常java.lang.NullPointerException处理
Hbase版本:Hortonworks Hbase 1.1.2 问题描述:使用Scala操作Hbase时,发生空指针异常(java.lang.RuntimeException: java.lang.N ...
- IDEA 代码风格设置
1.类注释 File -> Settings -> Editor -> File and Code Templates -> Includes -> FileHeader ...