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 个城堡 ...
随机推荐
- 使用 certbot 申请泛域名https证书
使用 certbot 申请泛域名https证书 Intro Certbot 是一个基于 Let's Encrypt 的自动化申请证书的工具,支持的系统和web server也很多,详见 Certbot ...
- .NET 基金会完成第一次全面改选
.NET基金会是一个独立的组织,支持.NET社区和开源,旨在拓宽和加强.NET生态系统和社区.这可以通过多种方式完成,包括项目指导,指导,法律和营销帮助,技术和财务支持设置等,2014年微软组织成立. ...
- gulp源码解析(一)—— Stream详解
作为前端,我们常常会和 Stream 有着频繁的接触.比如使用 gulp 对项目进行构建的时候,我们会使用 gulp.src 接口将匹配到的文件转为 stream(流)的形式,再通过 .pipe() ...
- Quartz简单实现定时任务管理(SSM+Quartz)
首先你得有一个用Maven搭好的SSM框架,数据库用的Mysql,这里只有关于Quartz的部分.其实有大神总结的很好了,但做完后总有些地方不一样,所以写这篇作为笔记.这里先把大神的写的分享给大家:h ...
- 使用 docker-compose 快速安装Jenkins
本文分享在 docker 环境中,使用 docker-compose.yml 快速安装 Jenkins,以及使用主机中的 docker 打包推送镜像到阿里云 博客园的第100篇文章达成,2019的第一 ...
- 286万QPS!腾讯云TDSQL打造数据库领域的“超音速战机”
Bloodhound SSC超音速汽车将陆地极限速度提升到1678公里/小时,号称陆地“超音速战斗机”.无独有偶,同样也在2017年,在英特尔®.腾讯金融云团队的共同见证下,腾讯云数据库TDSQL采用 ...
- OAuth2实现单点登录SSO
1. 前言 技术这东西吧,看别人写的好像很简单似的,到自己去写的时候就各种问题,“一看就会,一做就错”.网上关于实现SSO的文章一大堆,但是当你真的照着写的时候就会发现根本不是那么回事儿,简直让人抓 ...
- WindowUtils【窗口工具类】
版权声明:本文为HaiyuKing原创文章,转载请注明出处! 前言 判断当前界面是横屏还是竖屏: 获取当前界面方向. 效果图 代码分析 isLandscape(Context context): ...
- NavUtils【底部虚拟导航栏工具类】
版权声明:本文为HaiyuKing原创文章,转载请注明出处! 前言 获取底部虚拟导航栏的高度值 效果图 代码分析 checkDeviceHasNavigationBar(Context context ...
- 我的那些年(9)~我来团队了,Mvc兴起了
回到目录 我的那些年(9)~我来团队了,Mvc兴起了 在一次后出办事后直接去面试了 面试就是答卷子 六里桥一个好地址 搬回老家了 在老婆的建议下学驾照了 拿到大专毕业证了 买车了 愉一切可以愉的时间学 ...