PAT (Advanced Level) Practice 用到图的存储方式,但没有用到图的算法的题目

目录

  • 1122 Hamiltonian Cycle (25)
  • 1126 Eulerian Path (25)
  • 1134 Vertex Cover (25)
  • 1142 Maximal Clique (25)
  • 1154 Vertex Coloring (25)

1122 Hamiltonian Cycle (25)

题目思路

  • n != queryV.size() 检查是否 query 覆盖了所有结点
  • kn != n + 1 检查 query 是否多走或少走
  • query[0] != query[kn-1] 检查是否成环
  • 遍历 query 检查是否每一步都可到达
#include<iostream>
#include<unordered_set>
using namespace std;
bool G[201][201] = {false};
int main()
{
int n, m, u, v, k, kn;
scanf("%d%d", &n, &m);
for (int i = 0; i < m; i++){
scanf("%d%d", &u, &v);
G[u][v] = G[v][u] = true;
}
scanf("%d", &k);
for (int i = 0; i < k; i++){
scanf("%d", &kn);
int query[kn];
unordered_set<int> queryV;
for (int j = 0; j < kn; j++){
scanf("%d", &query[j]);
queryV.insert(query[j]);
}
bool isC = true;
for (int j = 0; j < kn - 1; j++)
if (!G[query[j]][query[j+1]]) isC = false;
if (!isC || kn != n + 1 || n != queryV.size() || query[0] != query[kn-1]) printf("NO\n");
else printf("YES\n");
}
return 0;
}

1126 Eulerian Path (25)

题目思路

  • 用邻接表存储图,可以用每个结点对应邻接点的个数(Adj[i].size())表示每个结点的度
  • 题干给出根据度奇偶性和个数判断的先决条件是要是连通图
  • 先用深搜判断连通,从任一结点开始,看可以遍历到多少结点
  • 若遍历到的结点数与结点总数不同,则不是连通图,直接输出 Non-Eulerian
  • 若相同则说明是连通图,再遍历邻接表输出结点度数并记录度数为奇数的结点个数
  • 根据奇度数结点个数输出是否为 Eulerian
#include<iostream>
#include<vector>
#include<queue>
using namespace std;
vector<int> Adj[501];
bool vis[501] = {false};
int connected = 0;
void DFS(int root){
connected++;
vis[root] = true;
for (int i = 0; i < Adj[root].size(); i++)
if (!vis[Adj[root][i]]) DFS(Adj[root][i]);
}
int main()
{
int n, m, u, v, oddnum = 0;
scanf("%d%d", &n, &m);
for (int i = 0; i < m; i++){
scanf("%d%d", &u, &v);
Adj[u].push_back(v);
Adj[v].push_back(u);
}
DFS(1);
for (int i = 1; i < n + 1; i++){
printf("%d%c", Adj[i].size(), i == n ? '\n' : ' ');
if (Adj[i].size() % 2) oddnum++;
}
if (connected != n) printf("Non-Eulerian\n");
else printf("%s\n", !oddnum ? "Eulerian" : oddnum == 2 ? "Semi-Eulerian" : "Non-Eulerian");
return 0;
}

简化前代码

  • 用邻接矩阵存储图
  • 单独开数组记录结点度数
  • 用 BFS 判断是否连通
#include<iostream>
#include<vector>
#include<queue>
using namespace std;
int G[501][501] = {0};
bool vis[501] = {false};
int main()
{
int n, m, u, v, connected = 0, oddnum = 0;
scanf("%d%d", &n, &m);
int degree[n+1] = {0};
for (int i = 0; i < m; i++){
scanf("%d%d", &u, &v);
degree[u]++;
degree[v]++;
G[u][v] = G[v][u] = 1;
}
queue<int> q;
q.push(n);
vis[n] = true;
while (!q.empty()){
int now = q.front();
q.pop();
connected++;
for (int i = 1; i < n + 1; i++){
if (!vis[i] && G[now][i]){
q.push(i);
vis[i] = true;
}
}
}
for (int i = 1; i < n + 1; i++){
printf("%d%c", degree[i], i == n ? '\n' : ' ');
if (degree[i] % 2) oddnum++;
}
if (connected != n) printf("Non-Eulerian\n");
else printf("%s\n", !oddnum ? "Eulerian" : oddnum == 2 ? "Semi-Eulerian" : "Non-Eulerian");
return 0;
}

1134 Vertex Cover (25)

题目思路

  • vector<pair<int,int>> 保存边的端点
  • 对每个 query,新建一个 map 标记给出的 vertex
  • 遍历所有边,检查是否有边两个端点均不在给出的 vertex 中
  • 若有说明给出的 vertex 不能覆盖所有边,标记变量退出循环
  • 按标记变量输出要求内容
#include<iostream>
#include<vector>
#include<unordered_map>
using namespace std;
int main()
{
int n, m, k, nv, v;
scanf("%d%d", &n, &m);
vector<pair<int,int>> edges(m);
for (int i = 0; i < m; i++) scanf("%d%d", &edges[i].first, &edges[i].second);
scanf("%d", &k);
for (int i = 0; i < k; i++){
scanf("%d", &nv);
unordered_map<int,bool> vertex;
bool iscover = true;
for (int j = 0; j < nv; j++){
scanf("%d", &v);
vertex[v] = true;
}
for (int j = 0; j < m; j++){
if (!vertex[edges[j].first] && !vertex[edges[j].second]){
iscover = false;
break;
}
}
printf("%s\n", iscover ? "Yes" : "No");
}
return 0;
}

1142 Maximal Clique (25)

题目思路

  • 输入无向边,用邻接矩阵将两个方向的边均存储起来
  • 每输入一个待检查的序列,就将标记变量 isclique & isMax 均设为 true,新建保存序列的数组和保存序列结点的集合
  • 输入待查序列同时将结点压入集合
  • 首先用二重循环检查序列是否两两相邻,若不是,说明现有序列非 clique,设置标记输出内容并跳出循环
  • 若通过上个检查,已知是 clique,要检查是否是最大的,也就是是否有其他结点与序列中每个点都相邻
  • 用一个变量按顺序遍历结点标号,取出不在序列中的结点,与序列结点两两配对检查是否相邻
    • 若有一对不相邻就从序列结点中 break 取下一个结点
    • 若能一直检查到序列结尾,发现此结点与序列每个结点均相邻,说明现有序列非 max clique,设置标记输出内容跳出循环
  • 最后检查变量按要求输出内容
#include<iostream>
#include<set>
using namespace std;
bool G[201][201] = {false};
int main()
{
int nv, ne, u, v, m, K;
scanf("%d%d", &nv, &ne);
for (int i = 0; i < ne; i++){
scanf("%d%d", &u, &v);
G[u][v] = G[v][u] = true;
}
scanf("%d", &m);
for (int i = 0; i < m; i++){
bool isclique = true, isMax = true;
scanf("%d", &K);
int clique[K];
set<int> cliqueV;
for (int j = 0; j < K; j++){
scanf("%d", &clique[j]);
cliqueV.insert(clique[j]);
}
for (int j = 0; j < K - 1; j++){
for (int k = j + 1; k < K; k++){
if (!G[clique[j]][clique[k]]){
isclique = false;
printf("Not a Clique\n");
break;
}
}
if (!isclique) break;
}
if (isclique){
for (int j = 1; j <= nv; j++){
if (cliqueV.find(j) == cliqueV.end()){
for (int k = 0; k < K; k++){
if (!G[clique[k]][j]) break;
if (k == K - 1) isMax = false;
}
}
if (!isMax){
printf("Not Maximal\n");
break;
}
}
if (isMax) printf("Yes\n");
}
}
return 0;
}

1154 Vertex Coloring (25)

题目思路

  • vector<pair<int,int>>保存所有边
  • 将所有点的颜色存起来,同时放入set统计颜色个数
  • 枚举所有边,检查是否每条边的两点个颜色是否相同
  • 若有相同的边,设置标记
  • 根据标记 输出颜色个数 或 输出No
#include<iostream>
#include<vector>
#include<set>
using namespace std;
int main()
{
int n, m, k, u, v;
scanf("%d%d", &n, &m);
vector<pair<int,int>> edges(m);
for (int i = 0; i < m; i++)
scanf("%d%d", &edges[i].first, &edges[i].second);
scanf("%d", &k);
for (int i = 0; i < k; i++){
vector<int> colors(n);
set<int> coloring;
bool isokay = true;
for (int j = 0; j < n; j++){
scanf("%d", &colors[j]);
coloring.insert(colors[j]);
}
for (int j = 0; j < m; j++)
if (colors[edges[j].first] == colors[edges[j].second])
isokay = false;
if (!isokay) printf("No\n");
else printf("%d-coloring\n", coloring.size());
}
return 0;
}

PAT甲级 图 相关题_C++题解的更多相关文章

  1. PAT甲级 Dijkstra 相关题_C++题解

    Dijkstra PAT (Advanced Level) Practice Dijkstra 相关题 目录 <算法笔记>重点摘要 1003 Emergency (25) <算法笔记 ...

  2. PAT甲级 二叉树 相关题_C++题解

    二叉树 PAT (Advanced Level) Practice 二叉树 相关题 目录 <算法笔记> 重点摘要 1020 Tree Traversals (25) 1086 Tree T ...

  3. PAT甲级 二叉查找树 相关题_C++题解

    二叉查找树 PAT (Advanced Level) Practice 二叉查找树 相关题 目录 <算法笔记> 重点摘要 1099 Build A Binary Search Tree ( ...

  4. PAT甲级 树 相关题_C++题解

    树 目录 <算法笔记>重点摘要 1004 Counting Leaves (30) 1053 Path of Equal Weight (30) 1079 Total Sales of S ...

  5. PAT甲级 堆 相关题_C++题解

    堆 目录 <算法笔记>重点摘要 1147 Heaps (30) 1155 Heap Paths (30) <算法笔记> 9.7 堆 重点摘要 1. 定义 堆是完全二叉树,树中每 ...

  6. PAT甲级 散列题_C++题解

    散列 PAT (Advanced Level) Practice 散列题 目录 <算法笔记> 重点摘要 1002 A+B for Polynomials (25) 1009 Product ...

  7. PAT甲级 字符串处理题_C++题解

    字符串处理题 目录 <算法笔记> 重点摘要 1001 A+B Format (20) 1005 Spell It Right (20) 1108 Finding Average (20) ...

  8. PAT甲级 图的遍历 相关题_C++题解

    图的遍历 PAT (Advanced Level) Practice 图的遍历 相关题 目录 <算法笔记>重点摘要 1021 Deepest Root (25) 1076 Forwards ...

  9. PAT甲级 并查集 相关题_C++题解

    并查集 PAT (Advanced Level) Practice 并查集 相关题 <算法笔记> 重点摘要 1034 Head of a Gang (30) 1107 Social Clu ...

随机推荐

  1. python 最小二乘 leastsq 函数实现 法线式 解决与x轴垂直问题

    当使用y=kx+b时,与x轴垂直的直线无法计算.因此使用法线式ysin(theta)+xcos(theta) = dist.貌似这么用有点复杂了,直接使用ax+by=1不知道能不能计算,未测试. # ...

  2. BZOJ3236作业

    这东西是个应用为O(logn)的莫队. 正常莫队的updata函数转移是O(1)的,可这个题时间非常宽泛,可以套两个树状数组,那两个东西很好维护,第一个直接普通权值树状数组维护,第二个开一个桶,记录当 ...

  3. lucene正向索引——正向信息,Index –> Segments (segments.gen, segments_N) –> Field(fnm, fdx, fdt) –> Term (tvx, tvd, tvf)

    转自:http://www.cnblogs.com/forfuture1978/archive/2009/12/14/1623599.html 上面曾经交代过,Lucene保存了从Index到Segm ...

  4. regasm注册com组件

    注意: regasm.exe在不同framework版本下的系统路径 一般存储的路径为:C:\Windows\Microsoft.NET\Framework\v2.0.50727\ 系统的版本不同,运 ...

  5. 本周JavaScript学习小结

    应组长杨老师号召,写个js阶段性学习小结. emmm这周学了Linux进程通讯,学正则表达式尝试完成第一次编程作业,中秋还去平潭露营(所以...js学得很少hhh). 现在还处于感性认识阶段,浏览了一 ...

  6. case设计及验证:入口+页面+展示

    测试个性CB问题, 功能整体结构为:入口+页面+展示 总结: 1. 产品文档为主,其次是服务端接口返回.数据结构及字段值确认.结合实际场景检查是否有遗漏或不合理. 2. 以字段为维度,每个字段的检查点 ...

  7. Cesium中常用的一些地理数据文件 以及数据相关的东西

    KML Cesium.KmlDataSource.load CZML Cesium.CzmlDataSource.load GeoJson Cesium.GeoJsonDataSource.load ...

  8. js vue --- T Z 去掉 T Z 时间

    export const formatDate = (timestamp) => { return timestamp.replace(/T/g,' ').replace(/Z/g,'') } ...

  9. Mercury:唯品会全链路应用监控系统解决方案详解(含PPT)

    Mercury:唯品会全链路应用监控系统解决方案详解(含PPT) 原创: 姚捷 高可用架构 2016-08-08    

  10. Vuex 通俗版教程告诉你Vuex怎么用

    写在文前: 最近一直在用vue开发项目,写来写去就是那么些方法,对于简单的项目一些常用的vue方法足以解决,但是涉及到页面状态,权限判断等一些复杂的传值,vuex是必须的.对于vuex也运用一段时间, ...