原题传送门

题目让我们最大化\(val=\sqrt[k]{\prod_{i=1}^k w_i}\),其中\(k\)是咒语的个数,\(w_i\)是第\(i\)个咒语的神力

看着根号和累乘不爽,我们两边同取\(\ln\)

$$\ln val=\frac{1}{k}\sum_{i=1}^k \ln w_i$$

易知当\(\ln val\)最大化时,\(val\)也最大化。所以我们将问题转化成了最大化\(\frac{1}{k}\sum_{i=1}^k \ln w_i\),我们发现这是算数平均数。我们珂以通过二分答案找到它的最大值,问题就是二分答案如何check是否合法:

当\(\frac{1}{k}\sum_{i=1}^k \ln w_i>mid\)时才合法

即当\(\sum_{i=1}^k(w_i-mid)>0\)时才合法

我们先对所有咒语建AC自动机,在上面跑dp求出\(\sum_{i=1}^k(w_i-mid)\)的最大值,判断是否可行

具体dp:就像其他很多AC自动机上的dp一样,设\(f[i][j]\)表示神杖前\(i\)个字符,匹配到了AC自动机上\(j\)号节点,依照套路转移,就是不要忘了题目原有的限制

此算法精度误差较大,但本题还是珂以通过

#include <bits/stdc++.h>
#define db double
#define N 1505
#define eps 1e-6
using namespace std;
inline int read()
{
register int x=0,f=1;register char ch=getchar();
while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
while(ch>='0'&&ch<='9')x=(x<<3)+(x<<1)+ch-'0',ch=getchar();
return x*f;
}
int n,m;
char T[N],s[N],ans[N];
db V[N],val[N],f[N][N];
pair<int,int> pre[N][N];
struct node{
int son[10],fail,cnt;
db val;
}tr[N];
int tot=0;
inline void Insert(register char *s,register db v)
{
int len=strlen(s+1),now=0;
for(register int i=1;i<=len;++i)
{
if(!tr[now].son[s[i]-'0'])
tr[now].son[s[i]-'0']=++tot;
now=tr[now].son[s[i]-'0'];
}
++tr[now].cnt,tr[now].val+=v;
}
inline void getfail()
{
queue<int> q;
for(register int i=0;i<10;++i)
if(tr[0].son[i])
q.push(tr[0].son[i]);
while(!q.empty())
{
int u=q.front();
q.pop();
tr[u].cnt+=tr[tr[u].fail].cnt;
tr[u].val+=tr[tr[u].fail].val;
for(register int i=0;i<10;++i)
{
if(tr[u].son[i])
{
tr[tr[u].son[i]].fail=tr[tr[u].fail].son[i];
q.push(tr[u].son[i]);
}
else
tr[u].son[i]=tr[tr[u].fail].son[i];
}
}
}
inline void updateans(register int i,register int j)
{
if(!i)
return;
updateans(i-1,pre[i][j].first);
ans[i]=pre[i][j].second+'0';
}
inline bool check(register db mid)
{
memset(f,-0x3f,sizeof(f));
db inf=-f[0][0];
f[0][0]=0;
for(register int i=1;i<=n;++i)
for(register int j=0;j<=tot;++j)
{
if(fabs(f[i-1][j]+inf)<1)
continue;
if(T[i]=='.')
{
for(register int k=0;k<10;++k)
{
int v=tr[j].son[k];
if(f[i-1][j]+tr[v].val-tr[v].cnt*mid>f[i][v])
{
f[i][v]=f[i-1][j]+tr[v].val-tr[v].cnt*mid;
pre[i][v]=make_pair(j,k);
}
}
}
else
{
int k=T[i]-'0',v=tr[j].son[k];
if(f[i-1][j]+tr[v].val-tr[v].cnt*mid>f[i][v])
{
f[i][v]=f[i-1][j]+tr[v].val-tr[v].cnt*mid;
pre[i][v]=make_pair(j,k);
}
}
}
int pos=0;
for(register int i=1;i<=tot;++i)
if(f[n][i]>f[n][pos])
pos=i;
if(f[n][pos]>eps)
{
updateans(n,pos);
return 1;
}
else
return 0;
}
int main()
{
n=read(),m=read();
scanf("%s",T+1);
db L=0,R=0;
for(register int i=1;i<=m;++i)
{
scanf("%s",s+1);
V[i]=log(read());
R=max(R,V[i]);
Insert(s,V[i]);
}
getfail();
while(R-L>eps)
{
db mid=(L+R)/2.0;
if(check(mid))
L=mid;
else
R=mid;
}
for(register int i=1;i<=n;++i)
putchar(ans[i]);
return 0;
}

【题解】Luogu P5319 [BJOI2019]奥术神杖的更多相关文章

  1. luogu P5319 [BJOI2019]奥术神杖

    传送门 要求的东西带个根号,这玩意叫几何平均数,说到平均数,我们就能想到算术平均数(就是一般意义下的平均数),而这个东西是一堆数之积开根号,所以如果每个数取对数,那么乘法会变成加法,开根号变成除法,所 ...

  2. luoguP5319 [BJOI2019]奥术神杖(分数规划,AC自动机DP)

    luoguP5319 [BJOI2019]奥术神杖(分数规划,AC自动机DP) Luogu 题解时间 难点在于式子转化,设有c个满足的子串,即求最大的 $ ans = \sqrt[c]{\prod_{ ...

  3. [BJOI2019]奥术神杖(分数规划,动态规划,AC自动机)

    [BJOI2019]奥术神杖(分数规划,动态规划,AC自动机) 题面 洛谷 题解 首先乘法取\(log\)变加法,开\(c\)次根变成除\(c\). 于是问题等价于最大化\(\displaystyle ...

  4. [BJOI2019]奥术神杖——AC自动机+DP+分数规划+二分答案

    题目链接: [BJOI2019]奥术神杖 答案是$ans=\sqrt[c]{\prod_{i=1}^{c}v_{i}}=(\prod_{i=1}^{c}v_{i})^{\frac{1}{c}}$. 这 ...

  5. 题解 [BJOI2019]奥术神杖

    题目传送门 题目大意 给出一个残缺的字符串,每个位置都 \(\in[0,9]\).有 \(m\) 中贡献,即 \(s,k\),表示该字符串中没出现一次 \(s\),贡献便乘上 \(k\).最后对贡献求 ...

  6. [BJOI2019]奥术神杖

    https://www.luogu.org/problemnew/show/P5319 题解 首先观察我们要求的答案的形式: \[ \biggl(\prod V_i \biggr)^x\ \ \ x= ...

  7. [BJOI2019]奥术神杖(分数规划+AC自动机+DP)

    题解:很显然可以对权值取对数,然后把几何平均值转为算术平均值,然后很显然是分数规划.先对每个模式串建立AC自动机,每个节点w[i],sz[i]分别表示以其为前缀的字符串,然后再二分最优解k,然后w[i ...

  8. #loj3089 [BJOI2019]奥术神杖

    卡精度好题 最关键的一步是几何平均数的\(ln\)等于所有数字取\(ln\)后的算术平均值 那么现在就变成了一个很裸的01分数规划问题,一个通用的思路就是二分答案 现在来考虑二分答案的底层怎么写 把所 ...

  9. [BJOI2019]奥术神杖(AC自动机,DP,分数规划)

    题目大意: 给出一个长度 $n$ 的字符串 $T$,只由数字和点组成.你可以把每个点替换成一个任意的数字.再给出 $m$ 个数字串 $S_i$,第 $i$ 个权值为 $t_i$. 对于一个替换方案,这 ...

随机推荐

  1. idea使用过程中的一些常见问题,做个笔记

    :当实现这个接口方法时重载是不允许的. 首先我相信我的代码肯定没问题,因为我实现的接口确实有这个方法.在编程阶段就提示这个错误,于是我有理由相信应该是编译错误!通过google,解决办法so easy ...

  2. TCP链接异常断开后,对端仍然ESTABLISH

    双方建立TCP链接,其中一方拔掉网线,另一端依然是ESTABLISHED,那么要过多长时间才会发觉链接被断开了呢? [root@node1 ~]# sysctl -a |grep keepalive ...

  3. Eclipse properties配置文件中文乱码设置

    1. eclipse中properties的默认编码为  ISO-8859-1, 输入汉字会被转换为unicode 2. 点击  Windows-->preferences  按下图找到更改编码 ...

  4. Linux的DNS正向解析部署

    前面介绍了DNS的作用及其相关的结果.Linux服务之DNS介绍 下面开始有关DNS的服务部署.<DNS正向解析示例> 工具:虚拟机 centos7 配置:Linux   IP 192.1 ...

  5. maven仓库报错 sqljdbc4、ojdbc6、tomcat-jdbc-8.5.14

    报错:Cannot resolve com.microsoft.sqlserver:sqljdbc4:4.0  和  Missing artifact com.microsoft.sqlserver: ...

  6. Nginx 核心配置-长连接配置

    Nginx 核心配置-长连接配置 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.长连接配置参数说明 keepalive_timeout number; 设定保持连接超时时长,0 ...

  7. 子网掩码的作用与IP网段的划分

    公有IP地址分类 A类:1.0.0.0 到 127.255.255.255主要分配 给大量主机而局域网网络数量较少的大型网络 B类:128.0.0.0 到191.255.255.255 一般用于国际性 ...

  8. asp.net core mvc基于Redis实现分布式锁,C# WebApi接口防止高并发重复请求,分布式锁的接口幂等性实现

    使用背景:在使用app或者pc网页时,可能由于网络原因,api接口可能被前端调用一个接口重复2次的情况,但是请求内容是一样的.这样在同一个短暂的时间内,就会有两个相同请求,而程序只希望处理第一个请求, ...

  9. tensorflow运行时错误:服务似乎挂掉了,但是会立刻重启的.

    以前在POD里跑起来,没问题的示例代码. 移到jupyter中,多给两个GPU,有时运行就会爆出这个错误: 于是,按网上的意见,暂时加了个使用GPU的指定, 暂时搞定. 如下红色部分. import ...

  10. php状态模式(state pattern)

    ... <?php /* The state pattern encapsulates the varying behavior for the same object based on its ...