n皇后问题的最优解及优化

2023-12-04 16:44

本文主要是介绍n皇后问题的最优解及优化,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

n皇后问题的最优解

时间复杂度    O(n^{n})

package algorithm;public class NQueen {public static int num(int n){if(n < 1){return 0;}int[] record = new int[n];//n皇后的n*n的棋盘,record[i]表示第i行的皇后放在了第几列return process(0,record,n);}/***返回n皇后的摆法* i表示从第几行开始,n表示皇后数*/private static int process(int i, int[] record, int n) {//如果当前的位置在i行,那么即表示i-1及之前的行的皇后一定是满足条件的,在摆第i行的皇后要满足条件不与之前的皇后共行、共列或共线//如果行数来到n,说明已经来到最后,此时即为出现一种摆法if(i == n){return 1;}int res = 0;//遍历列for (int j = 0; j < n; j++) {if(isValid(record,i,j)){//判断是否能摆record[i] = j;//能摆久记录摆下的皇后的列res += process(i+1,record,n);//递归,摆下一行,在下一行依旧遍历所有列}}return res;}//判断是否共行、共列、共斜线//由于record数组的索引存储的就是行数,因此不需要检查是否共行private static boolean isValid(int[] record, int i, int j) {for (int k = 0; k < i; k++) {//检查是否共列、共斜线(纵坐标之差 == 横坐标之差)if(j == record[k] || Math.abs(record[k] - j) == Math.abs(i-k)){return false;}}return true;}
}

上述最优解在n的值较大的时候跑出来仍是很慢,但无法对此进行结构上的优化,即无法进行时间复杂度的优化,但是可以使用位运算进行常数时间上的优化

优化

条件:不超过32皇后问题,超过32皇后问题需要把int转化为long类型

举个栗子

八皇后问题,n = 8

1位置不能放皇后,0位置可以

第1行00001000
列限制00001000
左斜线限制00010000
右斜线限制00000100
三个限制求或00011100
1
第2行01000000
第2行+第1行限制11101010

左斜线限制 = 列数 -  行差(左移)

右斜线限制 = 列数 + 行差(右移)

利用位运算的特性,限制哪些列可以放皇后,而不是每一列都遍历,节省时间

int pos = limit & (~(colLim | leftDiaLim | rightDiaLim));

第一个数表示前面的数,limit即前面都是0,后面8位都是1

limit011111111
列限制000001000
左斜线限制000010000
右斜线限制000000100
三种限制或000011100
~取反111100011
&并limit011100011

为什么要有limit?

因为可以和limit求并截去左侧的所有的数,在最终求得的pos数中,1的位置可以放置皇后,0的位置不可以放置皇后

mostRightOne = pos & (~pos + 1);
pos011100011
~pos100011100
~pos+1100011101
pos&(~pos+1)000000001

package algorithm;public class NQueenOptimize {public static int num(int n) {//不超过32皇后问题if (n < 1 || n > 32) {return 0;}//生成一个二进制位数,后n位都是1,前面都是0//limit数本身的值不重要,只是用它的位信息int limit = -1;//n == 32if (n != 32) {limit = (1 << n) - 1;}return process(limit, 0, 0, 0);}/*** 1的位置不可以放皇后,0可以* colLim: 列的限制* leftDiaLim: 左斜线的限制* rightDiaLim: 右斜线的限制*/private static int process(int limit, int colLim, int leftDiaLim, int rightDiaLim) {//列上的限制 == limit,说明全都放满了,此时即为一种摆法if (colLim == limit) {return 1;}/*** pos表示所有候选皇后的位置,1的位置可以放皇后,0的位置不可以放皇后* (colLim | leftDiaLim | rightDiaLim) 三种限制或* ~ 表示取反,此时在棋盘上1表示可以放皇后,0表示不能放皇后* & 和limit求与(0和1与为0,1和1与为1)*/int pos = limit & (~(colLim | leftDiaLim | rightDiaLim));int res = 0;int mostRightOne = 0;//在每个1的位置上尝试皇后的位置,在尝试之后将值修改为0while (pos != 0) {mostRightOne = pos & (~pos + 1);pos = pos - mostRightOne;//左右限制变化res += process(limit,colLim | mostRightOne,(leftDiaLim | mostRightOne) << 1,(rightDiaLim | mostRightOne) >> 1);}return res;}}

这篇关于n皇后问题的最优解及优化的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/454153

相关文章

线上Java OOM问题定位与解决方案超详细解析

《线上JavaOOM问题定位与解决方案超详细解析》OOM是JVM抛出的错误,表示内存分配失败,:本文主要介绍线上JavaOOM问题定位与解决方案的相关资料,文中通过代码介绍的非常详细,需要的朋... 目录一、OOM问题核心认知1.1 OOM定义与技术定位1.2 OOM常见类型及技术特征二、OOM问题定位工具

Vue3绑定props默认值问题

《Vue3绑定props默认值问题》使用Vue3的defineProps配合TypeScript的interface定义props类型,并通过withDefaults设置默认值,使组件能安全访问传入的... 目录前言步骤步骤1:使用 defineProps 定义 Props步骤2:设置默认值总结前言使用T

Web服务器-Nginx-高并发问题

《Web服务器-Nginx-高并发问题》Nginx通过事件驱动、I/O多路复用和异步非阻塞技术高效处理高并发,结合动静分离和限流策略,提升性能与稳定性... 目录前言一、架构1. 原生多进程架构2. 事件驱动模型3. IO多路复用4. 异步非阻塞 I/O5. Nginx高并发配置实战二、动静分离1. 职责2

从原理到实战解析Java Stream 的并行流性能优化

《从原理到实战解析JavaStream的并行流性能优化》本文给大家介绍JavaStream的并行流性能优化:从原理到实战的全攻略,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的... 目录一、并行流的核心原理与适用场景二、性能优化的核心策略1. 合理设置并行度:打破默认阈值2. 避免装箱

解决升级JDK报错:module java.base does not“opens java.lang.reflect“to unnamed module问题

《解决升级JDK报错:modulejava.basedoesnot“opensjava.lang.reflect“tounnamedmodule问题》SpringBoot启动错误源于Jav... 目录问题描述原因分析解决方案总结问题描述启动sprintboot时报以下错误原因分析编程异js常是由Ja

Python实战之SEO优化自动化工具开发指南

《Python实战之SEO优化自动化工具开发指南》在数字化营销时代,搜索引擎优化(SEO)已成为网站获取流量的重要手段,本文将带您使用Python开发一套完整的SEO自动化工具,需要的可以了解下... 目录前言项目概述技术栈选择核心模块实现1. 关键词研究模块2. 网站技术seo检测模块3. 内容优化分析模

Java实现复杂查询优化的7个技巧小结

《Java实现复杂查询优化的7个技巧小结》在Java项目中,复杂查询是开发者面临的“硬骨头”,本文将通过7个实战技巧,结合代码示例和性能对比,手把手教你如何让复杂查询变得优雅,大家可以根据需求进行选择... 目录一、复杂查询的痛点:为何你的代码“又臭又长”1.1冗余变量与中间状态1.2重复查询与性能陷阱1.

Python内存优化的实战技巧分享

《Python内存优化的实战技巧分享》Python作为一门解释型语言,虽然在开发效率上有着显著优势,但在执行效率方面往往被诟病,然而,通过合理的内存优化策略,我们可以让Python程序的运行速度提升3... 目录前言python内存管理机制引用计数机制垃圾回收机制内存泄漏的常见原因1. 循环引用2. 全局变

MySQL 表空却 ibd 文件过大的问题及解决方法

《MySQL表空却ibd文件过大的问题及解决方法》本文给大家介绍MySQL表空却ibd文件过大的问题及解决方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考... 目录一、问题背景:表空却 “吃满” 磁盘的怪事二、问题复现:一步步编程还原异常场景1. 准备测试源表与数据

解决Nginx启动报错Job for nginx.service failed because the control process exited with error code问题

《解决Nginx启动报错Jobfornginx.servicefailedbecausethecontrolprocessexitedwitherrorcode问题》Nginx启... 目录一、报错如下二、解决原因三、解决方式总结一、报错如下Job for nginx.service failed bec