题目地址

hdu5438

题干

代码和解释

解答本题时参考了一篇代码较短的博客,比较有意思,使用了STL vector二维数组。

可以结合下面的示例代码理解:

#include<iostream>
#include<vector>
using namespace std;
int main()
{
vector<int> n[100];
int i;
for(i=0;i<100;i++){
n[i].clear();
}
n[5].push_back(1);
n[5].push_back(3);
printf("%d\n",n[5].size());//为2
printf("%d\n",n[5][0]);//为1
printf("%d\n",n[5][1]);//为3
//printf("%d\n",n[5]);//编译不通过
return 0;
}

可以理解为100行一维数组,每行长度不定,vector相当于变长数组。所以之前一维数组时对 n (vector<int> n)的操作在这里都对 n[i] (vector<int> n[100];int i;)应用。

本题的dfs跟网上的dfs基本模板差别有点大,不太易于作为dfs学习的开始,这里不详细解释。

这里是c++代码。

#include<iostream>
#include<vector>
#include<cstring>
#include<cstdio>
using namespace std;
int v[10010],f[10010],l[10010];//v存储每个池塘的值,f存储其是否被遍历过,l存储每个池塘连接管子数
vector<int> x[10010];//x存储每个池塘与哪个池塘相连的具体情况
int tmp;//tmp是已经遍历的节点个数
long long sum1,sum2;
void dfs(int t);
int main()
{
int T,p,m,a,b;//T为样例组数,p为池塘数,m为连接组合数,a、b为每组连接中两个池塘的位置(编号)
int i,j;
int flag;
scanf("%d",&T);
while(T--){
scanf("%d%d",&p,&m);
sum1=0;
memset(v,0,sizeof(v));
memset(f,0,sizeof(f));
memset(l,0,sizeof(l));//初始化
for(i=0;i<10010;i++){
x[i].clear();//初始化
}
for(i=1;i<=p;i++){//从1开始到p结束,不然会错,因为a和b存储位置是从1开始的
scanf("%d",&v[i]);
}
for(i=0;i<m;i++){
scanf("%d%d",&a,&b);
x[a].push_back(b);//a与b相连
x[b].push_back(a);//b与a相连
l[a]++;//与a相连的池塘(管子)数加1
l[b]++;//与b相连的池塘(管子)数加1
}
flag=1;
while(flag==1){
flag=0;//如果所有剩下的池塘连接管子数都大于等于2,则flag就会保持为0
for(i=1;i<=p;i++){//i从1开始到p
if(l[i]==0||l[i]==1){
//连接管子数小于2
flag=1;//只要这样的池塘还存在,就让flag为1,重新执行循环
f[i]=1;//表示已经遍历过
for(j=0;j<x[i].size();j++){//x[i].size表示位置为i的这个池塘连接的管子数
l[x[i][j]]--;//让所有与位置为i的池塘相连的池塘的连接管子数减1,即删除了位置为i的池塘与其他池塘的连接关系
}
l[i]=-1;//表示连接管子数小于2
}
}
}
//这样处理完后就只剩下连接管子数大于等于2的池塘,接下来用dfs判断每个连接组合是否包含奇数个池塘
for(i=1;i<=p;i++){//i从1开始到p
if(f[i]==0){
//说明是还没有遍历过的
sum2=0;
tmp=0;
dfs(i);//经过这个操作,tmp变成这个组合中池塘的数量,sum2变成这个组合中所有池塘值的和,并且这个组合中所有池塘都被遍历过了
if(tmp%2==1){//如果包含奇数个池塘
sum1+=sum2;
}
}
}
printf("%lld\n",sum1);
}
return 0;
}
void dfs(int t){
int i;
tmp++;
f[t]=1;//表示已经遍历过,这样就不会对同一个组合中的每个池塘多次计算了
sum2+=v[t];//加上这个池塘的值
for(i=0;i<x[t].size();i++){
if(f[x[t][i]]==0){//对于与这个池塘相连的所有未被遍历过的池塘
dfs(x[t][i]);
}
}
return;
}

解本题时一开始又读错了题意,最终要求加起来的是组合中池塘个数为奇数的,而我理解成了组合中的含值为奇数的池塘的。

参考

HDU 5438.Ponds【2015 ACM/ICPC Asia Regional Changchun Online】【DFS】9月13

hdu5438 Ponds[DFS,STL vector二维数组]的更多相关文章

  1. C++ vector二维数组

    C++ 构建二维动态数组 int **p; p = ]; //注意,int*[10]表示一个有10个元素的指针数组 ; i < ; ++i) { p[i] = ]; } 这样就构成10*5的数组 ...

  2. SDUT OJ 图练习-BFS-从起点到目标点的最短步数 (vector二维数组模拟邻接表+bfs , *【模板】 )

    图练习-BFS-从起点到目标点的最短步数 Time Limit: 1000ms   Memory limit: 65536K  有疑问?点这里^_^ 题目描述 在古老的魔兽传说中,有两个军团,一个叫天 ...

  3. c++ vector二维数组常见写法

    vector<vector <int> > array(3);//定义了行数为3列数不定的二维数组 array.size()//返回二维数组的行数 array[0].size( ...

  4. c++ vector & 二维数组 & MessageBox

    vector: https://www.cnblogs.com/mr-wid/archive/2013/01/22/2871105.html c++ 二维数组: int **p; p = new in ...

  5. Vector 二维数组 实现

    1.C++实现动态二维数组 int **p; p = ]; //注意,int*[10]表示一个有10个元素的指针数组 ; i < ; ++i) { p[i] = ]; } 2.利用指针数组实现二 ...

  6. c++用vector创建二维数组

    1 vector二维数组的创建和初始化 std::vector <int> vec(10,90); //将10个一维动态数组初始为90std::vector<std::vector& ...

  7. 转:用STL中的vector动态开辟二维数组

    用STL中的vector动态开辟二维数组 源代码:#include <iostream>#include <vector>using namespace std;int mai ...

  8. stl vector创建二维数组

    vector<vector<); for (auto it = v.begin(); it != v.end(); it++) { ; (*it).reserve();//预留空间为5,但 ...

  9. C++ vector 实现二维数组

    在STL中Vector这一容器,无论是在封装程度还是内存管理等方面都由于传统C++中的数组.本文主要是关于使用Vector初始化.遍历方面的内容.其他二维的思想也是类似的. 这里简单叙述一下C++ 构 ...

随机推荐

  1. PHP实现智能语音播报

    原文地址 https://www.jianshu.com/p/91a046ec6ebc 大家估计都知道现在很多AI音响能够给你播报天气,叫你起床...甚至能够接受语音指令!所谓的人工智能音响,听起来很 ...

  2. php exec执行视频图片转换

    首先安装ffmpeg <?php set_time_limit(0) ; $cmd = "ffmpeg -i 'input/3.mp4' -r 1 -q:v 2 -f image2 i ...

  3. 【故障处理】 DBCA建库报错CRS-2566

    [故障处理] DBCA建库报错CRS-2566 PRCR-1071 PRCR-1006 一.1  BLOG文档结构图       一.2  前言部分   一.2.1  导读和注意事项 各位技术爱好者, ...

  4. 基于TCP协议的远程终端控制并发socketserver实现以及粘包问题处理

    # 客户端 # -*- coding: utf-8 -*- import socketserver import struct import json import subprocess class ...

  5. linux下nm命令的使用

    linux下强大的文件分析工具 -- nm 什么是nm nm命令是linux下自带的特定文件分析工具,一般用来检查分析二进制文件.库文件.可执行文件中的符号表,返回二进制文件中各段的信息. 目标文件. ...

  6. 一套不错的docker lnmp

    github地址:https://github.com/yeszao/dnmp 镜像基于linux alpine 内存占用小 安装常用软件采用apk命令

  7. poi读写doc和docx

    https://www.cnblogs.com/always-online/p/4800131.html POI是 Apache 旗下一款读写计算机中的 word 以及 excel 文件的工具. po ...

  8. 深度学习Keras框架笔记之Activation类使用

    使用 keras.layers.core.Activation(activation) Apply an activation function tothe input.(貌似是把激活函数应用到输入数 ...

  9. 学习Microsoft Visio(2)

    常用业务设计图示法 一.业务设计基础 1.名词概念 业务流程图:在公司.部门.岗位的层面上描述一个业务流程的宏观过程. 基本流程图:对某个处理过程的详细逻辑流程进行描述. 静态业务对象图(UML表示法 ...

  10. 1.zookeeper是干什么的?

    Zookeeper是Hadoop的一个子项目,虽然源自hadoop,但是我发现zookeeper脱离hadoop的范畴开发分布式框架的运用越来越多.今天我想谈谈zookeeper,本文不谈如何使用zo ...