八皇后问题的遗传算法实现过程详解
1、八皇后问题描述
19 世纪著名的数学家Gauss 在1850 年提出八皇后问题后, 该问题成为各类语言程序设计的经典题目。八皇后问题要求在8×8 格的国际象棋上摆放八个皇后,使横、竖、斜方向上都不能有两个及两个以上皇后在同一条直线上, 问题也可以推广到N 个皇后。穷举法在问题规模不大的情况下还可适用,回溯法是求解此问题的经典算法。但N 皇后问题是个NP 难问题, 随着皇后数目的增多, 求解复杂度激增, 就需利用非常规的技术求
解。遗传算法在求解一些NP 完全问题上得到了广泛地应用,本文用遗传算法求解八皇后问题,给出详细的实现过程。
2、基本遗传算法求解过程
基本遗传以初始种群为基点, 经过选择、交叉、变异操作生成新的种群,如此更新种群直到满足终止条件。其计算步骤如下:
(1) 将问题空间转换为遗传空间, 也就是编
码;
(2)随机生成P 个染色体作为初始种群;
(3)染色体评价,也就是按确定的适应度函数
计算各个染色体的适应度;
(4)根据染色体适应度,按选择算子进行染色
体的选择;
(5)按交叉概率进行交叉操作;
(6)按变异概率进行变异操作;
(7)返回(4)形成新的种群,继续迭代,直到满足终止条件。
基本遗传算法给出了基本框架, 针对求解的问题不同, 遗传算法在相应的计算步骤中有不同的设计。本文针对八皇后问题, 设计了相应的编码,适应度计算方法,交叉和变异操作。
3、用遗传算法求解八皇后问题实现过程详解
3.1 编码
遗传算法中传统的编码是二进制编码, 本文采用多值编码。染色体长度取决于皇后的个数。染色体中每个基因所处的位置表示其在棋谱中所在的行数, 基因值表示其所在的列数。如染色体40752613 表示:从0 开始数,第0 个4 表示在第零行的皇后在第4 列, 第1 个0 表示第一行的皇后在第0 列,以此类推。八皇后问题中皇后不能处于同行同列, 意味着染色体中0~7 的基因取值不能出现重复。
3.2 个体评价
染色体通常表示了问题的可行解, 对可行解进行遗传操作寻找最优解。但在八皇后问题中,染色体仅仅体现同行同列中未出现互攻, 在对角线上是否出现互攻还未做考虑。在对皇后的位置做比较的时候, 可以对两个棋子的行数差与列数差进行对比, 实现了互攻次数的统计。公式为:|绝对值((y2-y1)/(x2-x1)) |=1。公式中(x1,y1),(x2,y2)分别表示两个皇后所在的位置,即所在的行数和列数。当两个皇后的行数差与列数差比值的绝对值为1 的时候,两皇后在同一对角线上,即出现了互攻。每个染色体内的互攻次数为Value,初始值设为0;第0 行与1~7 行进行比较, 每出现一次互攻则Value 的值增加1;第1 行与2~7 行进行比较,以此类推来计算Value 值。当Value 为0 表示没有发生互攻,此染色体就是其中的一个可行解。当Value 不为0则进行适应度的计算。一般来说, 适应度越大越
好,而互攻次数为越小越好,所以可以将适应度的计算函数设置为:F=28-Value。
3.3 选择
选择使用的是经典的赌轮选择方法, 与基本遗传算法的实现无特别之处,此处不赘述。
3.4 交叉
经典的单点, 多点等交叉因染色体中不能出现重复的基因值,在该问题中不适用。本文使用部分匹配交叉,具体操作如下:
1)在染色体中随机选取两个点标记为y,
如:染色体a:01y24y3675;
染色体b:12y30y4576;
两个y 之间的基因段称为中间段, 记录其对应关系2-3,4-0;
2)对染色体a,b 的中间段进行交换,
形成染色体a':01y30y3675;染色体b': 12y24y4576;
3) 利用对应关系,对染色体a', b' 中间段外的基因进行交换,
形成 染色体a'': 41y30y2675;
染色体b'': 13y24y0576;
交叉完成。
3.5 变异
采用多值编码后, 变异操作并不能通过简单的0,1 反转实现。
本文采取随机地选取染色体内的两个基因进行交换来实现。
例如随机选取的是
6 和1 两个基因,那么
变异前染色体: 7 (6) 5 4 3 2 (1) 0
变异后染色体: 7 (1) 5 4 3 2 (6) 0
3.6 终止策略
本文采用的终止策略为: 当群体中出现染色体的适应值为0 时, 即表示算法搜索到了一个可行解,终止算法。若算法运行设置的代数还未找到可行解,同样终止程序运行。
4、总结
本文详细介绍了用遗传算法求解八皇后问题的求解过程, 但要注意的是这只是其中的一种编码,交叉,变异等操作设计方法,还有许多其他的方法可以选择。对于各操作采取不同设计方案的遗传算法,其算法性能值得比较讨论。
1 #include <stdio.h> 2 #include <stdlib.h> 3 #include <time.h> 4 #include <stdbool.h> 5 #include <math.h> 6 7 // 8 // 编程题 9 // 遗传算法(八皇后问题) 10 // 11 12 13 #define N 8 //皇后数 14 #define Cluster_size 12//默认种群大小 15 #define LASTG 100 /*终止后代*/ 16 #define MRATE 0.8 /*突变的概率*/ 17 int array[Cluster_size][N];//染色体集合 18 int copyarray[Cluster_size][N];;//array的副本 19 int narray[Cluster_size * 2][N];//下一代染色体集合 20 int values[Cluster_size];//评估数组 21 int max_array[N];//保存最佳数组 22 int generation = 0; //记录代数 23 int signal = -1;//信号 24 int max=-1;//记录当前最优值 25 int max_generation;//记录当前最优值代数 26 27 28 struct MyStruct 29 { 30 int key[Cluster_size]; 31 int values[Cluster_size]; 32 } rember; //交叉时记录对应关系 33 34 35 /* 36 1、八皇后问题描述 37 19 世纪著名的数学家Gauss 在1850 年提出 38 八皇后问题后, 该问题成为各类语言程序设计的 39 经典题目。八皇后问题要求在8×8 格的国际象棋 40 上摆放八个皇后,使横、竖、斜方向上都不能有两 41 个及两个以上皇后在同一条直线上, 问题也可以 42 推广到N 个皇后。穷举法在问题规模不大的情况 43 下还可适用,回溯法是求解此问题的经典算法。但 44 N 皇后问题是个NP 难问题, 随着皇后数目的增 45 多, 求解复杂度激增, 就需利用非常规的技术求 46 解。遗传算法在求解一些NP 完全问题上得到了 47 广泛地应用,本文用遗传算法求解八皇后问题,给 48 出详细的实现过程。 49 */ 50 51 52 /* 基本遗传算法求解过程 53 基本遗传以初始种群为基点, 经过选择、交 54 叉、变异操作生成新的种群,如此更新种群直到满 55 足终止条件。其计算步骤如下: 56 (1) 将问题空间转换为遗传空间, 也就是编 57 码; 58 (2)随机生成P 个染色体作为初始种群; 59 (3)染色体评价,也就是按确定的适应度函数 60 计算各个染色体的适应度; 61 (4)根据染色体适应度,按选择算子进行染色 62 体的选择; 63 (5)按交叉概率进行交叉操作; 64 (6)按变异概率进行变异操作; 65 (7)返回(4)形成新的种群,继续迭代,直到满 66 足终止条件。*/ 67 68 //用遗传算法求解八皇后问题实现过程详解 69 70 71 /*1 编码 72 遗传算法中传统的编码是二进制编码, 本文 73 采用多值编码。染色体长度取决于皇后的个数。染 74 色体中每个基因所处的位置表示其在棋谱中所在 75 的行数, 基因值表示其所在的列数。如染色体 76 40752613 表示:从0 开始数,第0 个4 表示在第 77 零行的皇后在第4 列, 第1 个0 表示第一行的皇 78 后在第0 列,以此类推。八皇后问题中皇后不能处 79 于同行同列, 意味着染色体中0~7 的基因取值不 80 能出现重复*/ 81 82 83 /*2 个体评价 84 染色体通常表示了问题的可行解, 对可行解 85 进行遗传操作寻找最优解。但在八皇后问题中,染 86 色体仅仅体现同行同列中未出现互攻, 在对角线 87 上是否出现互攻还未做考虑。在对皇后的位置做 88 比较的时候, 可以对两个棋子的行数差与列数差 89 进行对比, 实现了互攻次数的统计。公式为: 90 |(y2-y1)/(x2-x1)|=1。公式中(x1,y1),(x2,y2)分别表示两个 91 皇后所在的位置,即所在的行数和列数。当两个皇 92 后的行数差与列数差比值的绝对值为1 的时候, 93 两皇后在同一对角线上,即出现了互攻。每个染色 94 体内的互攻次数为Value,初始值设为0;第0 行 95 与1~7 行进行比较, 每出现一次互攻则Value 的 96 值增加1;第1 行与2~7 行进行比较,以此类推来 97 计算Value 值。当Value 为0 表示没有发生互攻, 98 此染色体就是其中的一个可行解。当Value 不为0 99 则进行适应度的计算。一般来说, 适应度越大越 100 好,而互攻次数为越小越好,所以可以将适应度的 101 计算函数设置为:F=1/Value。*/ 102 103 104 105 106 107 /*编码方案 108 遗传算法的常用编码方案有排列编码、二进制编码、实值编码等,考虑到编码方案需要较好地对应一种棋盘皇后排列顺序, 109 同时八皇后问题需要解决的是实体目标(即皇后)的排列问题,不涉及到浮点数层面的逼近问题,本程序采用排列编码作为编码方案,具体描述如下: 110 用一维n元数组x0,1…,n-1来表示一个个体,其中x[i]∈{0,1…,n-1},x[i]表示皇后i放在棋盘的第i行第x[i]列,即第i行第x[i]列放置一个皇后。 111 例如,x[0]=0表示棋盘的第0行第0列放一个皇后。数组第i个元素表示第i行的放置情况,可以看作一个基因。 112 这种编码可以自然的解决了某一行只能放一个皇后的约束,如果数组的每一个元素x[i]都不重复,可以看成0 — 7的一种排列,就自然保证每一列只能放一个皇后。 113 在编码方案实现过程中,我们对染色体编码是否可重复问题进行了讨论,经实验发现不允许染色体编码重复的方案, 114 即每一个编码必须为0-7排列的编码方案的搜索空间为8!种,而允许编码出现重复的方案搜索空间为8^8种, 115 在实验运行过程中第一种编码方案运行所需代数平均值比第二种方案要小一个数量级,因此最终决定采用了不允许染色体编码重复的方案, 116 n元数组x[0,1…,n-1]中,每一个x[i]不允许重复,在程序运行过程中,无论是初始种群生成,选择,交叉,变异阶段都会维护这一编码准则。 117 */ 118 119 120 /*初始种群 121 初始化种群的主要工作为:确定种群的大小及产生初始种群.种群的大小能够对遗传算法的收敛性产生很大的影响, 122 种群较小算法收敛速度较快,但它的搜索面不够广,容易导致局部收敛;而种群较大算法收敛全局最优的概率也大, 123 但是算法的收敛速度较慢。根据N皇后问题的特点,默认初始种群大小一般为N ~ 2N (N为皇后数)。经多次实验, 124 初始种群大小的设置从8、12、16、24没有较大区别,对遗传代数没有突出影响,因此将默认初始种群大小设为12,界面中提供了用户自行设定初始种群大小的方法。 125 除此之外,程序提供了两种初始种群的生成方法,第一种是随机生成,程序将自动生成用户设定数量的随机个体作为初始种群, 126 第二种是用户输入,程序初始种群一栏提供了用户自行输入每一个个体基因编码的接口。值得注意的是,以上无论哪种方法, 127 生成的都是0-7不含重复的8位随机排列。系统不会接受任何编码出现重复的非法个体。 128 */ 129 130 131 132 int rndn(int l) 133 { 134 int rndno; 135 rndno = rand()%l; 136 return rndno; 137 138 } 139 140 141 void output_copy_array() 142 { 143 for (int i = 0; i < Cluster_size; i++) 144 { 145 for (int j = 0; j < N; j++) 146 printf("%d", copyarray[i][j]); 147 printf(" "); 148 } 149 } 150 151 152 153 void output_maxarray() 154 { 155 for (int i = 0; i < N; i++) 156 printf("%d", max_array[i]); 157 } 158 159 160 161 162 163 164 //判断是否有最优解 165 int the_answer(int* values, int size) 166 { 167 for (int i = 0; i < size; i++) 168 if (values[i] == 28) 169 return i; 170 return -1; 171 } 172 173 174 175 int judge(int a[], int n) 176 { 177 int i, j; 178 int value = -1; 179 for (i = 0; i < n; i++) { 180 value = a[i]; 181 for (j = i + 1; j < n; j++) { 182 if ((value == a[j])) { 183 return 0; 184 } 185 } 186 } 187 return 1; 188 189 } 190 191 192 193 194 195 //计算初始适应度值 196 void count_collidecount() //适应度函数设置为:value[i]=28-value 197 { 198 199 int i, j, x1, x2, y1, y2, m, value = 0; 200 for (i = 0; i < Cluster_size; i++) { 201 202 for (j = 0; j < N; j++) 203 { 204 x1 = j; 205 y1 = array[i][j]; 206 for (m = j + 1; m < N; m++) 207 { 208 x2 = m; 209 y2 = array[i][m]; 210 if (abs((y2 - y1) / (x2 - x1))==1) 211 value++; 212 } 213 214 } 215 values[i] = 28 - value; 216 value = 0; 217 } 218 219 signal = the_answer(values, Cluster_size); 220 221 } 222 223 224 //计算子代适应度值 225 void count_generation_collidecount(int* values, int cluster_size) 226 { 227 228 int i, j, x1, x2, y1, y2, m, value = 0; 229 for (i = 0; i < cluster_size; i++) { 230 231 for (j = 0; j < N; j++) 232 { 233 x1 = j; 234 y1 = narray[i][j]; 235 for (m = j + 1; m < N; m++) 236 { 237 x2 = m; 238 y2 = narray[i][m]; 239 if (abs((y2 - y1) / (x2 - x1))==1) 240 value++; 241 } 242 243 } 244 values[i] = 28 - value; 245 value = 0; 246 } 247 248 249 } 250 251 252 253 254 255 /************************/ 256 /* selectp()函数 */ 257 /* 父代的选择 */ 258 /************************/ 259 int selectp(int roulette[], int totalfitness) 260 { 261 int i;/* 循环的控制变量 */ 262 int ball;/* 球(选择位置的数值)*/ 263 int acc = 0;/*评价值的累积值*/ 264 265 ball = rndn(totalfitness); 266 for (i = 0; i < Cluster_size; i++) { 267 acc += roulette[i];/*评价值的累积*/ 268 if (acc > ball) break;/*对应的基因*/ 269 } 270 return i; 271 } 272 273 274 275 276 277 278 bool takeoutrepeat(int position)//去除有重复元素的数组,并添加无重复值的数组(染色体) 279 { 280 281 int i; 282 int value; 283 bool signal = true; 284 285 286 for (i = 0; i < N; i++) 287 { 288 value = narray[position*2][i]; 289 for (int j = i + 1; j < N; j++) 290 if (narray[position*2][j] == value) 291 { 292 printf("there have reapt number: %d ", (position*2)); 293 signal = false; 294 } 295 296 } 297 298 for (i = 0; i < N; i++) 299 { 300 value = narray[position * 2+1][i]; 301 for (int j = i + 1; j < N; j++) 302 if (narray[position * 2+1][j] == value) 303 { 304 printf("there have reapt number: %d ", (position*2+1)); 305 signal = false; 306 } 307 } 308 309 return signal; 310 311 312 } 313 314 315 //判断两个数组是否相等 316 bool judge_reapt(int c,int cluster) 317 { 318 319 int i,j; 320 int value=0; 321 bool arraysEqual = true; 322 323 for (i=0,j = 0; j < cluster; j++) 324 { 325 while (arraysEqual && i < N) 326 { 327 if (narray[c][i] != copyarray[j][i]) 328 arraysEqual = false; 329 i++; 330 } 331 //显示合适的消息 332 if (arraysEqual) 333 value++; 334 else 335 arraysEqual = true; 336 337 i = 0;//i置0 338 } 339 340 if (value > 0) 341 return false; 342 else 343 return true; 344 345 } 346 347 348 /************************/ 349 /* selection()函数 */ 350 /* 选择下一代 */ 351 /************************/ 352 void selection() 353 { 354 int i, j, c;/* 循环控制参数 */ 355 int totalfitness = 0;/*适应度的总计值*/ 356 int roulette[Cluster_size * 2];/*存放适应度*/ 357 int ball;/* 球(选择位置的数值)*/ 358 int acc = 0;/*适应度的累积值*/ 359 bool judge; 360 361 /*循环进行选择*/ 362 for (i = 0; i < Cluster_size; i++) { 363 /* 生成轮盘 */ 364 totalfitness = 0; 365 count_generation_collidecount(roulette, Cluster_size * 2); 366 367 for (c = 0; c < Cluster_size * 2; ++c) { 368 /*计算适应度的总计值*/ 369 totalfitness += roulette[c]; 370 } 371 signal = the_answer(roulette, Cluster_size * 2);//判断是否有最优解 372 373 do 374 { 375 /*选择一个染色体*/ 376 ball = rndn(totalfitness); 377 acc = 0; 378 for (c = 0; c < Cluster_size * 2; ++c) { 379 acc += roulette[c];/*累积评价值*/ 380 if (acc > ball) break;/*对应的基因*/ 381 } 382 judge= judge_reapt(c, Cluster_size ); 383 384 } while (judge==false); 385 386 387 /*染色体的复制*/ 388 for (j = 0; j < N; ++j) { 389 array[i][j] = narray[c][j]; 390 } 391 } 392 393 for (int q = 0; q< Cluster_size *2; q++) 394 { 395 if (roulette[q] > max) 396 { 397 max = roulette[q]; 398 max_generation = generation; 399 for (int i = 0; i < N; i++) 400 max_array[i] = narray[q][i]; 401 } 402 printf("%3.1d", roulette[q]); 403 404 } 405 printf(" "); 406 407 } 408 409 410 int judgein(int m, int location1, int location2) 411 { 412 for (int i = location1; i <= location2; i++) 413 if ((m == rember.key[i]) | (m == rember.values[i])) 414 return i; 415 return -1; 416 } 417 418 419 420 421 /************************/ 422 /* crossing()函数 */ 423 /* 特定2染色体的交叉 */ 424 /************************/ 425 void crossing(int mama, int papa, int position) 426 { 427 bool signal; 428 int cp1;/*交叉的点*/ 429 int cp2;/*交叉的点*/ 430 int location1; 431 int location2; 432 printf("mama=%d,papa=%d ", mama, papa); 433 434 do{ 435 436 /*确定交叉点*/ 437 do 438 { 439 cp1 = rndn(N); 440 cp2 = rndn(N); 441 // m = abs((cp2 - cp1)); 442 } while (cp1 == cp2); 443 //printf("%d,%d ", cp1, cp2); 444 if (cp1 < cp2) 445 { 446 location1 = cp1; 447 location2 = cp2; 448 } 449 else 450 { 451 location1 = cp2; 452 location2 = cp1; 453 } 454 455 456 for (int i = location1; i <= location2; i++) 457 { 458 rember.key[i] = array[mama][i]; 459 rember.values[i] = array[papa][i]; 460 //交换中间段 461 narray[position * 2][i] = array[papa][i]; 462 narray[position * 2 + 1][i] = array[mama][i]; 463 } 464 //利用对应关系,对染色体mama和papa, 中间段外的基因进行交换 465 /*交换前半部分*/ 466 for (int j = 0; j < location1; j++) 467 { 468 int weizhi = judgein(array[mama][j], location1, location2); 469 // printf("weizhi=%d ", weizhi); 470 if (weizhi == -1) 471 { 472 narray[position * 2][j] = array[mama][j]; 473 } 474 else 475 { 476 if (array[mama][j] == rember.key[weizhi]) 477 narray[position * 2][j] = rember.values[weizhi]; 478 else 479 narray[position * 2][j] = rember.key[weizhi]; 480 } 481 482 weizhi = judgein(array[papa][j], location1, location2); 483 if (weizhi == -1) 484 { 485 narray[position * 2 + 1][j] = array[papa][j]; 486 } 487 else 488 { 489 if (array[papa][j] == rember.key[weizhi]) 490 narray[position * 2 + 1][j] = rember.values[weizhi]; 491 else 492 narray[position * 2 + 1][j] = rember.key[weizhi]; 493 } 494 495 } 496 497 498 ///*交换后半部分*/ 499 for (int j = location2 + 1; j < N; j++) 500 { 501 int weizhi = judgein(array[mama][j], location1, location2); 502 if (weizhi == -1) 503 { 504 narray[position * 2][j] = array[mama][j]; 505 } 506 else 507 { 508 if (array[mama][j] == rember.key[weizhi]) 509 narray[position * 2][j] = rember.values[weizhi]; 510 else 511 narray[position * 2][j] = rember.key[weizhi]; 512 } 513 514 weizhi = judgein(array[papa][j], location1, location2); 515 if (weizhi == -1) 516 { 517 narray[position * 2 + 1][j] = array[papa][j]; 518 } 519 else 520 { 521 if (array[papa][j] == rember.key[weizhi]) 522 narray[position * 2 + 1][j] = rember.values[weizhi]; 523 else 524 narray[position * 2 + 1][j] = rember.key[weizhi]; 525 526 } 527 } 528 529 signal = takeoutrepeat(position); 530 printf(" --------------signal=%d-------------- ",signal); 531 } while (signal==false); 532 533 534 } 535 536 537 /************************/ 538 /* notval()函数 */ 539 /* */ 540 /************************/ 541 void notval(int i) 542 { 543 int position1; 544 int position2; 545 int temp; //两个基因点交换的中间变量 546 do 547 { 548 position1 = rndn(N); 549 position2 = rndn(N); 550 551 } while (position2 == position1); //当两个变异基因点相同时,循环。 552 553 temp = narray[i][position2]; 554 narray[i][position2] = narray[i][position1]; 555 narray[i][position1] = temp; 556 557 } 558 559 560 561 /***********************/ 562 /* mutation()函数 */ 563 /* 突变 */ 564 /***********************/ 565 void mutation() 566 { 567 int i;/* 循环的控制变量 */ 568 569 for (i = 0; i < Cluster_size * 2; ++i) 570 if ((rndn(100) / 100)<= MRATE) 571 /*染色体突变*/ 572 notval(i); 573 printf(" mutation is complete "); 574 575 } 576 577 578 579 580 void mating() 581 { 582 583 int i; 584 int totalfitness = 0; 585 int roulette[Cluster_size];//存放评价值 586 int mama, papa; //父代的基因的号码 587 588 // 生成轮盘 589 for (i = 0; i < Cluster_size; ++i) 590 { 591 roulette[i] = values[i]; 592 totalfitness += roulette[i]; 593 } 594 595 //选择和交叉的循环 596 for (i = 0; i < Cluster_size; i++) 597 { 598 do { //父代的选择 599 mama = selectp(roulette, totalfitness); 600 papa = selectp(roulette, totalfitness); 601 } while (mama == papa); 602 603 //特定2染色体的交叉 604 crossing(mama, papa, i); 605 } 606 607 } 608 609 610 611 612 613 614 void outputrember() 615 { 616 for (int i = 0; i < N; i++) 617 printf("key=%d,values=%d ", rember.key[i], rember.values[i]); 618 } 619 620 621 void outputnarray() 622 { 623 for (int i = 0; i < Cluster_size * 2; i++) 624 { 625 if (i % 2 == 0) 626 printf("--------------------------------------------------- "); 627 for (int j = 0; j < N; j++) 628 printf("%d", narray[i][j]); 629 printf(" "); 630 } 631 632 } 633 634 void copy_array() 635 { 636 637 for (int i = 0; i < Cluster_size; i++) 638 for (int j = 0; j < N; j++) 639 { 640 copyarray[i][j] = array[i][j]; 641 } 642 } 643 644 645 void outputarray() 646 { 647 for (int i = 0; i < Cluster_size; i++) 648 { 649 for (int j = 0; j < N; j++) 650 printf("%d", array[i][j]); 651 printf(" "); 652 } 653 654 } 655 656 657 void output() 658 { 659 int i; 660 for (i = 0; i < Cluster_size; i++) 661 { 662 if (values[i] > max) 663 { 664 max = values[i]; 665 max_generation = generation; 666 } 667 printf("%3.1d", values[i]); 668 669 } 670 printf(" "); 671 } 672 673 674 675 676 677 void init_Cluster() 678 { 679 int a[8]; 680 int x, y; 681 int count = 0; 682 683 for (; count < Cluster_size; count++) 684 { 685 686 for (y = 0; y < 8; y++) 687 { 688 x = rand() % 8; 689 a[y] = x; 690 } 691 692 if (judge(a, 8)) 693 { 694 for (int i = 0; i < 8; i++) 695 { 696 array[count][i] = a[i]; 697 } 698 } 699 else 700 --count; 701 702 } 703 704 } 705 706 707 708 709 int main() 710 { 711 srand((int)time(NULL)); //随机种子 712 init_Cluster(); 713 714 for (; generation < LASTG; generation++) 715 { 716 717 if (signal != -1) 718 break; 719 else 720 { 721 722 printf(" %d代数 ", generation); 723 count_collidecount(); 724 printf("-------------output------------values---------------------------- "); 725 output(); 726 printf("-------------outputarray---------------------------------------- "); 727 outputarray(); 728 mating(); 729 printf("------------- outputarray---------------------------------------- "); 730 outputarray(); 731 printf("-----------------mating选择交叉---------outputnarray--------------------------- "); 732 outputnarray(); 733 mutation(); 734 printf("---------- mutation-变异------------outputnarray------------------------------ "); 735 outputnarray(); 736 printf("-------------outputarray---------------------------------------- "); 737 outputarray(); 738 copy_array(); 739 printf("-------------------------------------------------------------------- "); 740 output_copy_array(); 741 printf("------------selection------outputarray下一代种群------------------------------ "); 742 selection(); 743 outputarray(); 744 } 745 } 746 747 printf(" signal = %d, max = %d, max_generation = %d ", signal,max,max_generation); 748 output_maxarray(); 749 return 0; 750 751 752 }