算法-N皇后问题(含加速版本)
package com.jiucaiyuan.net.question; /** * 【问题】N皇后问题,指的是n*n的棋盘,可以放n个皇后,要求任何两个皇后不同行、不同列,也不在同一条斜线上 * 给出一个整数n,返回n皇后的摆法有多少种 * 给n=1,返回1 * n=2或者3,2皇后和3皇后,无论怎么摆放都不行,返回0 * n=8,返回92 * 【思路】按照行往下摆放,然后一列一列尝试(加速版本会使用按照位操作来加速器处理,但是整体上也是所有都尝试一遍) * 时间复杂度O(N^N) * * @Author jiucaiyuan 2022/4/19 23:26 * @mail services@jiucaiyuan.net */ public class NQueens { /** * @param n 皇后数 * @return 摆放种数 */ public static int nQueens(int n) { System.out.println("--------------" + n + "----------------------"); if (n < 1) { return 0; } int[] record = new int[n]; //record[i] i行的皇后放到第几列上 return process1(0, record, n); } /** * @param i 当前处理第i行元素 * @param record record[0~i-1]表示之前的行,放了皇后的位置,不共行、不共列、不共斜线 * @param n 总共有多少行(皇后个数) * @return i行的摆法 */ public static int process1(int i, int[] record, int n) { if (i == n) { print(record); return 1; } int res = 0; for (int j = 0; j < n; j++) { //当前i行的皇后放到j列,会不会和之前i-1个皇后共列、共斜线 //如果是,无效,如果不是,有效 if (isValid(record, i, j)) { record[i] = j; res += process1(i + 1, record, n); } } return res; } /** * 检查是否存在共列、共斜线情况(不可能有共行情况,因为是一行一行处理的) * * @param record record[k] 表示k行摆放在record[k]列 0<=k<=i-1 * @param i 当前放到i行 * @param j i行放到j列 * @return 是否和之前的有共列、共斜线情况 */ public static boolean isValid(int[] record, int i, int j) { for (int k = 0; k < i; k++) { if (record[k] == j //共列 || Math.abs(k - i) == Math.abs(record[k] - j) //共斜线 ) { return false; } } return true; } private static void print(int[] record) { System.out.print(record[0]); for (int i = 1; i < record.length; i++) { System.out.print("," + record[i]); } System.out.println(); } /** * <pre> * n皇后问题(加速版本) * 【适用范围】由于限制用的是int类型值来表示,所以支持n<=32 * 【加速思路】:未加速版本是通过一个数组记录已经放置皇后,再放置时检查是否和已经放置的 * 共行、共列、共斜线,加速版本思路也是要检查是否共行、共列、共斜线,不过是每次处理 * 时往下传入列限制、左斜线限制、右斜线限制,其中这些限制是通过二进制运算得到,达到 * 加速的效果 * 【限制规则】:1的位置不能放皇后,0的位置可以 * 列限制 | 左斜线限制 | 右斜线限制 = 总限制 * colLim | leftDiaLim | rightDiaLim * </pre> * * @param n 皇后个数 * @return 返回摆法的种数 */ public static int nQueens2(int n) { if (n < 1 || n > 32) { System.out.println("加速方法的处理不支持超过32个皇后"); return 0; } //因为是32位,所以用upperLim来标识当前指定的是皇后个数,即22皇后时,upperLim值是 00000000001111111111111111111111 //右侧1的个数就是当前皇后个数 //(1<<n)-1 得到的是后面n个1,前面全是0 int upperLim = n == 32 ? -1 : (1 << n) - 1; return process2(upperLim, 0, 0, 0); } /** * <pre> * 【加速思路】:未加速版本是通过一个数组记录已经放置皇后,再放置时检查是否和已经放置的 * 共行、共列、共斜线,加速版本思路也是要检查是否共行、共列、共斜线,不过是每次处理 * 时往下传入列限制、左斜线限制、右斜线限制,其中这些限制是通过二进制运算得到,达到 * 加速的效果 * 【限制规则】:1的位置不能放皇后,0的位置可以 * 列限制 | 左斜线限制 | 右斜线限制 = 总限制 * colLim | leftDiaLim | rightDiaLim * </pre> * * @param upperLim * @param colLim 列的限制 * @param leftDiaLim 左斜线的限制 * @param rightDiaLim 右斜线的限制 * @return */ public static int process2(int upperLim, int colLim, int leftDiaLim, int rightDiaLim) { if (colLim == upperLim) {//列限制信息和皇后数一致了,即放完了所有皇后 return 1; } int pos = 0; //可以放皇后的位置 int mostRightOne = 0; //colLim | leftDiaLim | rightDiaLim 总限制 //~(colLim | leftDiaLim | rightDiaLim) 可以放皇后的位置 //pos 指定n个皇后时可以放皇后的位置 pos = upperLim & (~(colLim | leftDiaLim | rightDiaLim)); int res = 0; while (pos != 0) { mostRightOne = pos & (~pos + 1);//按照二进制的方式表示pos,取到pos最右侧的1在哪个位置 pos = pos - mostRightOne; //每次减去最右侧一个1(当前循环尝试了),即按照这种方式来尝试所有可以尝试的次数 res += process2(upperLim, colLim | mostRightOne, //当前处理的位置加入下次尝试的列限制中 (leftDiaLim | mostRightOne) << 1, //左侧限制向左侧移动一位作为下次尝试的左侧限制 (rightDiaLim | mostRightOne) >>> 1);// 右侧限制向右侧移动一位作为下次尝试的右侧限制 } return res; } public static void main(String[] args) { long start = System.currentTimeMillis(); System.out.println(nQueens(1)); System.out.println("耗时=" + (System.currentTimeMillis() - start)); start = System.currentTimeMillis(); System.out.println(nQueens(2)); System.out.println("耗时=" + (System.currentTimeMillis() - start)); start = System.currentTimeMillis(); System.out.println(nQueens(3)); System.out.println("耗时=" + (System.currentTimeMillis() - start)); start = System.currentTimeMillis(); System.out.println(nQueens(4)); System.out.println("耗时=" + (System.currentTimeMillis() - start)); start = System.currentTimeMillis(); System.out.println(nQueens(5)); System.out.println("耗时=" + (System.currentTimeMillis() - start)); start = System.currentTimeMillis(); System.out.println(nQueens(6)); System.out.println("耗时=" + (System.currentTimeMillis() - start)); start = System.currentTimeMillis(); System.out.println(nQueens(7)); System.out.println("耗时=" + (System.currentTimeMillis() - start)); start = System.currentTimeMillis(); System.out.println(nQueens(8)); System.out.println("耗时=" + (System.currentTimeMillis() - start)); start = System.currentTimeMillis(); System.out.println(nQueens(10)); System.out.println("耗时=" + (System.currentTimeMillis() - start)); start = System.currentTimeMillis(); System.out.println(nQueens(11)); System.out.println("耗时=" + (System.currentTimeMillis() - start)); start = System.currentTimeMillis(); System.out.println(nQueens(15)); System.out.println("耗时=" + (System.currentTimeMillis() - start)); start = System.currentTimeMillis(); System.out.println(nQueens2(15)); System.out.println("耗时=" + (System.currentTimeMillis() - start)); } }
日历
个人资料
diaba 寻求合作请留言或联系mail: services@jiucaiyuan.net
链接
最新文章
存档
- 2024年10月(1)
- 2024年8月(2)
- 2024年6月(4)
- 2024年5月(1)
- 2023年7月(1)
- 2022年10月(1)
- 2022年8月(1)
- 2022年6月(11)
- 2022年5月(6)
- 2022年4月(33)
- 2022年3月(26)
- 2021年3月(1)
- 2020年9月(2)
- 2018年8月(1)
- 2018年3月(1)
- 2017年3月(3)
- 2017年2月(6)
- 2016年12月(3)
- 2016年11月(2)
- 2016年10月(1)
- 2016年9月(3)
- 2016年8月(4)
- 2016年7月(3)
- 2016年6月(4)
- 2016年5月(7)
- 2016年4月(9)
- 2016年3月(4)
- 2016年2月(5)
- 2016年1月(17)
- 2015年12月(15)
- 2015年11月(12)
- 2015年10月(6)
- 2015年9月(11)
- 2015年8月(8)
分类
热门文章
- SpringMVC:Null ModelAndView returned to DispatcherServlet with name 'applicationContext': assuming HandlerAdapter completed request handling
- Mac-删除卸载GlobalProtect
- java.lang.SecurityException: JCE cannot authenticate the provider BC
- MyBatis-Improper inline parameter map format. Should be: #{propName,attr1=val1,attr2=val2}
- Idea之支持lombok编译
标签
最新评论
- logisqykyk
Javassist分析、编辑和创建jav... - xxedgtb
Redis—常见参数配置 - 韭菜园 ... - wdgpjxydo
SpringMVC:Null Model... - rllzzwocp
Mysql存储引擎MyISAM和Inno... - dpkgmbfjh
SpringMVC:Null Model... - tzklbzpj
SpringMVC:Null Model... - bqwrhszmo
MyBatis-Improper inl... - 乐谱吧
good非常好 - diaba
@diaba:应该说是“时间的度量依据”... - diaba
如果速度增加接近光速、等于光速、甚至大于...
最新微语
- 从今天起,做一个幸福的人。喂马,砍柴,(思想)周游世界
2022-03-21 23:31
- 2022.03.02 23:37:59
2022-03-02 23:38
- 几近崩溃后,找到解决方法,总是那么豁然开朗!所以遇到问题要坚持!
2018-07-18 10:49
- 2018年关键字“走心”
2018-03-19 16:07
- 保护好自己最大的方法是让自己更强大,不要柔弱的像一只绵羊一样,得谁巴拉,就谁巴拉!
2017-12-20 10:24
发表评论: