题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1415

乍一看和“游走”一样。于是高斯消元。n^2状态,复杂度n^6……

看看TJ,发现因为聪聪不是随便走的,所以聪聪一直逼近可可。故其实无环。可以记搜。

(1A还是不错的)

#include<iostream>
#include<cstdio>
#include<cstring>
#include<queue>
using namespace std;
const int N=;
int n,m,l0,l1,nxt[N][N],dfn[N],du[N];
double dp[N][N];
bool vis[N],vs[N][N];
priority_queue<int,vector<int>,greater<int> >ed[N];
void add(int x,int y)
{
ed[x].push(y);ed[y].push(x);
du[x]++;du[y]++;
}
void bfs(int cr)
{
priority_queue<int,vector<int>,greater<int> >tp;
memset(vis,,sizeof vis);memset(dfn,,sizeof dfn);
queue<int> q;q.push(cr);vis[cr]=;nxt[cr][cr]=cr;
while(q.size())
{
int k=q.front();q.pop();
tp=ed[k];
while(tp.size())
{
int v=tp.top();tp.pop();
if(vis[v])continue;
dfn[v]=dfn[k]+;vis[v]=;q.push(v);
if(dfn[v]>)nxt[v][cr]=nxt[k][cr];
else nxt[v][cr]=v;
}
}
}
double dfs(int x,int y)//coco->x ,cncn->y
{
// printf("x=%d y=%d\n",x,y);
if(vs[x][y])return dp[x][y];vs[x][y]=;
if(nxt[x][y]==x){/*printf("rt x=%d y=%d\n",x,y);*/return dp[x][y]=;}
double ret=(dfs(x,nxt[x][y])+(nxt[x][y]!=x))/(du[x]+);
priority_queue<int,vector<int>,greater<int> >tp=ed[x];//每层定义
// printf("siz[%d]=%d\n",x,tp.size());
while(tp.size())
{
int v=tp.top();tp.pop();
// printf("v=%d nxt[%d][%d]=%d\n",v,x,y,nxt[x][y]);
ret+=(dfs(v,nxt[x][y])+(nxt[x][y]!=v))/(du[x]+);
}
// printf("ret=%.3lf\n",ret);
return dp[x][y]=ret;
}
int main()
{
scanf("%d%d%d%d",&n,&m,&l0,&l1);int x,y;
for(int i=;i<=m;i++)
{
scanf("%d%d",&x,&y);add(x,y);
}
for(int i=;i<=n;i++)bfs(i);
printf("%.3lf",dfs(l1,l0));
return ;
}

版本1

看看TJ,发现判断得那么纠结是因为x==y时正常应返回0。

#include<iostream>
#include<cstdio>
#include<cstring>
#include<queue>
using namespace std;
const int N=;
int n,m,l0,l1,nxt[N][N],dfn[N],du[N];
double dp[N][N];
bool vis[N],vs[N][N];
priority_queue<int,vector<int>,greater<int> >ed[N];
void add(int x,int y)
{
ed[x].push(y);ed[y].push(x);
du[x]++;du[y]++;
}
void bfs(int cr)
{
priority_queue<int,vector<int>,greater<int> >tp;
memset(vis,,sizeof vis);memset(dfn,,sizeof dfn);
queue<int> q;q.push(cr);vis[cr]=;nxt[cr][cr]=cr;
while(q.size())
{
int k=q.front();q.pop();
tp=ed[k];
while(tp.size())
{
int v=tp.top();tp.pop();
if(vis[v])continue;
dfn[v]=dfn[k]+;vis[v]=;q.push(v);
if(dfn[v]>)nxt[v][cr]=nxt[k][cr];
else nxt[v][cr]=v;
}
}
}
double dfs(int x,int y)//coco->x ,cncn->y
{
if(vs[x][y])return dp[x][y];vs[x][y]=;
if(x==y)return dp[x][y]=;//
if(nxt[x][y]==x)return dp[x][y]=;
double ret=dfs(x,nxt[x][y])/(du[x]+)+;
priority_queue<int,vector<int>,greater<int> >tp=ed[x];//每层定义
while(tp.size())
{
int v=tp.top();tp.pop();
ret+=dfs(v,nxt[x][y])/(du[x]+);
}
return dp[x][y]=ret;
}
int main()
{
scanf("%d%d%d%d",&n,&m,&l0,&l1);int x,y;
for(int i=;i<=m;i++)
{
scanf("%d%d",&x,&y);add(x,y);
}
for(int i=;i<=n;i++)bfs(i);
printf("%.3lf",dfs(l1,l0));
return ;
}

版本2

但是自己代码巨慢……想来是用了优先队列的缘故(为了标号字典序)。尝试改一改。

#include<iostream>
#include<cstdio>
#include<cstring>
#include<queue>
using namespace std;
const int N=;
int n,m,l0,l1,nxt[N][N],dfn[N],du[N],head[N],xnt;
double dp[N][N];
bool vis[N],vs[N][N];
priority_queue<int,vector<int>,greater<int> >ed[N];
struct Edge{
int next,to;
Edge(int n=,int t=):next(n),to(t) {}
}edge[N<<];
void add(int x,int y)
{
edge[++xnt]=Edge(head[x],y);head[x]=xnt;
edge[++xnt]=Edge(head[y],x);head[y]=xnt;
ed[x].push(y);ed[y].push(x);
du[x]++;du[y]++;
}
void bfs(int cr)
{
priority_queue<int,vector<int>,greater<int> >tp;
memset(vis,,sizeof vis);memset(dfn,,sizeof dfn);
queue<int> q;q.push(cr);vis[cr]=;nxt[cr][cr]=cr;
while(q.size())
{
int k=q.front();q.pop();
tp=ed[k];
while(tp.size())
{
int v=tp.top();tp.pop();
if(vis[v])continue;
dfn[v]=dfn[k]+;vis[v]=;q.push(v);
if(dfn[v]>)nxt[v][cr]=nxt[k][cr];
else nxt[v][cr]=v;
}
}
}
double dfs(int x,int y)//coco->x ,cncn->y
{
if(vs[x][y])return dp[x][y];vs[x][y]=;
if(x==y)return dp[x][y]=;//
if(nxt[x][y]==x)return dp[x][y]=;
double ret=dfs(x,nxt[x][y])/(du[x]+)+;
for(int i=head[x];i;i=edge[i].next)
ret+=dfs(edge[i].to,nxt[x][y])/(du[x]+);
return dp[x][y]=ret;
}
int main()
{
scanf("%d%d%d%d",&n,&m,&l0,&l1);int x,y;
for(int i=;i<=m;i++)
{
scanf("%d%d",&x,&y);add(x,y);
}
for(int i=;i<=n;i++)bfs(i);
printf("%.3lf",dfs(l1,l0));
return ;
}

结果只是快了28ms。

别人用一些方法保证字典序。

这个人bfs+两步保证dis最小的前提下调整标号至最小。https://blog.csdn.net/clove_unique/article/details/62237321

这个人spfa的同时判断标号。(但只能记录一步的pre)https://blog.csdn.net/PoPoQQQ/article/details/40896403(bfs因为是bfs所以不能边求dis边调整标号)

可是我都没记录dis。用的dfn。所以懒得改了……比较欣赏第一个人的写法。

bzoj 1415 [Noi2005]聪聪和可可——其实无环的图上概率的更多相关文章

  1. BZOJ 1415: [Noi2005]聪聪和可可( 最短路 + 期望dp )

    用最短路暴力搞出s(i, j)表示聪聪在i, 可可在j处时聪聪会走的路线. 然后就可以dp了, dp(i, j) = [ dp(s(s(i,j), j), j) + Σdp(s(s(i,j), j), ...

  2. BZOJ 1415 [NOI2005]聪聪与可可 (概率DP+dfs)

    题目大意:给你一个无向联通图,节点数n<=1000.聪聪有一个机器人从C点出发向在M点的可可移动,去追赶并吃掉可可,在单位时间内,机器人会先朝离可可最近的节点移动1步,如果移动一步机器人并不能吃 ...

  3. BZOJ 1415: [Noi2005]聪聪和可可 [DP 概率]

    传送门 题意:小兔子乖乖~~~ 题意·真:无向图吗,聪抓可,每个时间聪先走可后走,聪一次可以走两步,朝着里可最近且点编号最小的方向:可一次只一步,等概率走向相邻的点或不走 求聪抓住可的期望时间 和游走 ...

  4. bzoj 1415: [Noi2005]聪聪和可可 期望dp+记忆化搜索

    期望dp水题~ 你发现每一次肯定是贪心走 2 步,(只走一步的话就可能出现环) 然后令 $f[i][j]$ 表示聪在 $i$,可在 $j$,且聪先手两个人碰上面的期望最小次数. 用记忆化搜索转移就行了 ...

  5. bzoj 1415: [Noi2005]聪聪和可可

    直接上记忆化搜索 #include<queue> #include<cstdio> #include<algorithm> using namespace std; ...

  6. bzoj 1415: [Noi2005]聪聪和可可【期望dp+bfs】

    因为边权为1所以a直接bfs瞎搞就行--我一开始竟然写了个spfa #include<iostream> #include<cstdio> #include<queue& ...

  7. BZOJ 1415: [Noi2005]聪聪和可可(记忆化搜索+期望)

    传送门 解题思路 还是比较简答的一道题.首先\(bfs\)把每个点到其他点的最短路求出来,然后再记忆化搜索.记搜的时候猫的走法是确定的,搜一下老鼠走法就行了. 代码 #include<iostr ...

  8. 【BZOJ】【1415】【NOI2005】聪聪和可可

    数学期望+记忆化搜索 论文:<浅析竞赛中一类数学期望问题的解决方法>——汤可因  中的第一题…… Orz 黄学长 我实在是太弱,这么简单都yy不出来…… 宽搜预处理有点spfa的感觉= = ...

  9. 【BZOJ 1415】 1415: [Noi2005]聪聪和可可 (bfs+记忆化搜索+期望)

    1415: [Noi2005]聪聪和可可 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 1640  Solved: 962 Description I ...

随机推荐

  1. 安装专业版的linux的方法 图解安装专业版的linux

    按装一个linux系统其实很简单,不再像以前那样光光盘就好些个,一不小心还又可能装错,实在也是一个大问题.现在好了基本上都是简单安装+网络升级先安装主要的后面如果需要什么在装什么?大大简化了流程.不行 ...

  2. SpringBoot 事务隔离性和传播性

    propergation 传播性 Spring中七种Propagation类的事务属性详解:  REQUIRED:支持当前事务,如果当前没有事务,就新建一个事务.这是最常见的选择.  SUPPORTS ...

  3. C语言排序算法之简单交换法排序,直接选择排序,冒泡排序

    C语言排序算法之简单交换法排序,直接选择排序,冒泡排序,最近考试要用到,网上也有很多例子,我觉得还是自己写的看得懂一些. 简单交换法排序 /*简单交换法排序 根据序列中两个记录键值的比较结果来对换这两 ...

  4. Pandas级联

    Pandas提供了各种工具(功能),可以轻松地将Series,DataFrame和Panel对象组合在一起. pd.concat(objs,axis=0,join='outer',join_axes= ...

  5. power shell添加vim

    1.去Vim官网下载适合操作系统的可执行文件 地址:https://www.vim.org/download.php#pc 2.找到Vim文件夹中的vimrc文件进行修改,增加下面这4行. set e ...

  6. java reflect反射---Java高级开发必须懂的

    理解反射对学习Java框架有很大的帮助,如Spring框架的核心就是使用Java反射实现的,而且对做一些Java底层的操作会很有帮助.  一.Class类的使用         1.万事万物皆对象,( ...

  7. Python globals() locals() vars() 三个内建函数的区别

    首先参考官方文档对这三个函数的介绍: 2. Built-in Functions – Python3.5 1.globals() 返回当前全局符号表, 通常是返回当前模块下的全局符号表, 比如全局内建 ...

  8. 用I/O口模拟总线时序

    在做总线通信过程中,我们很少会用到这样方法,一般在我们选择MCU的时候都会带有你所需要的通信接口.但是,对于一些简单的通信应该用的场合,一般在一些传感器的数据通信过程中,传感器厂商会将通信协议做一些改 ...

  9. Windows 使用技巧

    怎样添加“发送到”的快捷方式,cmd里面shell:sendto打开文件夹,把快捷方式放进去就可以了. vs中c#快速实现接口所有函数快捷键: 鼠标放在实现的接口上面+shift+alt+F10.

  10. poscms仿站知识点总结(一)

    最近在做基于poscms系统的企业站仿站项目,这个系列用于总结项目中遇到的一些前端问题,至于poscms,待我摸透之后再总结... 进入正题吧,仿站,首先是用仿站小工具把要仿的模板站扒下来,有时候会出 ...