C++回溯算法---图的m着色问题01

淩亂°似流年 2024-03-23 11:26 174阅读 0赞

C++回溯算法—-图的m着色问题

图的m着色问题是指给定一个图以及m种不同的颜色,尝试将每个节点涂上其中一种颜色,使得相邻的节点颜色不相同。这个问题可以转化为在解空间树中寻找可行解的问题,其中每个分支结点都有m个儿子结点,最底层有m的n次方个叶子结点。算法的思路是在解空间树中做深度优先搜索,并使用约束条件来剪枝优化搜索。

代码:

  1. #include <iostream>
  2. #include <cstring>
  3. /*
  4. * 图的m着色问题
  5. */
  6. using namespace std;
  7. const int maxn = 1005;
  8. int G[maxn][maxn], color[maxn];//用于存储图中的边--用于存储每个节点的颜色
  9. int n, m; //n表示图中节点的数量,m表示可供选择的颜色数目。
  10. bool ok(int u, int c) {
  11. for (int i = 1; i <= n; i++) {
  12. if (G[u][i] == 1 && color[i] == c)
  13. return false;
  14. }
  15. return true;
  16. }
  17. bool dfs(int u) {
  18. if (u > n)
  19. return true;
  20. for (int i = 1; i <= m; i++) {
  21. if (ok(u, i)) {
  22. color[u] = i;
  23. if (dfs(u + 1))
  24. return true;
  25. color[u] = 0;
  26. }
  27. }
  28. return false;
  29. }
  30. int main() {
  31. memset(G, 0, sizeof(G));//初始化邻接矩阵为0
  32. memset(color, 0, sizeof(color));//初始化颜色为0
  33. int e;//e表示图中边的数量
  34. cin >> n >> e >> m;
  35. for (int i = 0; i < e; i++) {
  36. int u, v;
  37. cin >> u >> v;
  38. G[u][v] = G[v][u] = 1;
  39. }
  40. if (dfs(1)) {
  41. cout << "Yes" << endl;
  42. for (int i = 1; i <= n; i++) {
  43. cout << "结点 " << i << " 的颜色为: " << color[i] << endl;
  44. }
  45. }
  46. else {
  47. cout << "No" << endl;
  48. }
  49. return 0;
  50. }

分析:

这个算法的实现包括两个主要步骤:

  1. 判断颜色是否符合要求

对于每个节点 u,如果它与另一个节点 v 有边相连,且这两个节点颜色相同,那么就不能把节点 u 涂为该颜色。因此,需要定义一个函数 ok() 来判断某个节点染上某种颜色是否符合要求。具体来说,ok(u, c) 函数返回值为true表示节点 u 可以涂上颜色 c,否则返回false。

2.使用深度优先搜索

使用深度优先搜索(DFS)从解空间树的根节点开始搜索,并在每个分支结点处调用 ok() 函数来剪枝。如果在整棵解空间树中找到了一组可行解,那么算法就停止搜索并输出结果。如果找不到任何一个可行解,则算法输出无解信息。

具体实现过程:

首先,需要定义一个二维数组 G[ ][ ],用于存储图中的边。其中,G[u][v] == 1 表示节点 u 和节点 v 之间有边相连,反之为 0。同时,还需要定义一个一维数组 color[ ],用于存储每个节点的颜色。

首先将所有边权赋值为 0,即不存在边。然后,读入所有边,将对应的边权赋值为 1。读入颜色数 m,并从节点 1 开始做深度优先搜索,依次尝试给每个节点涂上不同的颜色。在每个分支结点处,使用 ok() 函数来判断是否符合要求。如果染色成功,则继续对下一个节点做深度优先搜索。如果找到了一组可行解,则输出结果。

运行结果:

93997af14e6f49429b94a117812ab129.png


问题描述

 给定无向连通图G=(V, E)和m种不同的颜色,用这些颜色为图G的各顶点着色,每个顶点着一种颜色。是否有一种着色法使G中相邻的两个顶点有不同的颜色。这个问题是图的m可着色判定问题。若一个图最少需要m种颜色才能使图中每条边连接的两个顶点着不同颜色,则称这个数m为该图的色数。求一个图的色数m的问题称为图的m可着色优化问题。

例如:点个数n=7,颜色m=3的涂色方案

7f74bd29f1cc400da250202134958f84.png

算法设计

 一般连通图的可着色问题,不仅限于可平面图。

 给定图G=(V,E)和m种颜色,如果该图不是m可着色,给出否定回答;若m可着色,找出所有不同着色方法。

算法思路
设图G=(V, E), |V|=n, 颜色数= m, 用邻接矩阵a表示G, 用整数1, 2…m来表示

m种不同的颜色。顶点i所着的颜色用x[i]表示。

问题的解向量可以表示为n元组x={ x[1],…,x[n] }. x[i]∈{1,2,…,m},

解空间树为排序树,是一棵n+1层的完全m叉树.

在解空间树中做深度优先搜索, 约束条件:如果a[j][i]=1 , x[i] ≠ x[j]
f2a74a4375d4421bb141f18c3648827a.png c41a8018718c4944bba5e19a300a3379.png

m=3,n=3时的解空间树

61151aaa904845e7836ca244332d81dd.png

再举个例子

对于下图,写出图着色算法得出一种着色方案的过程。

顶点数量n=4, 色数:m=3

831d324ffc014a838122ff50b68e6f74.png

m=4,n=3时的解空间树

0cf24ed41f614bfaa436cc66969b49a4.png

X[1]←1 , 返回 true
X[2]←1, 返回false; X[2]←2, 返回 true
X[3]←1 ,返回false; X[3]←2, 返回false;X[3]←3, 返回 true
X[4]←1, 返回false; X[4]←2, 返回false;X[4]←3, 返回 true

着色方案:(1,2,3,3)

复杂度分析

图m可着色问题的解空间树中,内结点个数是:

88323978e7ed492fb40f22262082d75c.png

对于每一个内结点,在最坏情况下,用ok检查当前扩展结点每一个儿子的颜色可用性需耗时O(mn)。

因此,回溯法总的时间耗费是

在这里插入图片描述

  1. #include"图的着色.h"
  2. const int NUM = 100;
  3. int n;//顶点数
  4. int m;//颜色数
  5. int a[NUM][NUM];//图的邻接矩阵
  6. int x[NUM];//当前的解向量
  7. int sum = 0;//解的数量
  8. bool same(int t) {
  9. int i;
  10. for (i = 1; i < n; i++) {//修正同色判断函数的循环范围
  11. if ((a[t][i] == 1) && (x[i] == x[t]))
  12. return false;
  13. }
  14. return true;
  15. }
  16. void BackTrack(int t) {
  17. int i;
  18. if (t > n) {
  19. sum++;
  20. cout << "解" << sum << ":" << endl;//输出当前解的编号
  21. for (i = 1; i <= n; i++) {
  22. cout << x[i] << " ";//按照节点顺序输出颜色
  23. }
  24. cout << endl;
  25. }
  26. else
  27. {
  28. for (i = 1; i <= m; i++) {
  29. x[t] = i;
  30. if (same(t))
  31. BackTrack(t + 1);
  32. x[t] = 0;
  33. }
  34. }
  35. }
  36. int main() {
  37. cin >> n >> m;
  38. //初始化邻接矩阵为0
  39. for (int i = 0; i < n; i++) {
  40. for (int j = 0; j < n; j++) {
  41. a[i][j] = 0;
  42. }
  43. }
  44. //读入边,构建邻接矩阵
  45. int u, v;
  46. while (cin >> u >> v) {
  47. if (u < 1 || u > n || v < 1 || v > n) {//判断输入是否合法
  48. cout << "输入不合法!" << endl;
  49. continue;
  50. }
  51. a[u - 1][v - 1] = 1;
  52. a[v - 1][u - 1] = 1;
  53. }
  54. BackTrack(1);//从第1个节点开始
  55. cout << "一共有" << sum << "个解。" << endl;//输出解的数量
  56. return 0;
  57. }

发表评论

表情:
评论列表 (有 0 条评论,174人围观)

还没有评论,来说两句吧...

相关阅读

    相关 m着色方案

    图的m着色 描述 Description 【问题描述】   给定无向连通图G和m种不同的颜色。用这些颜色为图G的各顶点着色,每个顶点着一种颜色。如果有一种着色法使G

    相关 m着色问题

      问题描述        给定无向连通图G和m种不同的颜色。用这些颜色为图G的各顶点着色,每个顶点着一种颜色。是否有一种着色法使G中每条边的2个顶点着不同颜色。这个问题是图