Java使用遗传算法,寻找十滴水问题的最优解

这篇具有很好参考价值的文章主要介绍了Java使用遗传算法,寻找十滴水问题的最优解。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

近期某手游出了个活动,经确认发现本质为十滴水游戏。

简单说一下规则,棋盘大小通常为6x6,在游戏开始时,棋盘随机有若干水珠,其大小范围为1-4。点击棋盘内的一格,会消耗玩家持有的1个小水滴,同时使得该单元格的水珠大小+1。如果水珠大小超过4,则水珠发生爆炸并消失,同时向四个方向各发射1个小水滴。小水滴匀速前进,前进时遇到第一个水珠则消失,同时该水珠大小+1,或者小水滴遇到棋盘边界而消失。当棋盘被清空时这一关通过,或者玩家持有的水滴耗尽而游戏结束。如果一次操作引发多个水珠爆炸,则每爆炸3个水珠,奖励1个水滴。如果只触发爆炸一次水珠通过这一关,则视为完美通关,额外奖励1个小水滴。对于该游戏,每次通关奖励2个小水滴。

查阅了相关资料后,得到了几条关键思路,即:

1、根据一个确定的盘面和一个坐标,通过计算,可以得到一个确定的执行结果,这样的操作称为一步。

2、一般不会在空白的单元格上设置小水滴,所以随机不停地在有水珠的单元格上添加小水滴,必然在有限的步骤后清空棋盘。

3、暴力遍历的话,由于状态空间太大,只要步数稍多(7步或更长),就不能在合理的时间内找到最优解。

所以,我们可以得到进一步的推论,我们可以使用一个有限且有序的点列表,解决一个盘面,并计算出该解法的小水滴的收益,期间最大投入小水滴的数量等结果。我们最终的目的是使得小水滴的收益最大化。

 因此,我决定使用遗传算法,解决本问题。

 

首先定义游戏关键对象:

Java使用遗传算法,寻找十滴水问题的最优解Java使用遗传算法,寻找十滴水问题的最优解
 1 package bean;
 2 
 3 import main.Game;
 4 
 5 import java.util.HashMap;
 6 import java.util.Map;
 7 
 8 /**
 9  * 位置点
10  */
11 public class Point {
12 
13     private static final Map<Integer, Map<Integer, Point>> POINT_MAP = new HashMap<>();
14 
15     /**
16      * X坐标
17      */
18     private final int x;
19 
20     /**
21      * Y坐标
22      */
23     private final int y;
24 
25     private Point(int x, int y) {
26         this.x = x;
27         this.y = y;
28     }
29 
30     /**
31      * 获取一个点对象
32      *
33      * @param x X坐标
34      * @param y Y坐标
35      * @return 生成的点对象
36      */
37     public static Point get(int x, int y) {
38         Map<Integer, Point> subPointMap = POINT_MAP.computeIfAbsent(x, i -> new HashMap<>());
39         return subPointMap.computeIfAbsent(y, i -> new Point(x, y));
40     }
41 
42     public int getX() {
43         return x;
44     }
45 
46     public int getY() {
47         return y;
48     }
49 
50     public boolean isValidPosition() {
51         return x >= 0 && y >= 0 && x < Game.X && y < Game.Y;
52     }
53 
54     @Override
55     public String toString() {
56         return String.format("(%s,%s)", x, y);
57     }
58 }
位置点对象
Java使用遗传算法,寻找十滴水问题的最优解Java使用遗传算法,寻找十滴水问题的最优解
 1 package bean;
 2 
 3 /**
 4  * 子弹对象,即爆裂发生时需要计算随时间变化的结果
 5  */
 6 public class Bullet {
 7 
 8     /**
 9      * X坐标
10      */
11     private int x;
12 
13     /**
14      * Y坐标
15      */
16     private int y;
17 
18     /**
19      * X变化率
20      */
21     private int dx;
22 
23     /**
24      * Y变化率
25      */
26     private int dy;
27 
28     public Bullet() {
29     }
30 
31     public Bullet(int x, int y, int dx, int dy) {
32         this.x = x;
33         this.y = y;
34         this.dx = dx;
35         this.dy = dy;
36     }
37 
38     public Bullet(Point point, int dx, int dy) {
39         this(point.getX(), point.getY(), dx, dy);
40     }
41 
42     public int getX() {
43         return x;
44     }
45 
46     public void setX(int x) {
47         this.x = x;
48     }
49 
50     public int getY() {
51         return y;
52     }
53 
54     public void setY(int y) {
55         this.y = y;
56     }
57 
58     public int getDx() {
59         return dx;
60     }
61 
62     public void setDx(int dx) {
63         this.dx = dx;
64     }
65 
66     public int getDy() {
67         return dy;
68     }
69 
70     public void setDy(int dy) {
71         this.dy = dy;
72     }
73 
74     /**
75      * 子弹移动一个回合
76      *
77      * @return 移动后的位置
78      */
79     public Point move() {
80         x += dx;
81         y += dy;
82         return Point.get(x, y);
83     }
84 }
子弹对象,即爆裂发生时需要计算随时间变化的结果
Java使用遗传算法,寻找十滴水问题的最优解Java使用遗传算法,寻找十滴水问题的最优解
 1 package bean;
 2 
 3 /**
 4  * 游戏的行动结果对象,会进行缓存
 5  */
 6 public class MoveResult {
 7 
 8     /**
 9      * 行动后的盘面
10      */
11     private int[][] arr;
12 
13     /**
14      * 行动后获得的收益(可以由combo计算出)
15      */
16     private int reward;
17 
18     /**
19      * 行动连击数量,即本次行动导致多少爆炸
20      */
21     private int combo;
22 
23     public MoveResult() {
24 
25     }
26 
27     public MoveResult(int[][] arr, int reward, int combo) {
28         this.arr = arr;
29         this.reward = reward;
30         this.combo = combo;
31     }
32 
33     public int[][] getArr() {
34         return arr;
35     }
36 
37     public void setArr(int[][] arr) {
38         this.arr = arr;
39     }
40 
41     public int getReward() {
42         return reward;
43     }
44 
45     public void setReward(int reward) {
46         this.reward = reward;
47     }
48 
49     public int getCombo() {
50         return combo;
51     }
52 
53     public void setCombo(int combo) {
54         this.combo = combo;
55     }
56 }
游戏的行动结果对象,会进行缓存
Java使用遗传算法,寻找十滴水问题的最优解Java使用遗传算法,寻找十滴水问题的最优解
 1 package bean;
 2 
 3 import java.util.ArrayList;
 4 import java.util.List;
 5 
 6 /**
 7  * 遗传算法使用的解答
 8  */
 9 public class Answer {
10 
11     /**
12      * 操作链,存储依次的执行操作
13      */
14     private final List<Point> pointList = new ArrayList<>();
15 
16     /**
17      * 爆炸次数
18      */
19     private int burstCount;
20 
21     /**
22      * 累计的收益
23      */
24     private int totalReward;
25 
26     /**
27      * 区间最低收益
28      */
29     private int minReward;
30 
31     public Answer() {
32     }
33 
34     public Answer(List<Point> pointList) {
35         this.pointList.addAll(pointList);
36     }
37 
38     public List<Point> getPointList() {
39         return pointList;
40     }
41 
42     public int getBurstCount() {
43         return burstCount;
44     }
45 
46     public void setBurstCount(int burstCount) {
47         this.burstCount = burstCount;
48     }
49 
50     public int getTotalReward() {
51         return totalReward;
52     }
53 
54     public void setTotalReward(int totalReward) {
55         this.totalReward = totalReward;
56     }
57 
58     public int getMinReward() {
59         return minReward;
60     }
61 
62     public void setMinReward(int minReward) {
63         this.minReward = minReward;
64     }
65 
66     /**
67      * 获取解的长度
68      *
69      * @return 解的长度
70      */
71     public int length() {
72         return pointList.size();
73     }
74 
75     @Override
76     public String toString() {
77         return String.format("%s, %s, %s, %s, %s", totalReward, burstCount, length(), minReward, pointList);
78     }
79 }
遗传算法使用的解答类

以及工具类

Java使用遗传算法,寻找十滴水问题的最优解Java使用遗传算法,寻找十滴水问题的最优解
  1 package util;
  2 
  3 import bean.Answer;
  4 import bean.MoveResult;
  5 import bean.Point;
  6 import main.Game;
  7 
  8 import java.util.ArrayList;
  9 import java.util.List;
 10 
 11 public class Utils {
 12 
 13     /**
 14      * 可视化一个Array(用于打印)
 15      *
 16      * @param game   游戏
 17      * @param answer 解答
 18      * @return 复制后的Array
 19      */
 20     public static String answerString(Game game, Answer answer) {
 21         StringBuilder builder = new StringBuilder();
 22 
 23         int[][] arr = game.getInitArr();
 24         int[][] moveArr = new int[Game.X][Game.Y];
 25         int moved = 0;
 26         int totalReward = 0;
 27         int minReward = 0;
 28         int burstCount = 0;
 29         StringBuilder moveBuilder = new StringBuilder();
 30         for (int i = 0; i < answer.length(); i++) {
 31             Point point = answer.getPointList().get(i);
 32             if (arr[point.getX()][point.getY()] <= 0) {
 33                 continue;
 34             }
 35 
 36             // 记录本次操作的结果,并更新棋盘状态
 37             MoveResult move = Game.move(arr, point);
 38             totalReward += move.getReward();
 39             if (minReward > totalReward) {
 40                 minReward = totalReward;
 41             }
 42             if (move.getCombo() > 0) {
 43                 burstCount++;
 44                 // 发生爆炸,则打印之前的行
 45                 if (moved > 0) {
 46                     answerStringArrLine(builder, moveBuilder, arr, moveArr);
 47                     builder.append("\r\n");
 48                     moveArr = new int[Game.X][Game.Y];
 49                     moved = 0;
 50                 }
 51             }
 52             arr = move.getArr();
 53             moveArr[point.getX()][point.getY()] += 1;
 54             moved++;
 55 
 56             builder.append(i).append(", Reward: ").append(move.getReward()).append(", TotalReward: ").append(totalReward).append(", Combo: ").append(move.getCombo()).append(", ").append(point);
 57             builder.append("\r\n");
 58             if (move.getCombo() > 0) {
 59                 answerStringArrLine(builder, moveBuilder, arr, moveArr);
 60                 builder.append("\r\n");
 61                 moveArr = new int[Game.X][Game.Y];
 62                 moved = 0;
 63             }
 64         }
 65 
 66         if (burstCount <= 1) {
 67             builder.append("Full Combo!").append("\r\n");
 68         } else {
 69             builder.append("BurstCount: ").append(burstCount).append("\r\n");
 70         }
 71 
 72         builder.append(answer.getTotalReward()).append(", ").append(answer.getBurstCount()).append(", ").append(answer.length()).append(", ").append(answer.getMinReward()).append(": ").append(moveBuilder).append("\r\n");
 73 
 74         return builder.toString();
 75     }
 76 
 77     private static void answerStringArrLine(StringBuilder builder, StringBuilder moveBuilder, int[][] arr, int[][] moveArr) {
 78         List<String> moveList = new ArrayList<>();
 79         for (int x = 0; x < arr.length; x++) {
 80             for (int y = 0; y < arr.length; y++) {
 81                 builder.append(arr[x][y]).append(", ");
 82             }
 83             builder.append("    ");
 84             for (int y = 0; y < arr.length; y++) {
 85                 builder.append(moveArr[x][y]).append(", ");
 86                 if (moveArr[x][y] > 0) {
 87                     moveList.add(String.format("(%s, %s, %s)", x, y, moveArr[x][y]));
 88                 }
 89             }
 90             builder.append("\r\n");
 91         }
 92         moveBuilder.append(moveList.toString()).append(" ");
 93     }
 94 
 95     /**
 96      * 将一个数组转化为String(配合Map使用)
 97      *
 98      * @param arr 待处理数组
 99      * @return 处理结果
100      */
101     public static String arrKey(int[][] arr) {
102         StringBuilder builder = new StringBuilder();
103         for (int i = 0; i < arr.length; i++) {
104             for (int j = 0; j < arr[i].length; j++) {
105                 builder.append(arr[i][j]);
106             }
107         }
108         return builder.toString();
109     }
110 
111     /**
112      * 可视化一个Array(用于打印)
113      *
114      * @param arr 待复制数组
115      * @return 复制后的Array
116      */
117     public static String arrString(int[][] arr) {
118         StringBuilder builder = new StringBuilder();
119         for (int i = 0; i < arr.length; i++) {
120             for (int j = 0; j < arr[i].length; j++) {
121                 builder.append(arr[i][j]);
122                 builder.append(", ");
123             }
124             if (i < arr.length - 1) {
125                 builder.append("\r\n");
126             }
127         }
128         return builder.toString();
129     }
130 
131     /**
132      * 复制一个Array
133      *
134      * @param arr 待复制数组
135      * @return 复制后的Array
136      */
137     public static int[][] arrCopy(int[][] arr) {
138         int[][] result = new int[arr.length][];
139         for (int i = 0; i < arr.length; i++) {
140             result[i] = new int[arr[i].length];
141             for (int j = 0; j < arr[i].length; j++) {
142                 result[i][j] = arr[i][j];
143             }
144         }
145         return result;
146     }
147 
148     /**
149      * 返回一个在0到num之间的随机数,包含0,不包含num
150      *
151      * @param num 数字
152      * @return 返回随机数
153      */
154     public static int randInt(int num) {
155         if (num >= 0) {
156             return randInt(0, num);
157         } else {
158             return randInt(num, 0);
159         }
160     }
161 
162     /**
163      * 返回一个在min到max之间的随机数,包含min,不包含max
164      *
165      * @param min 随机数下限
166      * @param max 随机数上限
167      * @return 返回随机数
168      */
169     public static int randInt(int min, int max) {
170         int diff = max - min;
171         int rand = (int) (Math.random() * diff);
172         return min + rand;
173     }
174 
175     /**
176      * 求一组数字的最大值
177      *
178      * @param nums 输入数列
179      * @return 最大值
180      */
181     public static int max(int... nums) {
182         int result = nums[0];
183         for (int i = 1; i < nums.length; i++) {
184             if (nums[i] > result) {
185                 result = nums[i];
186             }
187         }
188         return result;
189     }
190 
191     /**
192      * 求一组数字的最小值
193      *
194      * @param nums 输入数列
195      * @return 最小值
196      */
197     public static int min(int... nums) {
198         int result = nums[0];
199         for (int i = 1; i < nums.length; i++) {
200             if (nums[i] < result) {
201                 result = nums[i];
202             }
203         }
204         return result;
205     }
206 }
工具类

 

接下来需要模拟游戏过程了:

这部分的重点是,需要不停地扫描当前是否有水滴在活动,以及是否有水珠发生爆炸。两者要分开并依次进行,以防止多个水滴同时命中水珠后爆炸,导致结果误判。

计算完毕一个状态以后,可以把它缓存,之后无需重复计算。

  1 package main;
  2 
  3 import bean.Bullet;
  4 import bean.MoveResult;
  5 import bean.Point;
  6 import util.Utils;
  7 
  8 import java.util.ArrayList;
  9 import java.util.HashMap;
 10 import java.util.List;
 11 import java.util.Map;
 12 
 13 /**
 14  * 游戏的基本参数与功能
 15  * 如果需要分析别的盘面,在这里修改参数
 16  */
 17 public class Game {
 18 
 19     /**
 20      * 棋盘X大小
 21      */
 22     public static int X = 6;
 23 
 24     /**
 25      * 棋盘Y大小
 26      */
 27     public static int Y = 6;
 28 
 29     /**
 30      * 爆裂门槛
 31      */
 32     public static int BURST = 4;
 33 
 34     /**
 35      * 奖励倒数,COMBO除以这个数后得到奖励
 36      */
 37     public static int REWARD_INV = 3;
 38 
 39     /**
 40      * 完美通关的额外奖励
 41      */
 42     public static int REWARD_PFT = 1;
 43 
 44     /**
 45      * 缓存发生爆炸后的计算结果
 46      */
 47     public static Map<String, MoveResult> MOVE_MAP = new HashMap<>(524288);
 48 
 49     /**
 50      * 待求解初始状态
 51      */
 52     private int[][] initArr = {
 53             {0, 2, 2, 1, 4, 4},
 54             {2, 1, 1, 1, 3, 0},
 55             {2, 3, 1, 2, 2, 4},
 56             {0, 3, 2, 0, 0, 4},
 57             {1, 0, 3, 2, 1, 0},
 58             {3, 3, 0, 2, 0, 0},
 59 
 60     };
 61 
 62 //    public static void main(String[] args) {
 63 //        Game game = new Game();
 64 //        MoveResult move = move(game.initArr, 5, 3);
 65 //        System.out.println(Utils.arrString(move.getArr()));
 66 //        System.out.println(move.getCombo());
 67 //    }
 68 
 69     /**
 70      * @param arr   棋盘
 71      * @param point 坐标点
 72      * @return v1: 执行后的棋盘, v2: 本次操作的收益, v3: 是否发生了爆炸
 73      */
 74     public static MoveResult move(int[][] arr, Point point) {
 75         return move(arr, point.getX(), point.getY());
 76     }
 77 
 78     /**
 79      * 执行一次加水操作,返回执行后的棋盘,本次操作收益,是否发生爆炸
 80      * 如果发生爆炸,需要缓存对应结果
 81      *
 82      * @param arr 棋盘
 83      * @param x   X坐标
 84      * @param y   Y坐标
 85      * @return v1: 执行后的棋盘, v2: 本次操作的收益, v3: 是否发生了爆炸
 86      */
 87     public static MoveResult move(int[][] arr, int x, int y) {
 88         int[][] resArr = Utils.arrCopy(arr);
 89         resArr[x][y]++;
 90         if (resArr[x][y] <= BURST) {
 91             // 本次操作不会导致爆炸,直接返回
 92             return new MoveResult(resArr, -1, 0);
 93         }
 94         String arrKey = Utils.arrKey(arr) + x + y;
 95         MoveResult result = MOVE_MAP.get(arrKey);
 96         if (result != null) {
 97             // 本次操作命中缓存,直接返回
 98             return result;
 99         }
100 
101         // 循环计算爆炸结果
102         int combo = 0;
103         List<Bullet> bulletList = new ArrayList<>();
104         do {
105             // 1. 所有水滴移动1回合
106             List<Bullet> newBulletList = new ArrayList<>();
107             for (Bullet bullet : bulletList) {
108                 Point point = bullet.move();
109                 if (!point.isValidPosition()) {
110                     // 水滴出界,则移除掉它
111                     continue;
112                 }
113                 if (resArr[point.getX()][point.getY()] > 0) {
114                     // 水滴遇到了已存在的水球,则水球大小+1,移除水滴
115                     resArr[point.getX()][point.getY()]++;
116                     continue;
117                 }
118                 // 水滴通过校验,存活进入下一个回合
119                 newBulletList.add(bullet);
120             }
121 
122             // 2. 检查是否有水珠爆炸,是的话在该位置生成4个方向的新水滴,并将所在位置的格子置空
123             List<Point> burstPointList = availablePoints(resArr, 1);
124             for (Point point : burstPointList) {
125                 newBulletList.add(new Bullet(point, 1, 0));
126                 newBulletList.add(new Bullet(point, 0, 1));
127                 newBulletList.add(new Bullet(point, -1, 0));
128                 newBulletList.add(new Bullet(point, 0, -1));
129                 resArr[point.getX()][point.getY()] = 0;
130                 combo++;
131             }
132 
133             bulletList = newBulletList;
134         } while (!bulletList.isEmpty());
135 
136         int reward = combo / REWARD_INV - 1;
137         result = new MoveResult(resArr, reward, combo);
138         MOVE_MAP.put(arrKey, result);
139         return result;
140     }
141 
142     /**
143      * 获取可选的点
144      *
145      * @param arr   棋盘
146      * @param burst 点的状态,0为任意有水滴点,1为仅满,-1为仅不满
147      * @return 返回的可选点列表
148      */
149     public static List<Point> availablePoints(int[][] arr, int burst) {
150         List<Point> pointList = new ArrayList<>();
151         for (int i = 0; i < arr.length; i++) {
152             for (int j = 0; j < arr[i].length; j++) {
153                 if (arr[i][j] > 0) {
154                     int isBurst = arr[i][j] > BURST ? 1 : -1;
155                     int result = burst * isBurst;
156                     if (result >= 0) {
157                         pointList.add(Point.get(i, j));
158                     }
159                 }
160             }
161         }
162         return pointList;
163     }
164 
165     public int[][] getInitArr() {
166         return initArr;
167     }
168 
169     public void setInitArr(int[][] initArr) {
170         this.initArr = initArr;
171     }
172 }

 

接下来,就是使用遗传算法,计算出最优解了。

这里我想了想,找到了一个尽可能适合本问题的生成子代方法,大致是这样的:

1、分数最高的<精英个数>名后代,直接移入下一回合

2、分数最高的<精英个数>名后代,随机两两配对,独立地产生2个后代,可能变异

3、分数最高的<普通个数>名后代,随机两两配对,对立地产生2个后代,可能变异

4、分数最高的<随机个数>名后代,从随机位置截断,截断后的部分全部随机生成

配对时,以50%几率设置1个切断位点,25%几率设置2个切断位点,25%几率设置3个切断位点。重组位点在两个解长度较小值之内。

变异发生时,以50%几率修改一步,25%几率增加一步,25%几率删除一步。

生成子代后,计算它们的最终收益,爆炸次数、解的长度,以及期间最大投入成本,并排序。

如果计算出的解的最终收益、爆炸次数、解的长度发生更新,则回合数置零。当回合数超过<最大回合数>时,终止并输出结果。

经过参数调优,精英个数设置为128,普通个数为4096,随机个数为512,最大回合数为128,变异几率为0.125。

 

本算法和一般的遗传算法不同的地方:

1、不限制答案长度。(但理论最大是6x6x3步)

2、生成后代的方式比较丰富,但只允许适应度最高的一些解产生后代。但是,不在这范围内的解直接淘汰,而不是以几率被选择。

3、允许了最多3个重组位点,而不是固定1个。

4、允许突变发生增加和删除操作,而不是只有修改操作。

  1 package main;
  2 
  3 import bean.Answer;
  4 import bean.MoveResult;
  5 import bean.Point;
  6 import util.Utils;
  7 
  8 import java.util.ArrayList;
  9 import java.util.Collections;
 10 import java.util.Comparator;
 11 import java.util.List;
 12 
 13 /**
 14  * 遗传算法,寻找一个尽可能收益最高的解
 15  */
 16 public class Resolve {
 17 
 18     /**
 19      * 种群容量
 20      * 初始生成数量、生成普通后代时的配对数量
 21      */
 22     private int capacity = 4096;
 23 
 24     /**
 25      * 精英种群容量
 26      * 精英种群直接移入下一回合,同时两两配对产生2个后代
 27      */
 28     private int eliteCapacity = 128;
 29 
 30     /**
 31      * 随机种群容量
 32      * 每回合对该数量的成员,在某位置截断,之后部分重新生成
 33      * 它负责同时引入大量突变
 34      */
 35     private int randomCutCapacity = 512;
 36 
 37     /**
 38      * 经过该回合数,最优解的适应度不发生变化,则终止并退出遗传算法,输出结果
 39      */
 40     private int stableExitTurn = 128;
 41 
 42     /**
 43      * 重组时发生变异的几率
 44      * 一旦发生,随机位置以25%几率新增1个操作,25%删除1个操作,50%修改1个操作(修改后可能与原操作相同)
 45      */
 46     private double variationRate = 0.125D;
 47 
 48     public int getCapacity() {
 49         return capacity;
 50     }
 51 
 52     public void setCapacity(int capacity) {
 53         this.capacity = capacity;
 54     }
 55 
 56     public int getEliteCapacity() {
 57         return eliteCapacity;
 58     }
 59 
 60     public void setEliteCapacity(int eliteCapacity) {
 61         this.eliteCapacity = eliteCapacity;
 62     }
 63 
 64     public int getRandomCutCapacity() {
 65         return randomCutCapacity;
 66     }
 67 
 68     public void setRandomCutCapacity(int randomCutCapacity) {
 69         this.randomCutCapacity = randomCutCapacity;
 70     }
 71 
 72     public int getStableExitTurn() {
 73         return stableExitTurn;
 74     }
 75 
 76     public void setStableExitTurn(int stableExitTurn) {
 77         this.stableExitTurn = stableExitTurn;
 78     }
 79 
 80     public double getVariationRate() {
 81         return variationRate;
 82     }
 83 
 84     public void setVariationRate(double variationRate) {
 85         this.variationRate = variationRate;
 86     }
 87 
 88     public static void main(String[] args) {
 89         Resolve resolve = new Resolve();
 90         Game game = new Game();
 91 
 92 //        for (int i = 0; i < game.getInitArr().length; i++) {
 93 //            for (int j = 0; j < game.getInitArr()[i].length; j++) {
 94 //                game.getInitArr()[i][j] = Utils.randInt(5);
 95 //            }
 96 //        }
 97 //        System.out.println(Utils.arrString(game.getInitArr()));
 98 //        System.out.println();
 99 
100         Answer answer = resolve.select(game);
101         System.out.println(answer);
102     }
103 
104     /**
105      * 遗传算法执行入口
106      *
107      * @param game 游戏
108      * @return 找到的最优解
109      */
110     public Answer select(Game game) {
111         int turn = 0;
112         int stableTurn = 0;
113         int lastTotalReward = 0;
114         int lastBurstCount = 0;
115         int lastLength = 0;
116         List<Answer> answerList = new ArrayList<>(capacity);
117 
118         // 初始化,产生最初的答案
119         for (int i = 0; i < capacity; i++) {
120             Answer answer = new Answer();
121             calcTotalReward(game, answer);
122             answerList.add(answer);
123         }
124         sortAnswerList(answerList);
125 
126         // 执行遗传算法
127         while (stableTurn < stableExitTurn) {
128             // 1. 生成下一代,同时计算总奖励
129             List<Answer> newAnswerList = new ArrayList<>(capacity + eliteCapacity * 2 + randomCutCapacity);
130 
131             // A. 分数最高的<精英个数>名后代,直接移入下一回合
132             for (int i = 0; i < eliteCapacity; i++) {
133                 newAnswerList.add(answerList.get(i));
134             }
135 
136             // B. 分数最高的<精英个数>名后代,随机两两配对,独立地产生2个后代,可能变异
137             List<Answer> eliteMateAnswerList = mate(game, answerList, eliteCapacity, 0);
138             newAnswerList.addAll(eliteMateAnswerList);
139 
140             // C. 分数最高的<普通个数>名后代,随机两两配对,对立地产生2个后代,可能变异
141             List<Answer> mateAnswerList = mate(game, answerList, capacity, 2);
142             newAnswerList.addAll(mateAnswerList);
143 
144             // D. 分数最高的<随机个数>名后代,从随机位置截断,截断后的部分全部随机生成
145             for (int i = 0; i < randomCutCapacity; i++) {
146                 newAnswerList.add(randomCut(game, answerList.get(i)));
147             }
148 
149             // 2. 更新循环状态
150             Collections.shuffle(newAnswerList);
151             sortAnswerList(newAnswerList);
152             Answer bestAnswer = newAnswerList.get(0);
153             turn++;
154             if (bestAnswer.getTotalReward() == lastTotalReward && bestAnswer.getBurstCount() == lastBurstCount && bestAnswer.length() == lastLength) {
155                 stableTurn++;
156             } else {
157                 stableTurn = 0;
158             }
159             lastTotalReward = bestAnswer.getTotalReward();
160             lastBurstCount = bestAnswer.getBurstCount();
161             lastLength = bestAnswer.length();
162             answerList = newAnswerList;
163             System.out.println(turn + ", " + stableTurn + ": " + bestAnswer);
164         }
165 
166         // 打印最终找到的最优解(16个)
167         System.out.println();
168         System.out.println("最优Top16:");
169         for (int i = 0; i < 16; i++) {
170             Answer answer = answerList.get(i);
171             System.out.println(answer);
172         }
173 
174         // 打印最优解
175         System.out.println();
176         System.out.println("最优解关键盘面:");
177         Answer bestAnswer = answerList.get(0);
178         System.out.println(Utils.answerString(game, bestAnswer));
179         return bestAnswer;
180     }
181 
182     /**
183      * 两两配对,每一对产生2个后代
184      * 重组断点50%为1个,25%为2个,25%为3个。
185      *
186      * @param game       游戏
187      * @param answerList 排序后的解答列表
188      * @param limit      数量限制,只有最前面的解答有资格产生后代
189      * @param policy     0: 对立产生后代,生成2个子代,分别随机地独占亲代的基因,亲代基因不会丢失,1或以上:独立产生policy个后代,子代基因从亲代完全随机获得,互不影响
190      * @return 生成的子代列表
191      */
192     public List<Answer> mate(Game game, List<Answer> answerList, int limit, int policy) {
193         int pairNum = limit >> 1;
194         ArrayList<Answer> parentAnswerList = new ArrayList<>(answerList.subList(0, limit));
195         Collections.shuffle(parentAnswerList);
196         List<Answer> childAnswerList = new ArrayList<>();
197         for (int pairNo = 0; pairNo < pairNum; pairNo++) {
198             Answer parentAnswer0 = parentAnswerList.get(2 * pairNo);
199             Answer parentAnswer1 = parentAnswerList.get(2 * pairNo + 1);
200             List<Point> parentPointList0 = parentAnswer0.getPointList();
201             List<Point> parentPointList1 = parentAnswer1.getPointList();
202             int minLength = Utils.min(parentAnswer0.length(), parentAnswer1.length());
203 
204             if (policy > 0) {
205                 // 独立产生后代,子代基因从亲代完全随机获得,互不影响
206                 for (int childNo = 0; childNo < policy; childNo++) {
207                     int breakPointNum = Utils.max(Utils.randInt(4), 1);
208                     List<Integer> matePositionList = new ArrayList<>();
209                     for (int i = 0; i < breakPointNum; i++) {
210                         matePositionList.add(Utils.randInt(minLength));
211                     }
212                     matePositionList.sort(Integer::compareTo);
213 
214                     List<Point> childPointList = new ArrayList<>();
215                     int starts = Utils.randInt(2);
216                     int lastMatePosition = 0;
217                     for (int i = 0; i < breakPointNum + 1; i++) {
218                         List<Point> fragmentPointList;
219                         List<Point> parentPointList = (i + starts) % 2 == 0 ? parentPointList0 : parentPointList1;
220                         if (i < breakPointNum) {
221                             int matePosition = matePositionList.get(i);
222                             fragmentPointList = parentPointList.subList(lastMatePosition, matePosition);
223                             lastMatePosition = matePosition;
224                         } else {
225                             fragmentPointList = parentPointList.subList(lastMatePosition, parentPointList.size());
226                         }
227                         childPointList.addAll(fragmentPointList);
228                     }
229                     childAnswerList.add(new Answer(childPointList));
230                 }
231             } else {
232                 // 对立产生后代,生成2个子代,分别随机地独占亲代的基因,亲代基因不会丢失
233                 int breakPointNum = Utils.max(Utils.randInt(4), 1);
234                 List<Integer> matePositionList = new ArrayList<>();
235                 for (int i = 0; i < breakPointNum; i++) {
236                     matePositionList.add(Utils.randInt(minLength));
237                 }
238                 matePositionList.sort(Integer::compareTo);
239 
240                 List<Point> childPointList0 = new ArrayList<>();
241                 List<Point> childPointList1 = new ArrayList<>();
242                 int lastMatePosition = 0;
243                 for (int i = 0; i < breakPointNum + 1; i++) {
244                     // 按照断点指定的位置,获得亲代基因的碎片
245                     List<Point> fragmentPointList0;
246                     List<Point> fragmentPointList1;
247                     if (i < breakPointNum) {
248                         int matePosition = matePositionList.get(i);
249                         fragmentPointList0 = parentPointList0.subList(lastMatePosition, matePosition);
250                         fragmentPointList1 = parentPointList1.subList(lastMatePosition, matePosition);
251                         lastMatePosition = matePosition;
252                     } else {
253                         fragmentPointList0 = parentPointList0.subList(lastMatePosition, parentPointList0.size());
254                         fragmentPointList1 = parentPointList1.subList(lastMatePosition, parentPointList1.size());
255                     }
256 
257                     // 拼接亲代基因碎片,得到子代基因
258                     if (i % 2 == 0) {
259                         childPointList0.addAll(fragmentPointList0);
260                         childPointList1.addAll(fragmentPointList1);
261                     } else {
262                         childPointList0.addAll(fragmentPointList1);
263                         childPointList1.addAll(fragmentPointList0);
264                     }
265                 }
266                 childAnswerList.add(new Answer(childPointList0));
267                 childAnswerList.add(new Answer(childPointList1));
268             }
269         }
270 
271         // 引入变异
272         for (Answer answer : childAnswerList) {
273             if (Math.random() < variationRate && answer.length() > 0) {
274                 int variationType = Utils.max(Utils.randInt(4), 1);
275                 int position = Utils.randInt(answer.length());
276                 switch (variationType) {
277                     case 1:
278                         // 修改一个步骤
279                         answer.getPointList().remove(position);
280                         // BREAK-THROUGH
281                     case 2:
282                         // 插入一个步骤,需要遍历到当前为止的棋盘状态,然后随机插入一个合法的位置
283                         int[][] arr = game.getInitArr();
284                         for (int i = 0; i < position; i++) {
285                             Point point = answer.getPointList().get(i);
286                             if (arr[point.getX()][point.getY()] <= 0) {
287                                 continue;
288                             }
289                             // 记录本次操作的结果,并更新棋盘状态
290                             MoveResult move = Game.move(arr, point);
291                             arr = move.getArr();
292                         }
293                         List<Point> remainPointList = Game.availablePoints(arr, 0);
294                         if (!remainPointList.isEmpty()) {
295                             Point point = remainPointList.get(Utils.randInt(remainPointList.size()));
296                             answer.getPointList().add(position, point);
297                         }
298                         break;
299                     case 3:
300                         // 删除一个步骤
301                         answer.getPointList().remove(position);
302                         break;
303                 }
304             }
305             calcTotalReward(game, answer);
306         }
307         return childAnswerList;
308     }
309 
310     /**
311      * 从随机位置斩断一个解,之后部分随机生成
312      *
313      * @param game   游戏
314      * @param answer 解答
315      * @return 生成的新解答
316      */
317     public Answer randomCut(Game game, Answer answer) {
318         int saveLength = Utils.randInt(answer.length());
319         List<Point> savePointList = answer.getPointList().subList(0, saveLength);
320         Answer newAnswer = new Answer(savePointList);
321         calcTotalReward(game, newAnswer);
322         return newAnswer;
323     }
324 
325     /**
326      * 验证解答,并计算总奖励
327      * 如果解答序列不完整,则随机生成之后序列,使其补充完整。
328      *
329      * @param game   游戏
330      * @param answer 答案
331      * @return 总奖励
332      */
333     public static int calcTotalReward(Game game, Answer answer) {
334         int totalReward = 0;
335         int minReward = 0;
336         int burstCount = 0;
337         int[][] arr = game.getInitArr();
338 
339         // 遍历已有点的列表,验证并计算结果
340         List<Point> pointList = new ArrayList<>(answer.getPointList());
341         for (Point point : answer.getPointList()) {
342             if (arr[point.getX()][point.getY()] <= 0) {
343                 // 这个点已经被清空了,因此跳过它,将其从列表中删除
344                 pointList.remove(point);
345                 continue;
346             }
347 
348             // 记录本次操作的结果,并更新棋盘状态
349             MoveResult move = Game.move(arr, point);
350             arr = move.getArr();
351             totalReward += move.getReward();
352             if (minReward > totalReward) {
353                 minReward = totalReward;
354             }
355             if (move.getCombo() > 0) {
356                 burstCount++;
357             }
358         }
359 
360         // 检查是否棋盘未被清空,是的话则随机生成之后的解答
361         List<Point> remainPointList = Game.availablePoints(arr, 0);
362         while (!remainPointList.isEmpty()) {
363             Point point = remainPointList.get(Utils.randInt(remainPointList.size()));
364             pointList.add(point);
365 
366             // 记录本次操作的结果,并更新棋盘状态
367             MoveResult move = Game.move(arr, point);
368             arr = move.getArr();
369             totalReward += move.getReward();
370             if (minReward > totalReward) {
371                 minReward = totalReward;
372             }
373             if (move.getCombo() > 0) {
374                 burstCount++;
375             }
376 
377             remainPointList = Game.availablePoints(arr, 0);
378         }
379 
380         // 追加完美通关奖励
381         if (burstCount <= 1) {
382             totalReward += Game.REWARD_PFT;
383         }
384 
385         answer.getPointList().clear();
386         answer.getPointList().addAll(pointList);
387         answer.setBurstCount(burstCount);
388         answer.setTotalReward(totalReward);
389         answer.setMinReward(minReward);
390         return totalReward;
391     }
392 
393     public static void sortAnswerList(List<Answer> answerList) {
394         answerList.sort(Comparator.comparing(Answer::getTotalReward).reversed().thenComparing(Answer::getBurstCount).thenComparing(Answer::length));
395     }
396 }

计算出的结果最终打印到控制台中。

 

实际用于解决该游戏后,发现了以下现象与问题:

1、最优解几乎总是先在棋盘上设置大量小水滴,之后一把将其全部引爆。

2、最优解如果直接打印,很多在同一个单元格的操作不放在一起,增加操作难度,比如一不小心点错单元格。

3、不是每次都能找到最优解,有的最优解出现概率很低,需要多次执行确认。

4、直接照着盘面改Array太麻烦了。

最终,我采取了以下措施:

1、输出解时。将一组不引爆水珠的一系列操作合并,增加一个次数参数。

2、将执行入口设置为了允许并发运行。一般同时执行3个实例,若找到的解的收益不同,则再执行几个看看。

3、我写了个程序,辅助将盘面录入为Array,以便粘贴。

Java使用遗传算法,寻找十滴水问题的最优解Java使用遗传算法,寻找十滴水问题的最优解
 1 package main;
 2 
 3 import java.util.Scanner;
 4 
 5 public class FormatArray {
 6     public static void main(String[] args) {
 7         System.out.println("粘贴纯数字格式,获得格式化的Array对象。注意最后需要多加一个空行。");
 8         Scanner scanner = new Scanner(System.in);
 9         while (scanner.hasNext()) {
10             String line = scanner.next();
11             char[] chars = line.toCharArray();
12             StringBuilder builder = new StringBuilder();
13             builder.append("{");
14             for (int i = 0; i < chars.length; i++) {
15                 builder.append(chars[i]);
16                 if (i < chars.length - 1) {
17                     builder.append(", ");
18                 }
19             }
20             builder.append("},");
21             System.out.println(builder.toString());
22         }
23     }
24 }
Array辅助

 

如有兴趣,可以下载源文件:https://pan.baidu.com/s/1iWNIkNLUXDDadNLlaerIHg?pwd=bili

 

参考资料:

《也谈十滴水-启发式搜索》https://www.cnblogs.com/ComputingLife/archive/2010/09/19/1830873.html文章来源地址https://www.toymoban.com/news/detail-828254.html

到了这里,关于Java使用遗传算法,寻找十滴水问题的最优解的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 遗传算法及基于该算法的典型问题的求解实践

        遗传算法是一个很有用的工具,它可以帮我们解决生活和科研中的诸多问题。最近在看波束形成相关内容时了解到可以用这个算法来优化阵元激励以压低旁瓣,于是特地了解和学习了一下这个算法,觉得蛮有意思的,于是把这两天关于该算法的学习和实践的内容总结成了

    2024年03月21日
    浏览(48)
  • 遗传算法解决tsp问题(基于python)

    目录 1.遗传算法简要介绍 2.tsp问题简要介绍 3.遗传算法解决tsp问题的几个特殊点 4.源码         简单来说,遗传算法是用于解决最优化问题的一种搜索算法。其核心基于自然界种群进化的规律,即初始种群进行交配,在基因层面上,其中会发生交叉互换、基因突变等变异

    2023年04月12日
    浏览(79)
  • 路径规划问题的遗传算法实现(python代码)

        遗传算法是模拟达尔文生物进化论的自然选择和遗传学机理的生物进化过程的计算模型,是一种通过模拟自然进化过程搜索最优解的方法。该算法通过数学的方式,利用计算机仿真运算,将问题的求解过程转换成类似生物进化中的染色体基因的交叉、变异等过程。     路径

    2024年02月04日
    浏览(45)
  • 人工智能导论——遗传算法求解TSP问题实验

    一、实验目的: 熟悉和掌握遗传算法的原理、流程和编码策略,并利用遗传算法求解组合优化问题,理解求解TSP问题的流程并测试主要参数对结果的影响。 二、实验原理: 旅行商问题,即TSP问题(Traveling Salesman Problem)是数学领域中著名问题之一。假设有一个旅行商人要拜

    2023年04月13日
    浏览(45)
  • 模拟退火算法与遗传算法求解多目标优化问题的算法实现(数学建模)

    模拟退火算法是一种全局优化算法,解决的问题通常是找到一个最小化(或最大化)某个函数的全局最优解。它通过模拟物理退火的过程来搜索解空间,在开始时以一定的温度随机生成初始解,然后一步步降低温度,同时在当前解的周围随机搜索新的解,并根据一定概率接受

    2024年02月02日
    浏览(57)
  • 基于Python实现的遗传算法求最值问题

    遗传算法求最值问题 目录 人工智能第三次实验报告 1 遗传算法求最值问题 1 一 、遗传算法 1 1.1 遗传算法简介 1 1.2 遗传算法基本要素 2 4. 设定遗传操作: 2 1.3 遗传算法一般步骤 2 二 、程序说明 2 2.1 控制参数 2 2.2 编码规则 3 2.3 选择初始群体 3 2.4 适应度函数 4 三 、参数测试

    2023年04月25日
    浏览(37)
  • 遗传算法GA解决混合流水车间调度问题HFSP

    混合流水车间调度问题(HFSP)是传统流水车间调度问题(FSP)的拓展,本文针对HFSP问题进行描述、建模和求解。 通常模型做如下假设: HFSP符号描述: 决策变量: 主要约束: 优化目标: 本节使用带精英保留的遗传算法GA对HFSP问题进行求解。求解结果如下: 自定义算例如下:

    2024年02月11日
    浏览(49)
  • 遗传算法求解旅行商问题(含python源代码)

    目录 前言 编码初始化种群 计算适应度 选择 交叉 变异 完整代码 总结 这次的算法有一点不能确定是否正确,希望有大佬能够批评指正。 遗传算法的一般步骤 种群(population) 指同一时间生活在一定自然区域内,同种生物的所有个体。 所以种群是由个体组成的,所以先需要

    2024年01月23日
    浏览(68)
  • 人工智能原理实验4(1)——遗传算法、蚁群算法求解TSP问题

    TSP问题是组合数学中一个古老而又困难的问题,也是一个典型的组合优化问题,现已归入NP完备问题类。NP问题用穷举法不能在有效时间内求解,所以只能使用启发式搜索。遗传算法是求解此类问题比较实用、有效的方法之一。下面给出30个城市的位置信息: 应用遗传算法和蚁

    2024年01月24日
    浏览(58)
  • 【遗传模拟退火算法的Java实现及其应用】

    遗传模拟退火算法是一种基于遗传算法和模拟退火算法的启发式优化算法。它的基本思路是在解决优化问题时模拟生物进化的过程,利用遗传算法的遗传操作和模拟退火算法的搜索策略。 初始化种群 :初始化种群包含解和目标函数值。 适应度评估 :使用目标函数对种群中的

    2024年02月08日
    浏览(70)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包