【问题】给出一颗n个结点的树,树上每条边的边权都是1,这n个结点中有m个特殊点,请你求出树上距离这m个特殊点距离均不超过d的点的数量,包含特殊点本身。

输入:

输入第一行包含三个正整数,n、m、d分别表示树上有n个结点,其中有m个点是特殊点,d是如题所示的距离。(1<=n、m、d<=50000)

第二行m个整数,表示特殊的点的编号,编号再1-n之间。

第三行有n-1个整数,第个数表示第号结点的父亲结点的编号,同样在1-n之间。

输出:

输出仅包含一个整数,即符合题目要求的点的数量。

样例输入:

6 2 3

2 1

3 4 5 6 1

样例输出:

2

首先题目给出了说明,这棵树一共有n个节点,且边的权重为1,并且有m个特殊节点,虽然题目说了这是一个树,但是要求是找到距离m个特殊节点"均"不超过d的节点,因此如果从某一个特殊节点出发,它既可以向上查找,也可以向下查找,这明明就是一个图,题目忽悠人的好不!!!

其实想通了思路就很清晰了,首先我们建立邻接表,题目给出的节点标号是1~N,由于数组的索引从零开始,因此建立邻接表的时候要对节点标号做一个减一的操作!因为我们使用了动态数组vector,因此对于没有连接的两个节点,我们就不添加元素(并不是设置为权重为零)!还有一点需要注意:自身与自身的节点也是相连的,需要添加进去!(题目说包含特殊节点本身)

// 创建邻接表
for (int i = ; i < list.size();i++) {
neigbor[i+].push_back(list[i] - );
neigbor[list[i] - ].push_back(i+);
}
for (int i = ; i < neigbor.size(); i++) {
neigbor[i].push_back(i);
}

接着我们遍历特殊节点,然后使用bfs算法来进行广度搜索,用dis变量来标记距离,如果距离大于d,就不再搜索,跳出循环!注意我们在进行广度搜索的时候要一次性处理一层节点!这也是while(size--)的作用,这种做法很类似于"之字形打印数组"。对于一个特殊节点,我们在dis变量的限制下尽可能的去搜索符合条件的节点,使用set用来判断是否访问过该节点,然后将flag中对应的节点进行自加操作!换句话说,一个特殊节点进行广度搜索时,flag数组的值最多只会加一次或者不加!

此时,flag数组表示对于某一个特殊节点,符合要求的节点有哪些(包含特殊节点本身),接着,我们用这个思路遍历每一个特殊节点就可以了!

如果特殊节点有N个,那么flag数组中值为N的标号就是满足条件的节点!也就是距离每个特殊节点距离均小于d。当然题目让返回满足条件的个数,那就数一下值为N的节点个数就行了!!!

#include <iostream>
#include <vector>
#include <queue>
#include <unordered_set>
using namespace std; int main() {
int n, m, d;
cin >> n >> m >> d; vector<vector<int>> neigbor(n);
vector<int> sp(m);
vector<int> list(n - );
for (int i = ; i < m; i++) {
cin >> sp[i];
}
for (int i = ; i < n - ; i++) {
cin >> list[i];
} // 创建邻接表
for (int i = ; i < list.size();i++) {
neigbor[i+].push_back(list[i] - );
neigbor[list[i] - ].push_back(i+);
}
for (int i = ; i < neigbor.size(); i++) {
neigbor[i].push_back(i);
} // 输出邻接表
cout << "创建邻接表,索引从0开始,题目中编号从1开始,需要注意!" << endl;
for (int i = ; i < neigbor.size(); i++) {
for (int j = ; j < neigbor[i].size(); j++) {
cout << neigbor[i][j] << " ";
}
cout << endl;
} vector<int> flag(n, );
for (int i = ; i < m; i++) {
int dis = ;
queue<int> que;
unordered_set<int> set;
que.push(sp[i] - ); // set不需要添加,由于特殊节点本身也需要访问
while (!que.empty()) {
int size = que.size(); while (size--) { // 类似于之字形打印二叉树,直接处理一层
int tmp = que.front();
que.pop();
for (int j = ; j < neigbor[tmp].size(); j++) {
if (set.find(neigbor[tmp][j]) == set.end()) {
set.insert(neigbor[tmp][j]);
flag[neigbor[tmp][j]]++;
cout << neigbor[tmp][j] << " ! "; // 输出满足与某一个特殊点距离不大于d的数
}
}
for (auto i : neigbor[tmp]) {
que.push(i);
}
}
dis++; // 距离加一
if (dis > d) {
break;
}
}
}
int count = ;
for (int i = ; i < flag.size(); i++) {
if (flag[i] == m) {
count++;
}
}
cout << endl;
cout << "特殊点与那些数相近: "; // 题目要求是特殊点与某些点 (均) 小于d
for (auto i : flag) {
cout << i << " ";
}
cout << endl;
cout << count << endl; // 输出
system("PAUSE");
return ;
}

输出结果分析:
由于题中样例构建出来的树为:(注意这里索引从零开始)
1 -> 2 -> 3 -> 4 -> 5 -> 0
所以对于特殊节点1和0来说:
距离1小于d=3的有:1,2,3,4
距离0小于d=3的有:0,5,4,3
所以我们程序中flag数组应该是[1,1,1,2,2,1], 从而符合题目要求的是3和4节点!!!运行结果也表明我们的思路和程序是对的!

从5 0到4 0 5表示我们所创建的邻接表(neigbor)

多源D点(邻接表+bfs)的更多相关文章

  1. 数据结构学习笔记05图 (邻接矩阵 邻接表-->BFS DFS、最短路径)

    数据结构之图 图(Graph) 包含 一组顶点:通常用V (Vertex) 表示顶点集合 一组边:通常用E (Edge) 表示边的集合 边是顶点对:(v, w) ∈E ,其中v, w ∈ V 有向边& ...

  2. ACM/ICPC 之 数据结构-邻接表+BFS(TSH OJ-无线广播Broadcast)

    这道题中若能够构成互不干扰的区域,其构成的图其实就是汉密尔顿路(Hamilton road),因此如果能够观察出来可以直接转化为汉密尔顿路的存在性证明,即便不能观察,我相信ACMer也能转化为BFS问 ...

  3. 图结构练习——BFS——从起始点到目标点的最短步数(邻接表+BFS)

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

  4. 图的全部实现(邻接矩阵 邻接表 BFS DFS 最小生成树 最短路径等)

    1 /** 2 * C: Dijkstra算法获取最短路径(邻接矩阵) 3 * 6 */ 7 8 #include <stdio.h> 9 #include <stdlib.h> ...

  5. AOJ GRL_1_A: Single Source Shortest Path (Dijktra算法求单源最短路径,邻接表)

    题目链接:http://judge.u-aizu.ac.jp/onlinejudge/description.jsp?id=GRL_1_A Single Source Shortest Path In ...

  6. PAT天梯赛练习题——L3-008. 喊山(邻接表+BFS)

    L3-008. 喊山 时间限制 150 ms 内存限制 65536 kB 代码长度限制 8000 B 判题程序 Standard 作者 陈越 喊山,是人双手围在嘴边成喇叭状,对着远方高山发出“喂—喂喂 ...

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

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

  8. bfs 邻接表(需要优化 可能会RE *【模板】)

    //---基于邻接表的bfs #include <stdio.h> #include <string.h> #include <iostream> #include ...

  9. 数据结构 《2》----基于邻接表表示的图的实现 DFS(递归和非递归), BFS

    图通常有两种表示方法: 邻接矩阵 和 邻接表 对于稀疏的图,邻接表表示能够极大地节省空间. 以下是图的数据结构的主要部分: struct Vertex{ ElementType element; // ...

随机推荐

  1. Day5-T2

    原题目 根据社会学研究表明,人们都喜欢和自己身高相近的人做朋友. 现在有 N 名身高各不相同的同学依次走进教室. 调查人员想预测每个人在走入教室的瞬间最想和 已经在教室的哪个人做朋友.当有两名同学和这 ...

  2. 连接mysql报错java.sql.SQLException: The server time zone value '�й���׼ʱ��' is unrecognized...解决方法

    报错内容: java.sql.SQLException: The server time zone value '�й���׼ʱ��' is unrecognized or represents mo ...

  3. 【蓝桥】第八届C语言C组第7题 Excel地址(进制变形题,stack()简单使用)转载

    标题: Excel地址 Excel单元格的地址表示很有趣,它使用字母来表示列号. 比如, A表示第1列, B表示第2列, Z表示第26列, AA表示第27列, AB表示第28列, BA表示第53列, ...

  4. 新闻类网站的通用爬虫--GNE

    GNE(GeneralNewsExtractor)是一个通用新闻网站正文抽取模块,输入一篇新闻网页的 HTML, 输出正文内容.标题.作者.发布时间.正文中的图片地址和正文所在的标签源代码.GNE在提 ...

  5. imagenet下载及训练

    imagenet 种子 迅雷打开验证集http://academictorrents.com/download/5d6d0df7ed81efd49ca99ea4737e0ae5e3a5f2e5.tor ...

  6. 修正png

    这是修正+取MD5的方法 function MD5FileTextPng(filename: AnsiString): AnsiString; var buf: ..MAX_PATH - ] of C ...

  7. 关于js中异步问题的解决方案

    在js中有一个始终无法绕过的问题,如何优雅地解决异步问题.实际上,js在执行过程中,每遇到一个异步函数,都会将这个异步函数放入一个异步队列中,只有当同步线程执行结束之后,才会开始执行异步队列中的函数, ...

  8. 使用Hibarnate: 出现 java.sql.SQLException: ORA-00911: 无效字符, 解决思路

    1. 查看到: Hibernat自动生成的sql查询语句 Hibernate: select * from ( select module0_.MODULE_ID as MODULE_ID1_1_, ...

  9. HDU - 6195 cable cable cable

    题意:如下图,有K个光源,一个光源可同时照射一个显示屏,问从M个显示屏中任选K个,可以使每个显示屏都被不同光源照亮最少需要多少连接电缆. 分析:画图分析可知 1.选1~K个显示屏和选M-K+1~M个显 ...

  10. SpringBoot#自定义配置的封装

    _震惊,开局 不可避免的需要弄一些自定义的配置. 要点: 1. 把配置项都写出来,分析层次关系:2. 抽象成bean与bean之间的关系,写出bean对应的类,这时候配置项对应了bean的属性,属性可 ...