1 条题解
-
0
自动搬运
来自洛谷,原作者为

听取MLE声一片
如果我当时做的再多一点,结局会不会不同呢?搬运于
2025-08-24 22:26:46,当前版本为作者最后更新于2020-12-06 15:55:08,作者可能在搬运后再次修改,您可在原文处查看最新版自动搬运只会搬运当前题目点赞数最高的题解,您可前往洛谷题解查看更多
以下是正文
如果说去掉高精度的话,还是一道非常好的 拓扑 排序题目。
本篇代码无高精,请自行添加(因为太复杂了qwq)。
拓扑排序:
拓扑排序主要思路在一个有向无环图中,先统计出每个点的入度个数,然后将入度为0的点入队,接着把队中每个点向它的出边做一个运算(本题中是将水分流到与其相连节点),然后断边(相连的点入度-1),最后就会得出排水节点的水量。
有不明白的同学可以看 神经网络 车站分级 旅行计划 都是很好的拓扑排序模板题。
拓扑排序函数:
void tp(){ for(int i=1;i<=n;i++)//所有入度为0的点入队(1-m) if(!in[i]){ book[i]=1; q.push(i); xx[i]=1,yy[i]=1; } while(!q.empty()){ int p=q.front(); q.pop(); if(out[p]) continue; for(int i=0;i<a[p].size();i++){ add(a[p][i],xx[p],yy[p]*(1ll*a[p].size())); if(book[a[p][i]]) continue; in[a[p][i]]--; if(in[a[p][i]]==0){ book[a[p][i]]=1; q.push(a[p][i]); } } } return; }注意几点:
- 如果是
vector存边,一定不要访问排水节点的size()这样可能会炸,要事先存一下。
2.本题是前
m个点是入水口,一定要注意审好题(虽然只有前m个点入度为0)。
分数处理:
我主要运用的是通分思想:
$ \frac{a}{b}+\frac{c}{d}=\frac{ad}{bd}+\frac{cb}{bd}=\frac{ad+cb}{bd} $
紧接着用
gcd化简(考场差点写挂):ll gcd(ll x,ll y){ if(y==0) return x; return gcd(y,x%y); }下面是通分代码:
void add(int u,ll x,ll y){ if(y==0) return; if(yy[u]==0){ xx[u]=x; yy[u]=y; return; } ll p1=xx[u]*y+yy[u]*x; ll p2=yy[u]*y; ll p3=gcd(p1,p2); xx[u]=p1/p3; yy[u]=p2/p3; return; }注意事项:
1.一定要判出要添加的分母是否为
0,如果为0直接赋值即可。2.最后输出保险在约分一下。
全部代码(考试代码):
#include<iostream> #include<cstdio> #include<cmath> #include<algorithm> #include<queue> #include<stack> #include<vector> #include<map> #define ll long long using namespace std; inline ll read(){ ll x=0,f=1;char ch=getchar(); while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();} while(ch>='0'&&ch<='9'){x=x*10+ch-'0';ch=getchar();} return x*f; } int n,m,in[100001],out[100001],book[100001]; ll xx[100001],yy[100001]; ll gcd(ll x,ll y){ if(y==0) return x; return gcd(y,x%y); } void add(int u,ll x,ll y){ if(y==0) return; if(yy[u]==0){ xx[u]=x; yy[u]=y; return; } ll p1=xx[u]*y+yy[u]*x; ll p2=yy[u]*y; ll p3=gcd(p1,p2); xx[u]=p1/p3; yy[u]=p2/p3; return; } vector<int> a[500001]; queue<int> q; void tp(){ for(int i=1;i<=n;i++) if(!in[i]){ book[i]=1; q.push(i); xx[i]=1,yy[i]=1; } while(!q.empty()){ int p=q.front(); q.pop(); if(out[p]) continue; for(int i=0;i<a[p].size();i++){ add(a[p][i],xx[p],yy[p]*(1ll*a[p].size())); if(book[a[p][i]]) continue; in[a[p][i]]--; if(in[a[p][i]]==0){ book[a[p][i]]=1; q.push(a[p][i]); } } } return; } int main() { //freopen("water.in","r",stdin); //freopen("water.out","w",stdout); n=read(),m=read(); for(int i=1;i<=n;i++){ int d=read(); if(d==0){ out[i]=1; continue; } while(d--){ int v; v=read(); a[i].push_back(v); in[v]++; } } tp(); for(int i=1;i<=n;i++){ if(out[i]){ add(i,0,1); printf("%lld %lld\n",xx[i],yy[i]); } } return 0; }谢谢大家!
- 如果是
- 1
信息
- ID
- 6305
- 时间
- 1000ms
- 内存
- 512MiB
- 难度
- 4
- 标签
- 递交数
- 0
- 已通过
- 0
- 上传者