Skip to main content

2 posts tagged with "algorithm"

View All Tags

· 8 min read
jeesk

八皇后问题的程序解答

在8×8格的国际象棋上摆放8个皇后,使其不能互相攻击,即任意两个皇后都不能处于同一行、同一列或同一斜线上,问有多少种摆法。高斯认为有76种方案。1854年在柏林的象棋杂志上不同的作者发表了40种不同的解,后来有人用图论的方法解出92种结果。如果经过±90度、±180度旋转,和对角线对称变换的摆法看成一类,共有42类。计算机发明后,有多种计算机语言可以编程解决此问题。

由上面的概述可以得出, 任意两个皇后不能处于同一行,同一列, 或者同一斜线. 这条规则是校验八皇后是否合法的规则, 后面的代码将根据这条规则来编写.

1. 回溯法

​ 首先我们定义一个数组 array = {0 , 4, 7, 5, 2, 6, 1, 3},用单数组表示八皇后的位置,

上面的数组依次表示8皇后在棋盘的位置, array [0] = 0, array [1]=4, array[index] = val ,可以得出每个八皇后的位置在第index+1行,

在(val+1) 列, 入下图所示, 可以看到下面的表格是满足任意两个皇后不能处于同一行,同一列, 或者同一斜线.

1
2
3
4
5
6
7
8

1.1 八皇后的规则校验

    /**
* 校验该皇后是否和前面的皇后冲突
*
* @param index 第几个皇后(index表示在数组的索引)
* @return 该皇后的位置是否冲突
*/
public boolean isJudge(int index) {
if (index == 0) {
// 如果是在数组的第一个皇后直接返回不冲突
return true;
}
// 遍历该皇后前面的皇后
for (int i = 0; i < index; i++) {
// 1. 判断该皇后是否和前面的皇后在同一列
if (array[index] == array[i]) {
// 假如array[0] = 0, array[1]=1, array[2] = 1那么这个就是冲突的
return false;
}
// 2. 判断该皇后是否和其他的皇后在同一条斜线
// array = {0,1,2,3,4,5,6,7} , 这8个皇后在同一条斜线, 由此可以得到, 他们的差值相等的
// 第一个皇后在第一行第一列, 第二个皇后在第二行第二列, 第三个皇后在第三行第三列
// 由此可得第n个皇后和n-个皇后的位置(index)的差值,等于他们所在y轴(val)的差值
if (Math.abs(array[index] - array[i]) == Math.abs(index - i)) {
return false;
}
// 3. 判断是否在同一行由于我们已经规定好了 array = {0 , 4, 7, 5, 2, 6, 1, 3},
// 每一个皇后的位置的行在 array[index] =val, 由index 决定,所以肯定不在同一行

}
return true;
}

1.2 放入皇后

    /**
* @param index 皇后在数组的位置(index)
*/
public void putQueue(int index) {
if (index == max) {
// 当index = max的时候, 表示皇后已经放了8个,
num++;
printQueue();
return;
}

for (int i = 0; i < max; i++) {
// 将皇后放入数组, 这里的i代表皇后在棋盘的列
array[index] = i;
if (isJudge(index)) {
// 不冲突, 继续放入皇后到数组
putQueue(index + 1);
}
// 冲突的话, 继续循环放入新的皇后到数组(改变当前皇后的列)
}
}

1.3 完整代码如下


public class Queue8Demo {

//定义一个max表示共有多少个皇后
int max = 8;
//定义数组array, 保存皇后放置位置的结果,比如 arr = {0 , 4, 7, 5, 2, 6, 1, 3}
int[] array = new int[max];
// 解法计数器
static int num = 0;

public static void main(String[] args) {
long startTime = System.currentTimeMillis();
Queue8Demo queue8 = new Queue8Demo();
queue8.putQueue(0);
System.out.println("耗时:" + (System.currentTimeMillis() - startTime));
}

/**
* 校验该皇后是否和前面的皇后冲突
*
* @param index 第几个皇后(index表示在数组的索引)
* @return 该皇后的位置是否冲突
*/
public boolean isJudge(int index) {
if (index == 0) {
// 如果是在数组的第一个皇后直接返回不冲突
return true;
}
// 遍历该皇后前面的皇后
for (int i = 0; i < index; i++) {
// 1. 判断该皇后是否和前面的皇后在同一列
if (array[index] == array[i]) {
// 假如array[0] = 0, array[1]=1, array[2] = 1那么这个就是冲突的
return false;
}
// 2. 判断该皇后是否和其他的皇后在同一条斜线
// array = {0,1,2,3,4,5,6,7} , 这8个皇后在同一条斜线, 由此可以得到, 他们的差值相等的
// 第一个皇后在第一行第一列, 第二个皇后在第二行第二列, 第三个皇后在第三行第三列
// 由此可得第n个皇后和n-个皇后的位置(index)的差值,等于他们所在y轴(val)的差值
if (Math.abs(array[index] - array[i]) == Math.abs(index - i)) {
return false;
}
// 3. 判断是否在同一行由于我们已经规定好了 array = {0 , 4, 7, 5, 2, 6, 1, 3},
// 每一个皇后的位置的行在 array[index] =val, 由index 决定,所以肯定不在同一行

}
return true;
}


/**
* 放入皇后到数组
* @param index 皇后在数组的位置(index)
*/
public void putQueue(int index) {
if (index == max) {
// 当index = max的时候, 表示皇后已经放了8个,已经得到一种解法
num++;
printQueue();
return;
}

for (int i = 0; i < max; i++) {
// 将皇后放入数组, 这里的i代表皇后在棋盘的列
array[index] = i;
if (isJudge(index)) {
// 不冲突, 继续放入皇后到数组
putQueue(index + 1);
}
// 冲突的话, 继续循环放入新的皇后到数组(改变当前皇后的列)
}
}

public void printQueue() {
System.out.print("解法" + num + ":");
for (int i = 0; i < array.length; i++) {
System.out.print(array[i]);
}
System.out.println();
}

}

总结

  1. 通过递归解决了八皇后的解法, 学习递归的用法

· 5 min read
jeesk

约瑟夫问题的几种算法

据说著名犹太历史学家Josephus有过以下的故事:在罗马人占领乔塔帕特后,39 个犹太人与Josephus及他的朋友躲到一个洞中,39个犹太人决定宁愿死也不要被敌人抓到,于是决定了一个自杀方式,41个人排成一个圆圈,由第1个人开始报数,每报数到第3人该人就必须自杀,然后再由下一个重新报数,直到所有人都自杀身亡为止。然而Josephus 和他的朋友并不想遵从。首先从一个人开始,越过k-2个人(因为第一个人已经被越过),并杀掉第k个人。接着,再越过k-1个人,并杀掉第k个人。这个过程沿着圆圈一直进行,直到最终只剩下一个人留下,这个人就可以继续活着。问题是,给定了和,一开始要站在什么地方才能避免被处决。Josephus要他的朋友先假装遵从,他将朋友与自己安排在第16个与第31个位置,于是逃过了这场死亡游戏

现在简单的举个例子:

假如一共有8个, 从第一个开始报数, 然后每次报3个人出圈

1,2,3,4,5,6,7,8

  1. 1,2,3,4,5,6,7,8 3号出圈
  2. 1,2,4,5,6,7,8 6号出圈
  3. 1,2,4,5,7,8 1号出圈
  4. 2,4,5,7,8 5号出圈
  5. 2,4,7,8 2号出圈
  6. 4,7,8 8号出圈
  7. 4,7 4号出圈
  8. ​ 7

现在大家应该是明白了, 约瑟夫算法的玩法, 那么下面分享几种不同思路的解法

1. 环形链表解法

1.1 思路分析

  1. 首先是拿到环形链表的第一个数据first , 和最后一个数据last
  2. 在报数的时候, 循环报数, 每次first 和last 移动(m-1)次, 然后删除当前节点, 相当于出圈
  3. 当过first= last的时候, 只有最后一个人了

1.2 代码实战

package com.shanjiancaofu;

public class JosephuQuestion {
public static void main(String[] args) {
LinkedCircle linkedCircle = new LinkedCircle();
// 增加8个人, 计算出圈的人
linkedCircle.addEle(8);
linkedCircle.calu(1, 3);
}

public static class LinkedCircle {

/**
* @param startNo 从第几个人开始数
* @param countNum 报数
*/
public void calu(int startNo, int countNum) {
// 找到最后一个数据
Element lastEle = first;
while (true) {
if (lastEle.getNext() == first) {
break;
}
lastEle = lastEle.getNext();
}

// 将lastEle和first 移动(startNo-1), 这里移动的原因是因为实际情况可能不是从第一个人开始报数
for (int i = 0; i < startNo - 1; i++) {
lastEle = lastEle.getNext();
first = first.next;
}

// 循环报数
while (true) {
if (lastEle == first) {
// 只有一个数据
break;
}
// 执行报数, 移动first数据,找到出圈的人
for (int i = 0; i < countNum - 1; i++) {
lastEle = lastEle.getNext();
first = first.next;
}
// 得到的first 就是移除圈的人
System.out.println("移除圈的人NO:" + first.getNo());
first = first.next;
lastEle.setNext(first);
}

}


private Element first;

/**
* 增加几个数据
*
* @param nums 增加的数据个数
*/
public void addEle(int nums) {
if (nums <= 0) {
System.out.println("参数异常");
return;
}
// 链表的最后一个数据
Element lastElement = null;

for (int i = 1; i <= nums; i++) {
Element element = new Element();
element.setNo(i);
if (i == 1) {
// 第一个数据是first, 并且first的next也是first自己
first = element;
first.setNext(first);
lastElement = first;
} else {
// 其他的数据next是first,
element.next = first;
// 设置上一个数据的next是最后一个数据
lastElement.next = element;
// 并且更新lastElelment为最后一个数据
lastElement = element;
}
}
}


/**
* 数据
*/
public static class Element {
// 数据编号
private int no;
// 下一个为数据
private Element next;

public int getNo() {
return no;
}

public void setNo(int no) {
this.no = no;
}

public Element getNext() {
return next;
}

public void setNext(Element next) {
this.next = next;
}
}
}

}

1.3 控制台如下

移除圈的人NO:3
移除圈的人NO:6
移除圈的人NO:1
移除圈的人NO:5
移除圈的人NO:2
移除圈的人NO:8
移除圈的人NO:4