zoukankan      html  css  js  c++  java
  • N-Queens

    The n-queens puzzle is the problem of placing n queens on an n×n chessboard such that no two queens attack each other.

    Given an integer n, return all distinct solutions to the n-queens puzzle.

    Each solution contains a distinct board configuration of the n-queens' placement, where 'Q' and '.' both indicate a queen and an empty space respectively.

    For example,
    There exist two distinct solutions to the 4-queens puzzle:

    [
     [".Q..",  // Solution 1
      "...Q",
      "Q...",
      "..Q."],
    
     ["..Q.",  // Solution 2
      "Q...",
      "...Q",
      ".Q.."]
    ]

    n皇后问题,题意就是求n×n矩阵中,每行放一个棋子,使得棋子所在的列和两条斜线上没有其他棋子,枚举所有可能。

    dfs去遍历,考虑所有可能,row中记录每一行棋子的位置,col记录当前列是否有棋子,对角线的判断就是两点行差值和列差值是否相同。

    当dfs深度达到n时,就表示存在满足条件的解,把当前状态图存到结果中。

    temp(n, '.')先把字符串全部赋值成 ‘ . ' ,在吧存在棋子的位置改成’Q‘

    经典的8皇后,递归回溯可解。同时还学了StringBuilder里面一个setCharAt()很方便的方法 

    ref:http://www.2cto.com/kf/201311/256634.html

    public class Solution{
    
        public ArrayList<String[]> solveNQueens(int n) {  
            ArrayList<String[]> result = new ArrayList<String[]>();  
            int[] queenList = new int[n];  
            placeQueen(queenList, 0, n, result);  
            return result;  
        }  
           
        // 递归回溯8皇后,关键记录下到达了哪一行了  
        public void placeQueen(int[] queenList, int row, int n, ArrayList<String[]> result){  
            // Base Case, 已经完成任务了  
            if(row == n){  
                StringBuilder[] sol = new StringBuilder[n];  
                   
                // 对数组内每一个对象都要new出其对象, 并将其全设为'.' 
                for(int i=0; i<n; i++){  
                    sol[i] = new StringBuilder();  
                    for(int j=0; j<n; j++){  
                        sol[i].append(".");  
                    }  
                }  
                // 在相应的地方放置queen  
                for(int i=0; i<n; i++){  
                    sol[i].setCharAt(queenList[i], 'Q');  
                }  
                String[] ss = new String[n];  
                for (int i=0; i<n; i++) {  
                    ss[i] = sol[i].toString();  
                }  
                result.add(ss);  
                return;  
            }  
               
            // 开始这一行的查找  
            // 遍历第row行的所有列,测试哪一个位置是安全的,然后一行一行的放queen 
            for(int col=0; col<n; col++){  
                if(isSafe(queenList, row, col)){  
                    queenList[row] = col;  
                   //递归到下一行
                    placeQueen(queenList, row+1, n, result);  
                }  
            }  
        }  
           
        // 判断是否坐标(row,col)的位置是安全的(检查行,列,正反对角线)  
        // queenList里面存放行,列坐标pair,即queenList[row] = col  
        public boolean isSafe(int[] queenList, int row, int col){
          //遍历当前row之前的所有row
            for(int preRow=0; preRow<row; preRow++){  
                // 得到上一行queen 放在哪一个 col里
                int preCol = queenList[preRow];  
                if(preRow == row){      // 理论上不必检查,因为preRow是总是小于row的  
                    return false;  
                }  
                if(preCol == col){          // 检查是否在同一列  
                    return false;  
                }  
                if(row-preRow == col-preCol){       // 反对角线  
                    return false;  
                }  
                if(preRow+preCol == row+col){       // 正对角线  
                    return false;  
                }  
            }  
            return true;  
        }      
    
    }            
  • 相关阅读:
    【剑指Offer】面试题14- I. 剪绳子
    【剑指Offer】面试题56
    【LeetCode】202. 快乐数
    【LeetCode】1095. 山脉数组中查找目标值
    【LeetCode】260. 只出现一次的数字 III
    【剑指Offer】面试题56
    【LeetCode】33. 搜索旋转排序数组
    【LeetCode】23. 合并K个排序链表
    【LeetCode】46. 全排列
    mysql可视化工具下载地址2017.6.27
  • 原文地址:https://www.cnblogs.com/RazerLu/p/3535578.html
Copyright © 2011-2022 走看看