JS回溯算法--八皇后问题

八皇后问题

在 8×8 格的国际象棋上摆放八个皇后,使其不能互相攻击,即任意两个皇后都不能处于同一行、同一列或同一斜线上,问有多少种摆法。

回溯算法

回溯算法实际上一个类似枚举的搜索尝试过程,主要是在搜索尝试过程中寻找问题的解,当发现已不满足求解条件时,就“回溯”返回,尝试别的路径。

回溯算法和穷举法很像,都是树的深度优先遍历,但回溯法会进行'剪枝',比如第 5 层某 i 叶子结点时发现该节点已经无意义,会直接跳过该节点下面的遍历,提高了效率

求解

不考虑限制条件,问题变成了排列组合,一共有 C64 取 8(22307873454720)种

分析该问题,问题可以拆解为:

  • 从 64 个格子中取 8 个格子放入 8 个皇后
  • 限制条件:任意两个皇后都不能处于同一行、同一列或同一斜线上

我们用一个二维矩阵来记录皇后的位置和状态,其中 1 表示该位置已经有皇后了

let num = 8;
for (let i = 0; i < num; i++) {
  this.arr[i] = [];
  for (let j = 0; j < num; j++) {
    this.arr[i][j] = 0;
  }
}
// 横轴表示 i 行,纵轴表示第 j 列
//   1  2  3  4  5  6  7  8
1  [ 0, 0, 0, 0, 0, 0, 0, 0 ], # (0,0) 表示第一行第一列  (0,1) 表示第一行第二列
2  [ 0, 0, 0, 0, 0, 0, 0, 0 ],
3  [ 0, 0, 0, 0, 0, 0, 0, 0 ],
4  [ 0, 0, 0, 0, 0, 0, 0, 0 ],
5  [ 0, 0, 0, 0, 0, 0, 0, 0 ],
6  [ 0, 0, 0, 0, 0, 0, 0, 0 ],
7  [ 0, 0, 0, 0, 0, 0, 0, 0 ],
8  [ 0, 0, 0, 0, 0, 0, 0, 0 ]

基本步骤如下:

  1. 一个检查函数,确定该位置不会与其他皇后有冲突
  2. 从第一行开始,8 列中任意选一个列执行步骤 0,若安全则开始放第二行........直到放到第八行若有位置,则已经找到一个解输出;若第 i 行的操作中没有位置可以放入皇后回退回第 i-1 行的操作,若 i=0 即第一行则说明已经执行完毕程序结束

用递归可以很容易的实现,什么是递归?看下面这个斐波那契数列,就是一个很简单的递归,递归有两个很重要的特点:一个结束条件,不断调用自身。

recurFib(n) {
  if (n < 2) return n;
  return this.recurFib(n - 1) + this.recurFib(n - 2);
}
// 栈是一种先进后出的数据结构,先压进栈的最后出栈
n = 3时
1.recurFib(3)压入执行栈,执行到 recurFib(2) + recurFib(1),recurFib(2)被压入执行栈中
2.recurFib(2)执行,执行到 recurFib(1) + recurFib(0),recurFib(1)被压入执行栈中
recurFib(1)执行,返回1;recurFib(1)执行完出栈
继续执行recurFib(2), 1 + recurFib(0),recurFib(0)压入执行栈中
recurFib(0)执行,返回1;recurFib(0)执行完出栈
继续执行recurFib(2),1 + 1,返回2,recurFib(2)执行完出栈
3.继续执行recurFib(3),执行到 2 + recurFib(1),recurFib(1)被压入执行栈中
recurFib(1)执行,返回1;recurFib(1)执行完出栈
继续执行recurFib(3),2 + 1 ,返回 3,栈已经清空,程序结束

回到 8 皇后问题,关键代码如下:

buildList(list, row) {
  // 递归中止条件,找到一个解缓存起来
  if (row === list.length) {
    this.result.push(JSON.parse(JSON.stringify(list)));
    return;
  }
  for (let col = 0, len = list.length; col < len; col++) {
    if (this.isSafe(list, row, col)) {
      list[row][col] = 1;
      this.buildList(list, row + 1);
      // 走到这里,说明该次递归已经结束,不管找没找到,都需要重置,继续找下一个可放置的位置
      list[row][col] = 0;
    }
  }
}

isSafe 方法,确保该行该列不会与其他皇后冲突:

isSafe(list, row, col) {
  const len = list.length;
  // 同一列
  for (let i = 0; i < len; i++) {
    if (list[i][col] === 1) return false;
  }
  // 斜右上方
  for (let i = row - 1, j = col + 1; i >= 0 && j < len; i--, j++) {
    if (list[i][j] === 1) return false;
  }
  // 斜左上方
  for (let i = row - 1, j = col - 1; i >= 0 && j >= 0; i--, j--) {
    if (list[i][j] === 1) return false;
  }
  return true;
}

完整代码

class Queen {
  constructor(num) {
    this.num = num;
    this.arr = [];
    this.result = [];
    this.initList();
    this.buildList(this.arr, 0);
  }

  initList() {
    let num = this.num;
    for (let i = 0; i < num; i++) {
      this.arr[i] = [];
      for (let j = 0; j < num; j++) {
        this.arr[i][j] = 0;
      }
    }
    console.log(this.arr);
  }

  buildList(list, row) {
    // 递归中止条件,找到一个解缓存起来
    if (row === list.length) {
      this.result.push(JSON.parse(JSON.stringify(list)));
      return;
    }
    for (let col = 0, len = list.length; col < len; col++) {
      if (this.isSafe(list, row, col)) {
        list[row][col] = 1;
        this.buildList(list, row + 1);
        // 走到这里,说明该次递归已经结束,不管找没找到,都需要重置
        list[row][col] = 0;
      }
    }
  }

  isSafe(list, row, col) {
    const len = list.length;
    // 同一列
    for (let i = 0; i < len; i++) {
      if (list[i][col] === 1) return false;
    }
    // 斜右上方
    for (let i = row - 1, j = col + 1; i >= 0 && j < len; i--, j++) {
      if (list[i][j] === 1) return false;
    }
    // 斜左上方
    for (let i = row - 1, j = col - 1; i >= 0 && j >= 0; i--, j--) {
      if (list[i][j] === 1) return false;
    }
    return true;
  }
}
const queen = new Queen(8);
console.log(queen.result);

优化

其实解法跟之前一样,上面用了二维矩阵来记录位置,因为已经确定同一行不可能存在 2 个皇后实际上只用一维数组就表示:list[row] = col;减少空间消耗

isSafe 判断和之前不同

isSafe(row) {
  for (let i = 0; i < row; i++) {
    // 判断列
    if (this.arr[i] === this.arr[row]) return false;
    // 判断对角线
    if (Math.abs(this.arr[row] - this.arr[i]) === row - i) return false;
  }
  return true;
}

完整代码

class Queen {
  constructor(num) {
    this.num = num;
    this.arr = [];
    this.result = [];
    this.initList();
    this.buildList(0);
  }

  initList() {
    let num = this.num;
    for (let i = 0; i < num; i++) {
      this.arr[i] = 0;
    }
  }

  buildList(row) {
    // 递归中止条件,找到一个解缓存起来
    if (row === this.num) {
      this.result.push(JSON.parse(JSON.stringify(this.arr)));
      return;
    }
    for (let col = 0; col < this.num; col++) {
      this.arr[row] = col;
      if (this.isSafe(row)) {
        this.buildList(row + 1);
      }
    }
  }

  isSafe(row) {
    for (let i = 0; i < row; i++) {
      // 判断列
      if (this.arr[i] === this.arr[row]) return false;
      // 判断对角线
      if (Math.abs(this.arr[row] - this.arr[i]) === row - i) return false;
    }
    return true;
  }
}

const queen = new Queen(8);
console.log(queen.result);

END

跑一下,可知 8 皇后问题一共有 92 种摆法

尝试跑了一下 n=16 的情况,cpu 直接燃烧,几分钟没出结果,果断放弃......
可见递归有多么耗性能~~

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