博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
[ NOI 2005 ] 聪聪与可可
阅读量:6956 次
发布时间:2019-06-27

本文共 2291 字,大约阅读时间需要 7 分钟。

\(\\\)


一张\(N\)个点,\(M\)条边的有向图中,猫在\(A\)点,鼠在\(B\)点,每一秒两者按照以下规则移动:

  • 猫先走去往老鼠所在地的最短路,可以走一步或两步,多条路径的话走点编号最小的。
  • 鼠等概率移动到直接连通的一个点或不动(概率为\(\frac{1}{deg_v+1}\))。

当任意时刻猫到达鼠所在地时鼠被吃掉,求鼠被吃掉的期望时间。

  • \(N,M\in [0,10^3]\)

\(\\\)

\(Solution\)


BZOJ200题达成纪念

  • 注意到图非常稀疏,可以\(SPFA\)预处理出,对于任意点对\((x,y)\)\(x\)按规则走向\(y\)的第一步去往的点\(next[x][y]\)

  • 具体操作是现对每个点跑一遍\(SPFA\),设\(dis[u][v]\)表示从\(u\)\(v\)的最短路长度,则所找的点的编号\(next[x][y]=min\{v|dis[v][y]+1=dis[x][y]\}\)。那么对于每一个要处理的猫和鼠的地点\((A,B)\),猫下一步到达的点就是\(next[next[A][B]][B]\)

  • 注意到猫每一秒都会向靠近鼠的方向移动两步,而鼠每一秒会向不确定的方向移动一步,所以鼠是一定会被吃掉的,不妨暴力\(dfs\)下去找到猫捉到鼠的时间。而搜索状态量过多,所以需要记忆化。设\(f[i][j]\)表示猫在\(i\)号节点,鼠在\(j\)号节点时时间的期望,那么初始化就是\(f[i][i]=0,f[i][j]=1\big|dis[i][j]\le 2,i\not=j\)

  • 然后记搜就可以了,转移方程很好理解,就是猫走向该走的地方,鼠下一步枚举,没搜过就搜一下:

    \[ f[i][j]=\frac{\sum_{dis[j][v]=1}f\big[next[next[i][j]][j]\big]\big[v\big]\ +\ f\big[next[next[i][j]][j]\big]\big[j\big]}{deg[j]+1}+1 \]

\(\\\)

\(Code\)


#include
#include
#include
#include
#include
#include
#include
#include
#define N 1010#define R register#define gc getchar#define inf 2000000000using namespace std;inline int rd(){ int x=0; bool f=0; char c=gc(); while(!isdigit(c)){if(c=='-')f=1;c=gc();} while(isdigit(c)){x=(x<<1)+(x<<3)+(c^48);c=gc();} return f?-x:x;}bool vis[N];double f[N][N];int n,m,s,t,tot,hd[N],d[N][N],deg[N],nxt[N][N];struct edge{int to,nxt;} e[N<<1];inline void add(int u,int v){ e[++tot].to=v; e[tot].nxt=hd[u]; hd[u]=tot;}queue
q;inline void bfs(int x){ q.push(x); d[x][x]=0; while(!q.empty()){ int u=q.front(); q.pop(); vis[u]=0; for(R int i=hd[u],v;i;i=e[i].nxt) if(d[x][v=e[i].to]>d[x][u]+1){ d[x][v=e[i].to]=d[x][u]+1; if(!vis[v]) vis[v]=1,q.push(v); } }}inline void dfs(int u,int v){ double res=0.0; int nu=nxt[nxt[u][v]][v]; for(R int i=hd[v],nv;i;i=e[i].nxt){ if(f[nu][nv=e[i].to]<-10) dfs(nu,nv); res+=f[nu][nv]; } if(f[nu][v]<-10) dfs(nu,v); f[u][v]=(res+f[nu][v])/(deg[v]+1)+1;}int main(){ n=rd(); m=rd(); s=rd(); t=rd(); for(R int i=1,u,v;i<=m;++i){ u=rd(); v=rd(); add(u,v); add(v,u); } for(R int i=1;i<=n;++i) for(R int j=1;j<=n;++j) d[i][j]=nxt[i][j]=inf,f[i][j]=-(double)inf; for(R int i=1;i<=n;++i) bfs(i); for(R int i=1;i<=n;++i) for(R int j=1;j<=n;++j) if(d[i][j]

转载于:https://www.cnblogs.com/SGCollin/p/9679069.html

你可能感兴趣的文章
P2633 Count on a tree
查看>>
重读<软件性能测试>摘要
查看>>
毕业季
查看>>
测评报告:热门项目管理工具哪家强?
查看>>
java.sql.SQLSyntaxErrorException: ORA-00904: " ": invalid identifier错误
查看>>
vue2.0做移动端开发用到的相关插件和经验总结
查看>>
Linux查看文件夹大小
查看>>
系统集成项目管理工程师整理资料
查看>>
writexml方法:
查看>>
AutoLayout经常用到的一些布局(含StackView)
查看>>
HLG 1541 集合划分【01背包】
查看>>
Java IO 详解
查看>>
php生成随机密码的几种方法
查看>>
c#编程模仿的1stopt界面
查看>>
Castle ActiveRecord的一对多问题
查看>>
移山亦可有道 ——读《移山之道》
查看>>
python使用windows注册表
查看>>
MySQL5.6.13安装步骤(Windows7 64位)札记 1
查看>>
使用 nginx + thin 的配置启动 rails server
查看>>
服务器用户登录记录
查看>>