题目:

小sun最近为了应付考试,正在复习图论,他现在学到了图的遍历,觉得太简单了,于是他想到了一个更加复杂的问题:无向图有n个点,从点1开始遍历,但是规定:按照每次“走两步”的方式来遍历整个图。可以发现按照每次走两步的方法,不一定能够遍历整个图,所以现在小sun想问你,最少加几条边,可以完整的遍历整个图。


做法:

两步走不难往二分图上想。如果一幅图是二分图,将图分成2个集合。从1出发只能走到1所在集合中的点。而只要该图不是二分图,必定有一条边连接同一个集合中的2个点,进而可以从一个集合走到另一个集合,也就能走完所有点。
所以这题做法是:先求出所有联通块。联通块数量-1条边是必须加的。然后看一下每个联通块是否有非二分图。若有就不需另加边。否则加一条边。


代码:

#include <bits/stdc++.h>
#define IOS ios::sync_with_stdio(false), cin.tie(0)
#define debug(a) cout << #a ": " << a << endl
using namespace std;
typedef long long ll;
const int N = 1e5 + 7;
int n, m, col[N], extra;
vector<int> g[N];
void dfs(int u, int c){
    col[u] = c;
    for (int i = 0; i < g[u].size(); ++i){
        int v = g[u][i];
        if (!col[v]) dfs(v, 3-c);
        if (col[v] == col[u]) extra = 0;
    }
}
int main(void){ 
    IOS;
    cin >> n >> m;
    for (int i = 1; i <= m; ++i){
        int u, v; cin >> u >> v;
        g[u].push_back(v);
        g[v].push_back(u);
    }
    extra = 1;
    int cnt = -1;
    for (int i = 1; i <= n; ++i){
        if (!col[i]){
            dfs(i, 1), cnt++;
        }
    }
    cout << cnt+extra << endl;
    return 0;
}