BZOJ:4820: [Sdoi2017]硬币游戏&&BZOJ:1444: [Jsoi2009]有趣的游戏(高斯消元求概率)
1444: [Jsoi2009]有趣的游戏
4820: [Sdoi2017]硬币游戏
这两道题都是关于不断随机生成字符后求出现给定字符串的概率的问题。
第一题数据范围较小,将串建成AC自动机以后,以AC自动机上每个点为一个未知数,列出方程高斯消元求解即可,时间复杂度$O(n^{3}m^{3})$。
#include<queue>
#include<cstdio>
#include<algorithm>
#define MN 21
#define ld double
#define eps 1e-12
using namespace std; struct na{int t[],bo,f;na(){bo=;}}t[MN*MN];
int n,m,l,mo[MN],num=,p[MN],q[MN];
char s[];
int insert(char s[]){
int i=,j=;
for (;s[i];j=t[j].t[s[i]-'A'],i++)
if (!t[j].t[s[i]-'A']) t[j].t[s[i]-'A']=++num;
t[j].bo=;
for (i=;s[i];i++) if (p[s[i]-'A']==) return -;
return j;
}
queue<int> _q;
ld a[MN*MN][MN*MN],S;
ld abs(ld a){return a<?-a:a;}
inline void Gauss(){
int i,j,k;
for (i=;i<=num;i++){
for (j=i;j<=num;j++) if (abs(a[j][i])>eps) break;
for (k=;k<=num+;k++) swap(a[i][k],a[j][k]);
for (j=;j<=num;j++)
if (j!=i)
for (S=a[j][i]/a[i][i],k=;k<=num+;k++) a[j][k]-=S*a[i][k];
}
}
int main(){
register int i,j;
scanf("%d%d%d",&n,&l,&m);
for (i=;i<m;i++) scanf("%d%d",&p[i],&q[i]);
for (i=;i<n;i++) scanf("%s",s),mo[i]=insert(s);
t[].f=;
for (i=;i<m;i++) if (t[].t[i]) _q.push(t[].t[i]),t[t[].t[i]].f=;
while (!_q.empty()){
int k=_q.front();_q.pop();
for (i=;i<m;i++)
if (t[k].t[i]){
for (j=t[k].f;j&&!t[j].t[i];j=t[j].f);
t[t[k].t[i]].f=t[j].t[i];
_q.push(t[k].t[i]);
}
}
for (int k=;k<=num;a[k][k]-=.,k++)
if (!t[k].bo)
for (i=;i<m;i++){
for (j=k;j&&!t[j].t[i];j=t[j].f);
j=t[j].t[i];
a[j][k]+=.*p[i]/q[i];
}
a[][num+]=-;
Gauss();
for (i=;i<n;i++) if (mo[i]==-) puts("0.00");else printf("%.2lf\n",a[mo[i]][num+]/a[mo[i]][mo[i]]+eps);
}
1444: [Jsoi2009]有趣的游戏
第二题,数据范围较大,直接建AC自动机必定会TLE,所以考虑化简。我们需要的只是n个目标状态的答案,而不需要AC自动机上其他点的答案,那么这些点是否可以合并起来,统一考虑呢?
以样例为例:
$S_1=THT,S_2=TTH,S_3=HTT$
考虑一个状态X,表示所有未达到目标状态的字符串。
那么若在X后面接上一个$S_1$,肯定就到达了目标状态,也可能我们还没填完$S_1$我们就达成了目标状态,这些一起考虑起来,可以得到
$$\frac{X}{2^{3}}=S_1+\frac{S_1}{2^{2}}+\frac{S_2}{2}+\frac{S_3}{2^{2}}$$
这是什么意思?XTHT有可能是以下几种情况(我们用$F_{S_i}$表示$S_i$的出现概率)
XTHT =$F_{S_1}$
YTHT HT=$F_{S_1}$ HT(X以TH结尾,即X=YTH)
YTTH T=$F_{S_2}$ T(X以T结尾,即X=YT)
YHTT HT=$F_{S_2}$ HT(X以HT结尾,即X=YHT)
额外多出来的x个字符就需要花费$\frac{1}{2^{x}}$的概率去生成它,也就很显然地对应了上面的式子。
再考虑在X后面接上$S_2$和$S_3$我们又得到两个方程,再加上$\sumF_{S_i}$,一共4(n+1)个方程,解3(n)个变量刚刚好。
另外如果高斯消元的时候你是用的eps来找第一个非0位置,那么很有可能爆精度,这时候你需要将其修改为找到绝对值最大的位置(虽然不知道为什么这样就可以了)。
#include<queue>
#include<cstdio>
#include<algorithm>
#define MN 321
#define ld double
#define eps 1e-12
using namespace std; int n,m,l,mo[MN],num=,ne[MN<<];
char s[MN][MN],c[MN<<];
ld a[MN][MN],S,two[MN];
ld abs(ld a){return a<?-a:a;}
inline void Gauss(){
int i,j,k;
for (i=;i<=num;i++){
for (j=i;j<=num;j++) if (abs(a[j][i])>eps) break;
for (k=;k<=num+;k++) swap(a[i][k],a[j][k]);
for (j=;j<=num;j++)
if (j!=i)
for (S=a[j][i]/a[i][i],k=;k<=num+;k++) a[j][k]-=S*a[i][k]; }
}
int main(){
register int i,j,k,l;
scanf("%d%d",&n,&m);
for (i=;i<=n;i++) scanf("%s",s[i]); for (two[]=,i=;i<=m;i++) two[i]=two[i-]*0.5;
for (i=;i<=n;i++)
for (a[i][]=-two[m],j=;j<=n;j++){
for (k=;k<m;k++) c[k]=s[i][k],c[m+m-k-]=s[j][m-k-];c[m+m]=;
ne[]=-;
for (k=;k<m+m;ne[k]=l+(c[l+]==c[k]),k++)
for (l=ne[k-];l!=-&&c[l+]!=c[k];l=ne[l]);
for (k=ne[k-];k!=-;k=ne[k])
if (k<m) a[i][j]+=two[m--k];
} for (i=;i<=n+;i++) a[][i]=;
num=n;Gauss();
for (i=;i<=n;i++) printf("%.8lf\n",a[i][n+]/a[i][i]+eps);
}
4820: [Sdoi2017]硬币游戏
BZOJ:4820: [Sdoi2017]硬币游戏&&BZOJ:1444: [Jsoi2009]有趣的游戏(高斯消元求概率)的更多相关文章
- [BZOJ 4820] [SDOI2017] 硬币游戏(高斯消元+概率论+字符串hash)
[BZOJ 4820] [SDOI2017] 硬币游戏(高斯消元+概率论+字符串hash) 题面 扔很多次硬币后,用H表示正面朝上,用T表示反面朝上,会得到一个硬币序列.比如HTT表示第一次正面朝上, ...
- BZOJ 1444:[JSOI2009]有趣的游戏
BZOJ 1444:[JSOI2009]有趣的游戏 题目链接 首先我们建出Trie图,然后高斯消元. 我们设\(f_i\)表示经过第\(i\)个点的期望次数: \[ f_x=\sum i\cdot p ...
- BZOJ 1444: [Jsoi2009]有趣的游戏 [AC自动机 高斯消元]
1444: [Jsoi2009]有趣的游戏 题意:每种字母出现概率\(p_i\),有一些长度len的字符串,求他们出现的概率 套路DP的话,\(f[i][j]\) i个字符走到节点j的概率,建出转移矩 ...
- BZOJ 1444 [Jsoi2009]有趣的游戏 (AC自动机 + 概率DP + Gauss)
1444: [Jsoi2009]有趣的游戏 Time Limit: 10 Sec Memory Limit: 64 MBSubmit: 1382 Solved: 498[Submit][Statu ...
- 1444: [Jsoi2009]有趣的游戏
1444: [Jsoi2009]有趣的游戏 链接 分析: 如果一个点回到0号点,那么会使0号点的概率增加,而0号点的概率本来是1,不能增加,所以这题用期望做. 设$x_i$表示经过i的期望次数,然后初 ...
- 【bzoj3105】[cqoi2013]新Nim游戏 高斯消元求线性基
题目描述 传统的Nim游戏是这样的:有一些火柴堆,每堆都有若干根火柴(不同堆的火柴数量可以不同).两个游戏者轮流操作,每次可以选一个火柴堆拿走若干根火柴.可以只拿一根,也可以拿走整堆火柴,但不能同时从 ...
- BZOJ.4820.[SDOI2017]硬币游戏(思路 高斯消元 哈希/AC自动机/KMP)
BZOJ 洛谷 建出AC自动机,每个点向两个儿子连边,可以得到一张有向图.参照 [SDOI2012]走迷宫 可以得到一个\(Tarjan\)+高斯消元的\(O((nm)^3)\)的做法.(理论有\(6 ...
- BZOJ 4820 [SDOI2017] 硬币游戏
Description 周末同学们非常无聊,有人提议,咱们扔硬币玩吧,谁扔的硬币正面次数多谁胜利.大家纷纷觉得这个游戏非常符合同学们的特色,但只是扔硬币实在是太单调了.同学们觉得要加强趣味性,所以要找 ...
- ●BZOJ 1444 [Jsoi2009]有趣的游戏
题链: http://www.lydsy.com/JudgeOnline/problem.php?id=1444题解.1: 概率dp,矩阵乘法,快速幂. 对所有串建立AC自动机, 那么如果在trie树 ...
随机推荐
- 【CSS3】内联、内部、外部样式,样式优先级、层叠、继承
<!DOCTYPE html> <html> <head> <meta charset="utf-8"> <title> ...
- XMPP协议的基本理解
即时通讯技术简介 即时通讯技术(IM)支持用户在线实时交谈.如果要发送一条信息,用户需要打开一个小窗口,以便让用户及其朋友在其中输入信息并让交谈双方都看到交谈的内容.大多数常用的即时通讯发送程序都会提 ...
- Java多线程之赛跑游戏
在JavaSE中,多线程是一个重要的内容. 我们要了解多线程的概念,就要先了解进程的概念:要了解进程的概念,就离不开操作系统的概念. 在一台正常运行的电脑中,计算机硬件(如CPU.内存.硬盘.网卡.显 ...
- 解题思路:best time to buy and sell stock i && ii && iii
这三道题都是同一个背景下的变形:给定一个数组,数组里的值表示当日的股票价格,问你如何通过爱情买卖来发家致富? best time to buy and sell stock i: 最多允许买卖一次 b ...
- Java 本地开发环境搭建(框架采用 Spring+Spring MVC+Hibernate+Jsp+Gradle+tomcat+mysql5.6)
项目搭建采用技术栈为:Spring+Spring MVC+Hibernate+Jsp+Gradle+tomcat+mysql5.6 搭建环境文档目录结构说明: 使用Intellj Idea 搭建项目过 ...
- [置顶]
几行代码实现ofo首页小黄人眼睛加速感应转动
最新版的ofo 小黄车的首页小黄人眼睛随重力而转动,感觉有点炫酷,学习一下吧,以下代码是在xamarin android下实现 ofo首页效果图: xamarin android实现效果: 实现思路: ...
- 封装数据库配置文件App配置文件
<connectionStrings> <add name="strCon" connectionString="Data Source=.;Ini ...
- scrapy框架第一章
操作环境:python2.7+scrapy 安装比较简单,网上教程也超多,就不在此赘述. 示例网站:https://www.cnblogs.com/cate/python/ (爬去关于博客园所有pyt ...
- composer安装laravel
安装composer composer是一个很有用的工具,我将用它在本机(win7)上安装laravel 到composer的官网,根据自己的系统要求下载相应的版本 安装laravel 首先cmd下进 ...
- 教育改革——国家认证 “网红” 编程语言 Python
特大消息!!! 不止是上海计算机二级考试 ,全国计算机考试等级考试也有要求 如果你正打算考计算机等级,那你需要学习以下知识 要求学习的知识太多了,我就不一一在这里展示了! 一.考试改革的目标 据悉 ...