Loj #3089. 「BJOI2019」奥术神杖
Loj #3089. 「BJOI2019」奥术神杖
题目描述
Bezorath 大陆抵抗地灾军团入侵的战争进入了僵持的阶段,世世代代生活在 Bezorath 这片大陆的精灵们开始寻找远古时代诸神遗留的神器,试图借助神器的神秘
力量帮助她们战胜地灾军团。
在付出了惨痛的代价后,精灵们从步步凶险的远古战场取回了一件保存尚完好的神杖。但在经历过那场所有史书都视为禁忌的“诸神黄昏之战”后,神杖上镶嵌的奥术宝石
已经残缺,神力也几乎消耗殆尽。精灵高层在至高会议中决定以举国之力收集残存至今的奥术宝石,并重金悬赏天下能工巧匠修复这件神杖。
你作为神术一脉第五百零一位传人,接受了这个艰巨而神圣的使命。 神杖上从左到右镶嵌了 \(n\) 颗奥术宝石,奥术宝石一共有 \(10\) 种,用数字 0123456789
表示。有些位置的宝石已经残缺,用 . 表示,你需要用完好的奥术宝石填补每一处残缺的部分(每种奥术宝石个数不限,且不能够更换未残缺的宝石)。古老的魔法
书上记载了 \(m\) 种咒语 \((S_i,V_i)\),其中 \(S_i\) 是一个非空数字串,\(V_i\) 是这种组合能够激发的神力。
神杖的初始神力值 \(\mathrm{Magic} = 1\),每当神杖中出现了连续一段宝石与 \(S_i\) 相等时,神力值 \(\mathrm{Magic}\) 就会乘以 \(V_i\)。但神杖如果包含
了太多咒语就不再纯净导致神力降低:设 \(c\) 为神杖包含的咒语个数(若咒语类别相同但出现位置不同视为多次),神杖最终的神力值为 \(\sqrt[c]{\mathrm{Magic}}\)。(若 \(c = 0\) 则神杖最终神力值为 \(1\)。)
例如有两种咒语 \((01,3)\) 、\((10,4)\),那么神杖 0101 的神力值为 \(\sqrt[3]{ 3 \times 4 \times 3}\)。
输入格式
第一行两个正整数 \(n,m\),表示宝石数和咒语数。
第二行为一个长度为 \(n\) 的字符串 \(T\),表示初始的神杖。
接下来 \(m\) 行每行一个非空数字串 \(S_i\) 和一个正整数 \(V_i\),表示每种咒语。
输出格式
输出最终神杖上从左到右镶嵌的宝石,多解时任意输出一个即可。
数据范围与提示
\(n,\sum_{i=1}^m|S_i|\leq 150\)。\(V_i\leq 10^9\)
\(\\\)
首先将答案取一个对数,
\log_2^{\sqrt[c]{\prod_{i=1}^c V_i}}=\frac{\sum_{i=1}^c\log_2^{V_i}}{c}
\]
要求最大化\(\frac{\sum_{i=1}^c V_i}{c}\)就是裸的\(0/1\)分数规划问题(我竟然没看出来,真是越来越zz了)。具体实现时在\(AC\)自动机上\(DP\)就好了。
代码:
#include<bits/stdc++.h>
#define ll long long
#define N 1505
#define eps 1e-8
using namespace std;
inline int Get() {int x=0,f=1;char ch=getchar();while(ch<'0'||ch>'9') {if(ch=='-') f=-1;ch=getchar();}while('0'<=ch&&ch<='9') {x=(x<<1)+(x<<3)+ch-'0';ch=getchar();}return x*f;}
int n,m;
char s[N],t[N];
struct trie {
int ch[10];
int cnt,fail;
double sum;
}tr[N];
int cnt=1;
void Insert(char *s,double val) {
int len=strlen(s+1);
int v=1;
for(int i=1;i<=len;i++) {
int j=s[i]-'0';
if(!tr[v].ch[j]) tr[v].ch[j]=++cnt;
v=tr[v].ch[j];
}
tr[v].cnt++;
tr[v].sum+=val;
}
void build_fail() {
static queue<int>q;
for(int i=0;i<10;i++) {
if(!tr[1].ch[i]) tr[1].ch[i]=1;
else {
tr[tr[1].ch[i]].fail=1;
q.push(tr[1].ch[i]);
}
}
while(!q.empty()) {
int v=q.front();
q.pop();
tr[v].cnt+=tr[tr[v].fail].cnt;
tr[v].sum+=tr[tr[v].fail].sum;
for(int i=0;i<10;i++) {
if(!tr[v].ch[i]) tr[v].ch[i]=tr[tr[v].fail].ch[i];
else {
int sn=tr[v].ch[i];
tr[sn].fail=tr[tr[v].fail].ch[i];
q.push(sn);
}
}
}
}
double f[N][N];
double tag[N];
struct node {
int x,y,type;
node() {}
node(int _x,int _y,int _type) {
x=_x,y=_y,type=_type;
}
};
node fr[N][N];
bool chk(double ans) {
for(int i=1;i<=cnt;i++) tag[i]=tr[i].sum-tr[i].cnt*ans;
for(int i=0;i<=n;i++)
for(int j=1;j<=cnt;j++)
f[i][j]=-1e9;
f[0][1]=0;
for(int i=0;i<n;i++) {
for(int j=1;j<=cnt;j++) {
if(f[i][j]<-1e6) continue ;
if(s[i+1]=='.') {
for(int k=0;k<10;k++) {
int q=tr[j].ch[k];
if(f[i][j]+tag[q]>f[i+1][q]) {
f[i+1][q]=f[i][j]+tag[q];
fr[i+1][q]=node(i,j,k);
}
}
} else {
int q=tr[j].ch[s[i+1]-'0'];
if(f[i][j]+tag[q]>f[i+1][q]) {
f[i+1][q]=f[i][j]+tag[q];
fr[i+1][q]=node(i,j,s[i+1]-'0');
}
}
}
}
for(int i=1;i<=cnt;i++) if(f[n][i]>0) return 1;
return 0;
}
void out(int x,int y) {
if(!x) return ;
out(fr[x][y].x,fr[x][y].y);
cout<<fr[x][y].type;
}
int main() {
n=Get(),m=Get();
scanf("%s",s+1);
for(int i=1;i<=m;i++) {
scanf("%s",t+1);
double v=Get();
Insert(t,log2(v));
}
build_fail();
double l=0,r=100,mid;
while(l+eps<r) {
mid=(l+r)/2.0;
if(chk(mid)) l=mid;
else r=mid-eps;
}
chk(l);
for(int i=1;i<=cnt;i++) {
if(f[n][i]>0) {
out(n,i);
break ;
}
}
return 0;
}
Loj #3089. 「BJOI2019」奥术神杖的更多相关文章
- LOJ 3089 「BJOI2019」奥术神杖——AC自动机DP+0/1分数规划
题目:https://loj.ac/problem/3089 没想到把根号之类的求对数变成算数平均值.写了个只能得15分的暴力. #include<cstdio> #include< ...
- 【LOJ】#3089. 「BJOI2019」奥术神杖
LOJ#3089. 「BJOI2019」奥术神杖 看见乘积就取log,开根号就是除法,很容易发现这就是一道01分数规划.. 然后建出AC自动机直接dp就行,判断条件要设成>0,因为起点的值是1, ...
- LOJ 3089: 洛谷 P5319: 「BJOI2019」奥术神杖
题目传送门:LOJ #3089. 题意简述: 有一个长度为 \(n\) 的母串,其中某些位置已固定,另一些位置可以任意填. 同时给定 \(m\) 个小串,第 \(i\) 个为 \(S_i\),所有位置 ...
- Loj #3093. 「BJOI2019」光线
Loj #3093. 「BJOI2019」光线 题目描述 当一束光打到一层玻璃上时,有一定比例的光会穿过这层玻璃,一定比例的光会被反射回去,剩下的光被玻璃吸收. 设对于任意 \(x\),有 \(x\t ...
- loj 3090 「BJOI2019」勘破神机 - 数学
题目传送门 传送门 题目大意 设$F_{n}$表示用$1\times 2$的骨牌填$2\times n$的网格的方案数,设$G_{n}$$表示用$1\times 2$的骨牌填$3\times n$的网 ...
- LOJ 3094 「BJOI2019」删数——角标偏移的线段树
题目:https://loj.ac/problem/3094 弱化版是 AGC017C . 用线段树维护那个题里的序列即可. 对应关系大概是: 真实值的范围是 [ 1-m , n+m ] :考虑设偏移 ...
- LOJ 3090 「BJOI2019」勘破神机——斯特林数+递推式求通项+扩域
题目:https://loj.ac/problem/3090 题解:https://www.luogu.org/blog/rqy/solution-p5320 1.用斯特林数把下降幂化为普通的幂次求和 ...
- LOJ 3093 「BJOI2019」光线——数学+思路
题目:https://loj.ac/problem/3093 考虑经过种种反射,最终射下去的光线总和.往下的光线就是这个总和 * a[ i ] . 比如只有两层的话,设射到第二层的光线是 lst ,那 ...
- LOJ 3092 「BJOI2019」排兵布阵 ——DP
题目:https://loj.ac/problem/3092 同一个人的不同城堡之间没有什么联系,只是和<=m.所以对每个城堡的 s 个值排序,做一个 f[ i ][ j ] 表示第 i 个城堡 ...
随机推荐
- Tomcat设置HTTPS访问
根据查阅,选择了最方便的一种方法. 简单来说有两步,1:新生成一个密钥库keystore 2:配置 Tomcat 以使用该 keystore 文件.此时https可以访问,但会有证书不信任问题,介意的 ...
- LOJ #6060. 「2017 山东一轮集训 Day1 / SDWC2018 Day1」Set
有趣的思博套路题,想到了基本上加上个对线性基的理解就可以过了 首先考虑到这个把数分成两半的分别异或的过程不会改变某一位上\(1\)的总个数 因此我们求出所有数的\(\operatorname{xor} ...
- 开源的类似于Apache ab的压力测试命令行工具SuperBenchmarker
SuperBenchmarker 是ㄧ个开源的类似于Apache ab的压力测试命令行工具.可以在 .NET 4.52+ 或者 .NET Core 2.0+ 平台上运行. 可支持Get.Post.Pu ...
- JAVA之enum类详解
目录 一.简介 二.默认枚举类 三.多值枚举类 四.属性和方法 五.构造函数 六.重要方法 七.引用参考 一.简介 1.枚举类代表一组常量: 2. ...
- Yii2 解决2006 MySQL server has gone away问题
Yii2 解决2006 MySQL server has gone away问题 Yii2版本 2.0.15.1 php后台任务经常包含多段sql,如果php脚本执行时间较长,或者sql执行时间较长, ...
- Linux知识要点大全(第二章)
第二章 linux操作系统安装与配置主要内容 1:vmware虚拟机安装与使用 2:Linux系统安装前准备 3:Linux Centos 系统的安装 4:Centos 6.8的登录和关闭 5:C ...
- 【Netty】(4)—源码AbstractBootstrap
源码AbstractBootstrap 一.概念 AbstractBootstrap是一个工具类,用于服务器通道的一系列配置,绑定NioEventLoopGroup线程组,指定指定NIO的模式,指定子 ...
- 【Java】留下没有基础眼泪的面试题
前言 只有光头才能变强 本文力求简单讲清每个知识点,希望大家看完能有所收获 一.如何减少线程上下文切换 使用多线程时,不是多线程能提升程序的执行速度,使用多线程是为了更好地利用CPU资源! 程序在执行 ...
- Nginx反向代理和Node.js后端解决跨域问题
最近在写自己的博客,涉及到跨域的问题,自己捣鼓许久,终于解决了.然后总结一下,记录一下,日后遇到类似的问题的时候也可以得到一些启发. 一.什么是跨域 跨域,指的是浏览器不能执行其他网站的脚本.它是由浏 ...
- 基础设施DevOps演进之路
Related Links:Zuul https://github.com/Netflix/zuulCAT https://github.com/dianping/cat Apollo h ...