Regional Changchun Online--Ponds
网址:http://acm.hdu.edu.cn/showproblem.php?pid=5438
Ponds
Time Limit: 1500/1000 MS (Java/Others) Memory Limit: 131072/131072 K (Java/Others)
Total Submission(s): 376 Accepted Submission(s): 116
v.
Now Betty wants to remove some ponds because she does not have enough money. But each time when she removes a pond, she can only remove the ponds which are connected with less than two ponds, or the pond will explode.
Note that Betty should keep removing ponds until no more ponds can be removed. After that, please help her calculate the sum of the value for each connected component consisting of a odd number of ponds
T(1≤T≤30)
which is the number of test cases.
For each test case, the first line contains two number separated by a blank. One is the number
p(1≤p≤104)
which represents the number of ponds she owns, and the other is the number
m(1≤m≤105)
which represents the number of pipes.
The next line contains p
numbers v1,...,vp,
where vi(1≤vi≤108)
indicating the value of pond i.
Each of the last m
lines contain two numbers a
and b,
which indicates that pond a
and pond b
are connected by a pipe.
1
7 7
1 2 3 4 5 6 7
1 4
1 5
4 5
2 3
2 6
3 6
2 7
21
将孩子个数小于等于1的点进入队列,修改与该点相连的孩子数目,维护队列,最后通过dfs查看每一堆池塘的数目,奇数加,偶数忽略。。。
思路好像很行,不过还是在开始的时候理解错了题意了,当时不太懂 odd number of ponds是什么意思,所以开始的时候没有dfs
真的该祭奠一下我的时间了啊,结束前开了好几发,但是一直返回WR,之后改的成了RE了,不知道原因,果然最后就把这个题掉了
结束后开始一点点找错,因为代码太乱了,所以就压根没有发现什么错误,之后再PS的帮助下发现了两个可耻的错误,开始的RE是因为入队之后删边的时候用的是队列里的当前点的孩子数目,结果这个数目可能在入队之后修改了(因为判断完之后就会改变数组节点中的值了),实际上可能不再有边了,这样result可能返回的是end(),所以在删除的时候erase(end())肯定就出错了
另外一个就更SB了,最后dfs图的时候竟然把n用成m了,这能对吗。。。
错误代码:
#pragma comment(linker, "/STACK:102400000000,102400000000")
#include<iostream>
#include<stdio.h>
#include<math.h>
#include <string>
#include<string.h>
#include<map>
#include<queue>
#include<set>
#include<utility>
#include<vector>
#include<algorithm>
#include<stdlib.h>
using namespace std;
#define eps 1e-8
#define pii pair<int,int>
#define INF 0x3f3f3f3f
#define rd(x) scanf("%d",&x)
#define rd2(x,y) scanf("%d%d",&x,&y)
#define ll long long int
#define mod 1000000007
#define maxn 1005
#define maxm 1000005
int point[10005];
struct T{
int childnum;
int i;
int mark;
vector <int> vec;
friend bool operator < (T n1, T n2)
{
return n1.childnum > n2.childnum;
}
}node[10005]; bool vis[10005];
int cnt=0;
long long tsum=0;
void dfs(int x){
if(vis[x]==true) return ;
cnt++,tsum += point[ x ];
vis[x]=true;
for(int i=0;i<node[x].childnum;i++)
dfs( node[x].vec[i] );
} int main (){
int Case;
rd(Case);
while(Case--){
memset(vis,0,sizeof(vis));
int n,m,temp1,temp2;
ll sum=0;
vector<int>::iterator result;
rd2(n,m);
for(int i=1;i<=n;i++) node[i].childnum=0,node[i].i=i,node[i].mark=0 ,node[i].vec.clear();
for(int i=1;i<=n;i++){
rd(point[i]);
sum+=point[i];
}
for(int i=0;i<m;i++){
rd2(temp1,temp2);
//bool mark = (find( node[temp1].vec.begin(), node[temp1].vec.end(), temp2 ) == node[temp1].vec.end() ); if(temp1!=temp2){
node[temp1].vec.push_back(temp2);
node[temp1].childnum++;
node[temp2].vec.push_back(temp1);
node[temp2].childnum++;
}
}
priority_queue <T> que;
for(int i=1;i<=n;i++)
if(node[i].childnum<=1){
que.push(node[i]);
node[i].mark=1;
sum-=point[i];
}
//cout<<"***"<<endl;
while(!que.empty()){
// cout<<"&&&"<<que.top().i<<endl;
if(que.top().childnum==1){ //修改为 : node[que.top().i].childnum==1
int x=que.top().vec[0];//cout<<x<<endl;
result = find( node[x].vec.begin(), node[x].vec.end(), que.top().i );
node[x].vec.erase(result);
node[x].childnum--; int i = que.top().i ;
node[i].vec.erase(node[i].vec.begin());
node[i].childnum-- ; if(node[x].childnum<=1 && node[x].mark == 0){
que.push(node[x]);
node[x].mark=1;
sum-=point[x];
}
}
que.pop();
}
for(int i=1;i<=m;i++){ ///修改为 : i<=n
tsum=cnt=0;
if(!vis[i] && node[i].childnum>=1 ){
dfs(i);
if( cnt%2==0 )
sum-=tsum;
}
// cout<<endl;
}
printf("%I64d\n",sum);
} return 0 ;
}
其实当时没有做出来一点都不冤,因为之后看自己的代码简直就是一坨屎,思路根本不清晰,所以导致有点思路之后敲代码还是有点一头雾水,所以以后还是有了思路之后,最起码是能整体的脉络差不多清晰了再敲。
#pragma comment(linker, "/STACK:102400000000,102400000000")
#include<iostream>
#include<stdio.h>
#include<math.h>
#include <string>
#include<string.h>
#include<map>
#include<queue>
#include<set>
#include<utility>
#include<vector>
#include<algorithm>
#include<stdlib.h>
using namespace std;
#define eps 1e-8
#define pii pair<int,int>
#define INF 0x3f3f3f3f
#define rd(x) scanf("%d",&x)
#define rd2(x,y) scanf("%d%d",&x,&y)
#define ll long long int int point[10005];
vector <int> vec[10005];
bool vis[10005];
bool inqueue[10005];
int cnt=0;
long long tsum=0; void dfs(int x)
{
if(vis[x]) return ;
cnt++,tsum += point[ x ],vis[x]=true;
for(int i=0; i<vec[x].size(); i++)
dfs( vec[x][i] );
} void addedge(int temp1,int temp2)
{
vec[temp1].push_back(temp2);
vec[temp2].push_back(temp1);
} int main ()
{
int Case;
rd(Case);
while(Case--)
{
memset(vis,0,sizeof(vis));
memset(inqueue,0,sizeof(inqueue));
int n,m,temp1,temp2;
ll sum=0;
vector<int>::iterator result;
rd2(n,m);
for(int i=1; i<=n; i++)
{
vec[i].clear();
rd(point[i]);
sum+=point[i];
}
for(int i=0; i<m; i++)
{
rd2(temp1,temp2);
addedge(temp1,temp2);
} queue <int> que; ///没必要将点放入队列 建立索引关系就行
for(int i=1; i<=n; i++)
if(vec[i].size()<=1)
{
que.push(i);
vis[i]=true,inqueue[i]=1,sum-=point[i];
} while(!que.empty())
{
if( vec[que.front()].size()==1 )
{
int x=vec[que.front()][0];
result = find( vec[x].begin(), vec[x].end(), que.front() );
vec[x].erase(result); vec[que.front()].clear(); ///将队列中仅有一个元素清空 if(vec[x].size()<=1 && inqueue[x] == 0)
{
que.push(x);
inqueue[x]=1,vis[x]=true,sum-=point[x];
}
}
que.pop();
} for(int i=1; i<=n; i++){
tsum=cnt=0;
if( !vis[i] ){
dfs(i);
if( cnt%2==0 )
sum-=tsum;
}
}
printf("%I64d\n",sum);
} return 0 ;
}
其实在遍历队列中的元素的时候根本没有必要删除边的关系,因为已经在入队的时候为之后的dfs做准备了,只要入队之后便不再遍历了,删除边就没什么用了,只是更改vec[i]中的数目就可以了...
其实开始的思路清晰的话肯定早就A掉了,以后注意。。。
Regional Changchun Online--Ponds的更多相关文章
- 2015ACM/ICPC Asia Regional Changchun Online /HDU 5438 图
Ponds Time Limit: 1500/1000 MS (Java/Others) Memory Limit: 1310 ...
- 2013 Asia Regional Changchun C
Little Tiger vs. Deep Monkey Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65535/65535 K ( ...
- Regional Changchun Online--Elven Postman(裸排序二叉树)
Elven Postman Time Limit: 1500/1000 MS (Java/Others) Memory Limit: 131072/131072 K (Java/Others) Tot ...
- Regional Changchun Online--Travel(最小生成树&& 并查集)
Travel Time Limit: 1500/1000 MS (Java/Others) Memory Limit: 131072/131072 K (Java/Others) Total S ...
- Regional Changchun Online--Alisha’s Party
Alisha's Party Time Limit: 3000/2000 MS (Java/Others) Memory Limit: 131072/131072 K (Java/Others) ...
- hdu 5444 Elven Postman(二叉树)——2015 ACM/ICPC Asia Regional Changchun Online
Problem Description Elves are very peculiar creatures. As we all know, they can live for a very long ...
- (并查集)Travel -- hdu -- 5441(2015 ACM/ICPC Asia Regional Changchun Online )
http://acm.hdu.edu.cn/showproblem.php?pid=5441 Travel Time Limit: 1500/1000 MS (Java/Others) Memo ...
- (二叉树)Elven Postman -- HDU -- 54444(2015 ACM/ICPC Asia Regional Changchun Online)
http://acm.hdu.edu.cn/showproblem.php?pid=5444 Elven Postman Time Limit: 1500/1000 MS (Java/Others) ...
- 2015 ACM/ICPC Asia Regional Changchun Online HDU 5444 Elven Postman【二叉排序树的建树和遍历查找】
Elven Postman Time Limit: 1500/1000 MS (Java/Others) Memory Limit: 131072/131072 K (Java/Others)T ...
随机推荐
- [oracle] 解决X64操作系统PL/SQL连接报错问题 make sure you have the 32 bits oracle client installed
Windows 64位下装Oracle 11g 64位,PLSQL Developer使用出现以下问题: 1.Database下拉框为空: 2.强制输入用户名.密码及Database,登录弹出: In ...
- oracle数据库数据导出和导入
oracle的客户端里面的bin下面有两个可执行文件,名称分别为exp.exe和imp.exe. 他俩的用途就是导出和导入数据用的. 全库 导出:exp 用户名/密码@数据库名 full=y file ...
- 【jmeter】JMeter函数学习
JMeter函数是一些能够转化在测试树中取样器或者其他配置元件的域的特殊值.一个函数的调用就像这样:${_functionName(var1,var2,var3)},-functionName匹配函数 ...
- (转)mongodb分片
本文转载自:http://www.cnblogs.com/huangxincheng/archive/2012/03/07/2383284.html 在mongodb里面存在另一种集群,就是分片技术, ...
- Eclipse高效率开发技巧
工欲善其事,必先利其器.对于程序员来说,Eclipse便是其中的一个"器".本文会从Eclipse快捷键和实用技巧这两个篇章展开介绍.Eclipse快捷键用熟后,不用鼠标,便可进行 ...
- 迁移学习(Transfer Learning)(转载)
原文地址:http://blog.csdn.net/miscclp/article/details/6339456 在传统的机器学习的框架下,学习的任务就是在给定充分训练数据的基础上来学习一个分类模型 ...
- erlang尾递归的概括
网上看了些,自己总结了下 .没有局部变量,否则会爆栈 .递归函数的参数里面,至少有一个用来作为循环,另外一个一般用来保存临时结果,两者一起形成循环
- DB设计原则(一)字段名定义避免二义性。
字段名定义避免二义性.如:主数据中有库存信息表 KCDDID,KCDDMC.若要在业务表中存储库存地点ID的话,不要定义为KCDD,要定义为KCDDID.这样标明存储的就是ID,而不是名称.同样,单位 ...
- tcp-ip-状态详解(转)
TCP正常建立和关闭的状态变化 TCP连接的建立可以简单的称为三次握手,而连接的中止则可以叫做 四次握手. 建立连接 在TCP/IP协议中,TCP协议提供可靠的连接服务,采用三次握手建 ...
- eclipse 每次切换工作空间都要重新配置
首先,导出T1中的配置打开T1,选择file --> Export --> 在弹出框中选择General 下的preference --> next --> 在export p ...