昨天统计题数错联系锟神时提到了这道题,给他讲了我的思路,感觉博客写的太简略,正好百题之后准备重新理一遍博客,所以就先来写写这道题。
有v个城市,每个城市间都有道路连通,要检查e个道路,起点终点任选。输入v、e、检查每个道路的时间t。求最少总时间。
求最少总时间,感觉与UVa818切锁链中的环有些神似。
所给的e条道路是必经的,只需要找到最少的路使得给的路形成欧拉道路即可。
首先,对所有的道路判连通,假设形成了n个连通组,那么连起来就需要n-1条路。
然后,统计每组中度数为奇数的点,每一个奇数点都要构造另一条路使之度数为偶数,但是每一组仅需形成一条链而不是环,仅需形成欧拉道路而不是欧拉回路,所以统计个数要 减去链的端点2。
最后,求每组的和,减去一就是需要连成一个整体所用的道路数,加上e,总道路就求出来的。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48
| #include<iostream> #include<vector> #include<cstring> using namespace std; const int maxn=1010; int v,e,t,cnt; bool vis[maxn]; vector<int> g[maxn]; bool read(){ cin>>v>>e>>t; if(!v&&!e&&!t) return false; for(int i=0;i<maxn;++i) g[i].clear(); memset(vis,0,sizeof(vis)); for(int i=0;i<e;++i){ int a,b; cin>>a>>b; g[a].push_back(b); g[b].push_back(a); } return true; } void dfs(int cur){ if(vis[cur]) return; vis[cur]=true; cnt+=(int)g[cur].size()&1; for(int i=0;i<g[cur].size();++i) dfs(g[cur][i]); return; } int solve(){ int ans=0; for(int i=1;i<=v;++i){ if(!vis[i]&&!g[i].empty()){ cnt=0; dfs(i); ans+=max(cnt,2); } } return t*(max(ans/2-1,0)+e); } int main(){ ios::sync_with_stdio(false); int k=0; while(read()) cout<<"Case "<<++k<<": "<<solve()<<endl; return 0; }
|
** 本文迁移自我的CSDN博客,格式可能有所偏差。 **