S10-算法-马踏棋盘算法(骑士周游问题)【2021-02-12】

总目录:地址如下看总纲

https://www.jianshu.com/p/929ca9e209e8

1、马踏棋盘算法介绍

1、马踏棋盘算法也被称为骑士周游问题
2、将马随机放在国际象棋的8×8棋盘Board[0~7][0~7]的某个方格中,马按走棋规则(马走日字)进行移动。要求每个方格只进入一次,走遍棋盘上全部64个方格
3、游戏地址:http://www.4399.com/flash/146267_2.htm

2、算法实现思路

1、马踏棋盘问题(骑士周游问题)实际上是图的深度优先搜索(DFS)的应用。
2、如果使用回溯(就是深度优先搜索)来解决,假如马儿踏了53个点,如图:走到了第53个,坐标(1,0),发现已经走到尽头,没办法,那就只能回退了,查看其他的路径,就在棋盘上不停的回溯……


image.png

步骤:

  1. 创建棋盘 chessBoard , 是一个二维数组
  2. 将当前位置设置为已经访问,然后根据当前位置,计算马儿还能走哪些位置,并放入到一个集合中(ArrayList), 最多有8个位置, 每走一步,就使用step+1
  3. 遍历ArrayList中存放的所有位置,看看哪个可以走通 , 如果走通,就继续,走不通,就回溯.
  4. 判断马儿是否完成了任务,使用 step 和应该走的步数比较 , 如果没有达到数量,则表示没有完成任务,将整个棋盘置0
    注意:马儿不同的走法(策略),会得到不同的结果,效率也会有影响(优化)

使用贪心算法对原来的算法优化:
1、我们获取当前位置,可以走的下一个位置的集合

//获取当前位置可以走的下一个位置的集合
ArrayList<Point> ps = next(new Point(column, row));

2、我们需要对 ps 中所有的Point 的下一步的所有集合的数目,进行非递减排序,就ok ,

9, 7, 6, 5, 3, 2 , 1 //递减排序
1, 2, 3, 4,5,6, 10, //递增排序
1, 2, 2, 2, 3,3, 4, 5, 6 // 非递减
9, 7, 6,6, 6, 5,5, 3, 2 , 1 //非递增


image.png

3、代码

public class HorseChessboard {
    public static void main(String[] args) {
        // 测试骑士周游算法是否正确
        X = 8;
        Y = 8;
        int row = 1; //马儿初始位置的行,从1开始编号
        int column = 1; //马儿初始位置的列,从1开始编号
        // 创建棋盘
        int[][] chessboard = new int[X][Y];
        visited = new boolean[X * Y];//初始值都是false
        long start = System.currentTimeMillis ( );
        traversalChessboard (chessboard, row - 1, column - 1, 1);
        long end = System.currentTimeMillis ( );
        System.out.println ("共耗时: " + (end - start) + " 毫秒");

        // 输出棋盘的最后情况
        for (int[] rows : chessboard) {
            for (int step : rows) {
                System.out.print (step + "\t");
            }
            System.out.println ( );
        }
    }

    private static int X;// 棋盘的列数
    private static int Y;// 棋盘的行数
    // 用于标记棋盘的各个位置是否被访问过
    private static boolean[] visited;
    // 用于记录是否棋盘的所有位置都被访问过
    private static boolean finished;

    /**
     * 步骤三:贪心算法优化:选择(定制)排序
     * 根据当前这个一步的所有的下一步的选择位置,进行非递减排序, 减少回溯的次数
     *
     * @param ps 需要排序的下一次情况集合
     */
    public static void sort(ArrayList<Point> ps) {
        ps.sort (new Comparator<Point> ( ) {
            @Override
            public int compare(Point o1, Point o2) {
                //获取到o1的下一步的所有位置个数
                int count1 = next (o1).size ( );
                //获取到o2的下一步的所有位置个数
                int count2 = next (o2).size ( );
                if (count1 < count2) {
                    return -1;
                } else if (count1 == count2) {
                    return 0;
                } else {
                    return 1;
                }
            }
        });
    }

    /**
     * 步骤二:实现骑士周游问题算法
     *
     * @param chessboard 棋盘
     * @param row        行(马仔当前在第几行),从0开始
     * @param column     列(马仔当前在第几列),从0开始
     * @param step       第几步,从1开始:从左到右数
     */
    public static void traversalChessboard(int[][] chessboard, int row, int column, int step) {
        chessboard[row][column] = step;
        // row = 4 X = 8 column = 4 = 4 * 8 + 4 = 36 = step
        visited[row * X + column] = true;
        // 获取当前位置可以走的下一个位置 ---这些情况的集合
        ArrayList<Point> ps = next (new Point (column, row));

        //-------贪心优化-------
        // 对ps进行排序,排序的规则就是对ps的所有的Point对象的下一步的位置的数目,进行非递减排序
        sort (ps);
        //-------贪心优化-------

        // 遍历ps
        while (!ps.isEmpty ( )) {
            // 取出下一个可以走的位置坐标
            Point point = ps.remove (0);
            // 校验该点是否被访问过
            if (!visited[point.y * X + point.x]) {// 未被访问过
                // 回溯
                traversalChessboard (chessboard, point.y, point.x, step + 1);
            }
        }
        // 判断马儿是否完成了任务,使用   step 和应该走的步数比较 ,
        // 如果没有达到数量,则表示没有完成任务,将整个棋盘置0
        // 说明: step < X * Y  成立的情况有两种
        // 1. 棋盘到目前位置,仍然没有走完
        // 2. 棋盘处于一个回溯过程
        if (step < X * Y && !finished) {
            chessboard[row][column] = 0;// 棋盘坐标重置
            visited[row * X + column] = false;
        } else {
            // 全部走完
            finished = true;
        }
    }

    /**
     * 步骤一: 根据当前位置(Point对象),计算马仔还能走那些位置(Point),并且放入到一个集合中 ArrayList,最多有八个位置(马走日字)
     * 注:这里用 Point 点对象代表坐标
     *
     * @param curPoint 当前位置坐标
     * @return
     */
    public static ArrayList<Point> next(Point curPoint) {
        //创建一个ArrayList
        ArrayList<Point> ps = new ArrayList<> ( );
        //创建一个Point
        Point p1 = new Point ( );

        // 表示马仔可以走 5这个位置(见图)
        if ((p1.x = curPoint.x - 2) >= 0 && (p1.y = curPoint.y - 1) >= 0) {// 左移动两格可以走的通,并且上移动一格可以走的通
            // 成立,存入
            ps.add (new Point (p1));
        }
        //判断马仔可以走6这个位置
        if ((p1.x = curPoint.x - 1) >= 0 && (p1.y = curPoint.y - 2) >= 0) {
            ps.add (new Point (p1));
        }
        //判断马仔可以走7这个位置
        if ((p1.x = curPoint.x + 1) < X && (p1.y = curPoint.y - 2) >= 0) {
            ps.add (new Point (p1));
        }
        //判断马仔可以走0这个位置
        if ((p1.x = curPoint.x + 2) < X && (p1.y = curPoint.y - 1) >= 0) {
            ps.add (new Point (p1));
        }
        //判断马仔可以走1这个位置
        if ((p1.x = curPoint.x + 2) < X && (p1.y = curPoint.y + 1) < Y) {
            ps.add (new Point (p1));
        }
        //判断马仔可以走2这个位置
        if ((p1.x = curPoint.x + 1) < X && (p1.y = curPoint.y + 2) < Y) {
            ps.add (new Point (p1));
        }
        //判断马仔可以走3这个位置
        if ((p1.x = curPoint.x - 1) >= 0 && (p1.y = curPoint.y + 2) < Y) {
            ps.add (new Point (p1));
        }
        //判断马仔可以走4这个位置
        if ((p1.x = curPoint.x - 2) >= 0 && (p1.y = curPoint.y + 1) < Y) {
            ps.add (new Point (p1));
        }
        return ps;
    }
}

4、结果性能测试

1、非贪心结果


image.png

2、贪心性能


image.png

5、仓库坐标

image.png
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 202,905评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,140评论 2 379
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,791评论 0 335
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,483评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,476评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,516评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,905评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,560评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,778评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,557评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,635评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,338评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,925评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,898评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,142评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,818评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,347评论 2 342

推荐阅读更多精彩内容