●BZOJ 3640 JC的小苹果
题链:
http://www.lydsy.com/JudgeOnline/problem.php?id=3640
题解:
期望dp,高斯消元
设dp[i][h]在i位置且血量为h这个状态的期望经过次数。
因为每当到达n点就停止游戏,所以到达终点的概率就是dp[n][1]+dp[n][2]+...+dp[n][hp]
可以按血量把dp分成若干个层次,我们希望这样分层次后就可以把问题转变为DAG上的dp,
可是存在伤害值为0的点,所以我们对于每一层列出来的n的dp计算式,是可能存在环的。
所以要用高斯消元。复杂度(hp*N^3)
注意到在每个层次做高斯消元时,其实这些方程的系数都相同,
(规定一下方程的形式:a1*x1+a2*x2+a3*x3+...+an*xn=c,a都为系数,x为n个未知数,c为常数项)
所以我们可以预处理记录下高斯消元是怎么消的,或者说,
因为每个未知数一定是由n个常数项线性组合起来的,所以我们记录一下每个未知数是如何由n个方程的常数项构成的
具体做法是把n个常数项看成一个1×n的X矩阵,然后把每个未知数xi分别也看成1×n的一个矩阵Yi
然后如果知道了常数项的矩阵,就可以由Yi*X的倒置矩阵得到一个1×1的矩阵,而里面存的值就是xi的值。
所以高斯消元就是去求出这n个Yi矩阵。
复杂度:O(hp*n^2+n^3)
更详细的膜这里:https://blog.sengxian.com/solutions/bzoj-3640
代码:
#include<bits/stdc++.h>
#define MAXN 160
using namespace std;
const double eps=1e-12;
double ANS,a[MAXN][MAXN],*A[MAXN],dp[MAXN][10005];
int damage[MAXN],cnt[MAXN];
int N,M,HP;
int dcmp(double x){
if(fabs(x)<eps) return 0;
return x>0?1:-1;
}
struct Edge{
int ent;
int to[MAXN*MAXN],nxt[MAXN*MAXN],head[MAXN];
Edge():ent(2){}
void Adde(int u,int v){
to[ent]=v; nxt[ent]=head[u]; head[u]=ent++;
}
}E;
struct Matrix{
int r,c;
double a[2][MAXN];
void Reset(int _r,int _c){
r=_r; c=_c;
memset(a,0,sizeof(a));
}
Matrix operator - () const{
Matrix now; now.Reset(r,c);
for(int i=1;i<=now.r;i++)
for(int j=1;j<=now.c;j++)
now.a[i][j]=-a[i][j];
return now;
}
Matrix operator + (const Matrix &rtm) const{
Matrix now; now.Reset(r,c);
for(int i=1;i<=now.r;i++)
for(int j=1;j<=now.c;j++)
now.a[i][j]=a[i][j]+rtm.a[i][j];
return now;
}
Matrix operator - (const Matrix &rtm) const{
return *this+(-rtm);
}
Matrix operator * (const double k) const{
Matrix now; now.Reset(r,c);
for(int i=1;i<=now.r;i++)
for(int j=1;j<=now.c;j++)
now.a[i][j]=a[i][j]*k;
return now;
}
Matrix operator & (const Matrix &rtm) const{ //乘上rtm的倒置矩阵
Matrix now; now.Reset(r,rtm.r);
for(int i=1;i<=now.r;i++)
for(int j=1;j<=now.c;j++)
for(int k=1;k<=c;k++)
now.a[i][j]+=a[i][k]*rtm.a[j][k];
return now;
}
Matrix operator / (const double k) const{
return *this*(1/k);
}
}B[MAXN],c[MAXN],X;
Matrix *C[MAXN];
void buildequation(){
for(int i=1;i<=N;i++){
a[i][i]=-1; if(damage[i]) continue;
for(int e=E.head[i];e;e=E.nxt[e]){
int j=E.to[e]; if(j==N) continue;
a[i][j]+=1.0/cnt[j];
}
}
for(int i=1;i<=N;i++){
B[i].Reset(1,N); A[i]=a[i];
c[i].Reset(1,N); c[i].a[1][i]=1;
C[i]=&c[i];
}
}
void Gausselimination(int pos,int i){
if(pos==N+1||i==N+1) return;
for(int j=pos;j<=N;j++) if(dcmp(A[j][i])!=0){
swap(A[pos],A[j]); swap(C[pos],C[j]); break;
}
if(dcmp(A[pos][i])!=0)
for(int j=pos+1;j<=N;j++){
double k=A[j][i]/A[pos][i];
for(int l=i;l<=N;l++)
A[j][l]-=A[pos][l]*k;
(*C[j])=(*C[j])-(*C[pos])*k;
}
Gausselimination(pos+(dcmp(A[pos][i])!=0),i+1);
if(dcmp(A[pos][i])!=0){
for(int l=i+1;l<=N;l++)
B[i]=B[i]+B[l]*A[pos][l];
B[i]=(*C[pos])-B[i];
B[i]=B[i]/A[pos][i];
}
}
int main(){
ios::sync_with_stdio(0);
cin>>N>>M>>HP;
for(int i=1;i<=N;i++) cin>>damage[i];
for(int i=1,u,v;i<=M;i++){
cin>>u>>v;
E.Adde(u,v),cnt[u]++;
if(u!=v) E.Adde(v,u),cnt[v]++;
}
buildequation();
Gausselimination(1,1);
for(int h=HP;h>=1;h--){
X.Reset(1,N);
for(int i=1;i<=N;i++){
if(!damage[i]) continue;
if(h+damage[i]>HP) continue;
for(int e=E.head[i];e;e=E.nxt[e]){
int j=E.to[e]; if(j==N) continue;
X.a[1][i]-=dp[j][h+damage[i]]*1.0/cnt[j];
}
}
if(h==HP) X.a[1][1]-=1;
for(int i=1;i<=N;i++)
dp[i][h]=(B[i]&X).a[1][1];
ANS+=dp[N][h];
}
cout<<fixed<<setprecision(8)<<ANS<<endl;
return 0;
}
●BZOJ 3640 JC的小苹果的更多相关文章
- BZOJ 3640: JC的小苹果 [概率DP 高斯消元 矩阵求逆]
3640: JC的小苹果 题意:求1到n点权和\(\le k\)的概率 sengxian orz的题解好详细啊 容易想到\(f[i][j]\)表示走到i点权为j的概率 按点权分层,可以DP 但是对于\ ...
- BZOJ 3640: JC的小苹果
3640: JC的小苹果 Time Limit: 15 Sec Memory Limit: 256 MBSubmit: 425 Solved: 155[Submit][Status][Discus ...
- BZOJ 3640 JC的小苹果(逆矩阵)
题目链接:http://www.lydsy.com:808/JudgeOnline/problem.php?id=3640 题意:给出一个无向图,从1走到n.开始是血量H,从u到达v时血量减少a[v] ...
- 3640: JC的小苹果 - BZOJ
让我们继续JC和DZY的故事.“你是我的小丫小苹果,怎么爱你都不嫌多!”“点亮我生命的火,火火火火火!”话说JC历经艰辛来到了城市B,但是由于他的疏忽DZY偷走了他的小苹果!没有小苹果怎么听歌!他发现 ...
- 【BZOJ】3640: JC的小苹果
题解 我们考虑列出期望方程组,\(dp[i][j]\)表示在第\(i\)个点血量为\(j\)的时候到达\(N\)点的概率,所有的\(dp[N][j]\)都是1,所有\(j < 0\)都是0 答案 ...
- 【BZOJ 3640】JC的小苹果 (高斯消元,概率DP)
JC的小苹果 Submit: 432 Solved: 159 Description 让我们继续JC和DZY的故事. “你是我的小丫小苹果,怎么爱你都不嫌多!” “点亮我生命的火,火火火火火!” 话 ...
- 【BZOJ3640】JC的小苹果 概率DP+高斯消元
[BZOJ3640]JC的小苹果 Description 让我们继续JC和DZY的故事. “你是我的小丫小苹果,怎么爱你都不嫌多!” “点亮我生命的火,火火火火火!” 话说JC历经艰辛来到了城市B,但 ...
- bzoj千题计划291:bzoj3640: JC的小苹果
http://www.lydsy.com/JudgeOnline/problem.php?id=3640 dp[i][j] 表示i滴血到达j的概率 dp[i][j] = Σ dp[i+val[i]][ ...
- BZOJ3640 : JC的小苹果
设$f[i][j]$表示$hp$为$i$,在$j$点的概率,$d[i]$表示$i$的度数,$w[i]$表示经过$i$点要扣掉的血量. 对于$j$到$k$这条边,$f[i-w[k]][k]+=\frac ...
随机推荐
- 实验MyOD
实验MyOD 编写MyOD.java 用java MyOD XXX实现Linux下od -tx -tc XXX的功能 提交测试代码和运行结果截图,加上学号水印,提交码云代码链接. 代码如下: (刚开始 ...
- 项目Alpha冲刺Day11
一.会议照片 二.项目进展 1.今日安排 熟悉框架的使用以及编写用户查看的界面以及该页面内的操作. 2.问题困难 全局的日期转换出现问题,在序列化的时候是按照配置来的,但是反序列化的时候就错了,问题待 ...
- 基础篇 - SQL 的约束
基础篇 - SQL 的约束 约束 一.实验简介 约束是一种限制,它通过对表的行或列的数据做出限制,来确保表的数据的完整性.唯一性.本节实验将在实践操作中熟悉 MySQL 中的几种约束. 二 ...
- 201421123042 《Java程序设计》第13周学习总结
1. 本周学习总结 以你喜欢的方式(思维导图.OneNote或其他)归纳总结多网络相关内容. 答: 2. 为你的系统增加网络功能(购物车.图书馆管理.斗地主等)-分组完成 为了让你的系统可以被多个用户 ...
- java的socket通信
本文讲解如何用java实现网络通信,是一个非常简单的例子,我比较喜欢能够立马看到结果,所以先上代码再讲解具体细节. 服务端: import java.io.BufferedReader; import ...
- Browser Object Model
BOM:浏览器提供的一系列对象 window对象是BOM最顶层对象 * 计时器setInterval(函数,时间)设置计时器 时间以毫秒为单位 clearInterval(timer) 暂停计时器se ...
- JAVA_SE基础——18.方法的递归
方法的递归是指在一个方法的内部调用自身的过程,递归必须要有结束条件,不然就会陷入无限递归的状态,永远无法结束调用,接下来用一个最简单的例子来体现下方法递归,使用递归算法计算自然数之和: public ...
- JAVA_SE基础——16.方法
接触过C语言的同学,这小章节很容易接受.Java中的方法是类似与C语言中的函数 功能和调用方法都类似 只不过叫法不一样 因为java是面向对象 c是面向过程 仅仅是叫法不同.. . 看到 ...
- 第一次PTA作业
题目6-1拆分实数整数及小数部分 1设计思路 (1) 第一步:阅读题目要求及所给部分. 第二步:根据题意补全相应函数. (2)流程图 无 2.实验代码 #include <stdio.h> ...
- Python内置函数(44)——len
英文文档: len(s) Return the length (the number of items) of an object. The argument may be a sequence (s ...