题目地址

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. pandas-01 Series()的几种创建方法

    pandas-01 Series()的几种创建方法 pandas.Series()的几种创建方法. import numpy as np import pandas as pd # 使用一个列表生成一 ...

  2. jsonpath_rw操作json

    from jsonpath_rw import parse def get_key_from_data(key,data): # 定义匹配规则 json_expr=parse(key) result= ...

  3. Oracle 数据类型比较规则

    数值 较大的值被认为大于较小的值.所有负数都小于零,所有正数都小于零.因此,-1小于100:-100小于-1. 浮点值NaN(not a number))大于任何其他数值,且等于自身. 日期时间值 较 ...

  4. mysql性能优化之服务器参数配置-内存配置

    MySQL服务器参数介绍 MySQL获取配置信息路径 命令行参数 mysqld_safe --datadir=/data/sql_data 配置文件 mysqld --help --verbose | ...

  5. CVE-2019-10392:Jenkins Git client插件RCE复现

    0x00 简介 Jenkins是一个开源软件项目,是基于Java开发的一种持续集成工具,用于监控持续重复的工作,旨在提供一个开放易用的软件平台,使软件的持续集成变成可能. 0x01 漏洞概述 Git客 ...

  6. NLP从词袋到Word2Vec的文本表示

    在NLP(自然语言处理)领域,文本表示是第一步,也是很重要的一步,通俗来说就是把人类的语言符号转化为机器能够进行计算的数字,因为普通的文本语言机器是看不懂的,必须通过转化来表征对应文本.早期是基于规则 ...

  7. C++——STL(算法)

    以下对所有算法进行细致分类并标明功能:<一>查找算法(13个):判断容器中是否包含某个值adjacent_find:   在iterator对标识元素范围内,查找一对相邻重复元素,找到则返 ...

  8. Codeforces C. Jzzhu and Cities(dijkstra最短路)

    题目描述: Jzzhu and Cities time limit per test 2 seconds memory limit per test 256 megabytes input stand ...

  9. 一加5安卓P刷入twrp的recovery

    本文介绍的方法属于普适性的一般方法,比网上的各种工具箱会繁琐.但是工具箱不一定一直会更新(之前一加论坛的刷机工具箱已经停止更新了,估计是作者不用一加5了吧,毕竟已经好几年的手机了).并且如果你手机更新 ...

  10. .netcore发布时指定服务器的系统类型

    asp.net core 开发完成后发布,在IIS上面访问,直接报错  系统是windows2008 Application startup exception: System.DllNotFound ...