题目描述
知识点
图
实现
码前思考
三步走:
- 判断连通图;
- 每个点的度都为偶数;
- 如果违反2,那么只能有且仅有两个点度为奇数。
代码实现
#include "bits/stdc++.h"
using namespace std;
const int maxn = 550;
vector<int> adj[maxn];
int degree[maxn];
bool vis[maxn];
bool isC;
bool isE;
int cnt;
int n;
int m;
void dfs(int idx){
if(vis[idx] == true){
return;
}else{
vis[idx] = true;
//深度搜索
for(int i=0;i<adj[idx].size();i++){
dfs(adj[idx][i]);
}
return;
}
}
int main(){
fill(vis,vis+maxn,false);
fill(degree,degree+maxn,0);
isC = true;
isE = true;
cnt = 0;
scanf("%d %d",&n,&m);
for(int i=0;i<m;i++){
int u;
int v;
scanf("%d %d",&u,&v);
adj[u].push_back(v);
degree[u]++;
adj[v].push_back(u);
degree[v]++;
}
//首先进行是否是连通图的排查
//是否是连通图,使用深度优先搜索去检查
dfs(1);
//查看是否所有的都是已经访问
//数组下标从1开始
for(int i=1;i<=n;i++){
if(vis[i] == false){
isC = false;
break;
}
}
//打印度
for(int i=1;i<=n;i++){
printf("%d",degree[i]);
if(i!=n){
printf(" ");
}else{
printf("\n");
}
}
if(isC){
//遍历所有的点查看度的情况
for(int i=1;i<=n;i++){
if(degree[i]%2 == 1){
isE = false;
cnt++;
}
}
if(isE == true){
printf("Eulerian\n");
}else{
if(cnt == 2){
printf("Semi-Eulerian\n");
}else{
printf("Non-Eulerian\n");
}
}
}else{
printf("Non-Eulerian\n");
}
return 0;
}
码后反思
- 仔细读题,题目不难。
二刷代码
刚开始忘记判断是不是连通图了
//数组下标从1开始
#include <iostream>
#include <algorithm>
using namespace std;
const int maxn = 550;
const int inf = 0x3fffffff;
int n,m;
int d[maxn];
bool G[maxn][maxn];
bool vis[maxn]={false};
int cnt=0;
void dfs(int u){
vis[u]=true;
cnt++;
for(int v=1;v<=n;v++){
if(G[u][v]!=false&&vis[v]==false){
dfs(v);
}
}
}
int main(){
scanf("%d %d",&n,&m);
fill(d,d+maxn,0);
fill(G[0],G[0]+maxn*maxn,false);
for(int i=0;i<m;i++){
int u,v;
scanf("%d %d",&u,&v);
G[u][v]=true;
G[v][u]=true;
d[u]++;
d[v]++;
}
//需要判断连通性
dfs(1);
bool isAllEven=true;
int cntOdd=0;
for(int i=1;i<=n;i++){
printf("%d",d[i]);
if(i!=n){
printf(" ");
}else{
printf("\n");
}
if(d[i]&1==1){
isAllEven=false;
cntOdd++;
}
}
if(cnt==n){//是连通图
if(isAllEven){
printf("Eulerian\n");
}else if(cntOdd==2){
printf("Semi-Eulerian\n");
}else{
printf("Non-Eulerian\n");
}
}else{
printf("Non-Eulerian\n");
}
return 0;
}