结合问题说方案,首先先说问题:

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

嗯,这个问题已经被使用各种语言解答一万遍了,大多还是回溯法解决的。

关于回溯算法:个人理解为就是优化的穷举算法,穷举算法是指列出所有的可能情况,而回溯算法则是试探发现问题"剪枝"回退到上个节点,换一条路,能够大大提高求解效率。

具体到8皇后问题上来说,需要考虑以下几点:

1)将8个皇后定义为8行中的相对位置来标识,考虑增加新的皇后时,是否与之前的皇后位置冲突(即可以攻击之前摆放的皇后:位置相等或者斜率1or-1)

2)新放的皇后发生冲突时回溯至上一行继续试探,逐步回溯直至第一行为止

3)已经求出的解再次探索时避免重复

4)从第一行开始放皇后,然后开始循环往下放,可以设计为回调放皇后的方法

说了这么多废话,开始写吧,啪啪啪-- 12秒过去了,写完了,运行-----嘛结果也没有!!

贴上代码及注释

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text; namespace Queen
{
class Program
{
//定义解的个数
int sum = 0;
//定义皇后数组
int[] Queens = new int[8];
static void Main(string[] args)
{
Program Pro = new Program();
//开始求解
Pro.QueenSort(0);
}
//排序获取组合(1-8)
public void QueenSort(int num)
{ for (int j = 1; j < 9; j++)
{
if (num == 8)
{
sum++;
//打印输出
Write();
break;
}
Queens[num] = j;
//判断是否冲突
if (FooConflict(num, j))
{
num++;
QueenSort(num);
}
}
} /// <summary>
/// 判断皇后是否和之前所有的皇后冲突
/// </summary>
/// <param name="row">已放置完毕无冲突皇后的列数</param>
/// <param name="queen">新放置的皇后值</param>
/// <returns>是否冲突</returns>
public bool FooConflict(int row, int queen)
{
if (row == 0)
{
return true;
}
else
{
//循环判断与之前的皇后是否有冲突的
for (int pionter = 0; pionter < row; pionter++)
{
//如果有,返回false
if (!FooCompare(Queens[pionter], row - pionter, queen))
{
return false;
}
}
//与之前均无冲突,返回true
return true;
}
}
/// <summary>
/// 对比2个皇后是否冲突
/// </summary>
/// <param name="i">之前的一个皇后</param>
/// <param name="row">2个皇后的列数之差</param>
/// <param name="queen">新放置的皇后</param>
/// <returns></returns>
public bool FooCompare(int i, int row, int queen)
{
//判断2个皇后是否相等或者相差等于列数之差(即处于正反对角线)
if ((i == queen) || ((i - queen) == row) || ((queen - i) == row))
{
return false;
}
return true;
}
//打印皇后图案
public void Write()
{
//输出皇后的个数排序
Console.WriteLine("第{0}个皇后排列:", sum);
for (int i = 0; i < 8; i++)
{
for (int j = 1; j < 9; j++)
{
if (j == Queens[i])
{
Console.Write("■");
}
else
{
Console.Write("□");
}
}
//换行
Console.Write("\n");
}
}
}
}

PS:还好我写的方法分的很细,直接锁定QueenSort()这个方法,嗯,一定是它出了问题!

仔细一看num++这一行,本意是循环QueenSort(num+1)查询下一个皇后的解,这样写导致下次循环赋值Queens[num]出现了异常,果断改了,运行OK!

贴上代码:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text; namespace Queen
{
class Program
{
//定义解的个数
int sum = 0;
//定义皇后数组
int[] Queens = new int[8];
static void Main(string[] args)
{
Program Pro = new Program();
//开始求解
Pro.QueenSort(0);
}
//排序获取组合(1-8)
public void QueenSort(int num)
{
for (int j = 1; j < 9; j++)
{
if (num == 8)
{
sum++;
//打印输出
Write();
break;
}
//判断是否冲突
if (FooConflict(num, j))
{
Queens[num] = j;
QueenSort(num+1);
}
}
} /// <summary>
/// 判断皇后是否和之前所有的皇后冲突
/// </summary>
/// <param name="row">已放置完毕无冲突皇后的列数</param>
/// <param name="queen">新放置的皇后值</param>
/// <returns>是否冲突</returns>
public bool FooConflict(int row, int queen)
{
if (row == 0)
{
return true;
}
else
{
//循环判断与之前的皇后是否有冲突的
for (int pionter = 0; pionter < row; pionter++)
{
//如果有,返回false
if (!FooCompare(Queens[pionter], row - pionter, queen))
{
return false;
}
}
//与之前均无冲突,返回true
return true;
}
}
/// <summary>
/// 对比2个皇后是否冲突
/// </summary>
/// <param name="i">之前的一个皇后</param>
/// <param name="row">2个皇后的列数之差</param>
/// <param name="queen">新放置的皇后</param>
/// <returns></returns>
public bool FooCompare(int i, int row, int queen)
{
//判断2个皇后是否相等或者相差等于列数之差(即处于正反对角线)
if ((i == queen) || ((i - queen) == row) || ((queen - i) == row))
{
return false;
}
return true;
}
//打印皇后图案
public void Write()
{
//输出皇后的个数排序
Console.WriteLine("第{0}个皇后排列:", sum);
for (int i = 0; i < 8; i++)
{
for (int j = 1; j < 9; j++)
{
if (j == Queens[i])
{
Console.Write("■");
}
else
{
Console.Write("□");
}
}
//换行
Console.Write("\n");
}
}
}
}

运行结果如下图:

OK! 等等 为毛百度了一下C语言的实现只有几行!! 不过想想咱们的核心代码排序方法也就几行,还行吧,以后有空再考虑优化下

that’s all !

回溯算法-C#语言解决八皇后问题的写法与优化的更多相关文章

  1. C语言解决八皇后问题

    #include <stdio.h> #include <stdlib.h> /* this code is used to cope with the problem of ...

  2. 使用穷举法结合numpy解决八皇后问题

    一般说到八皇后问题,最先想到的就是回溯思想,而回溯思想往往是需要递归来实现的. 计算机很善长做重复的事情,所以递归正和它的胃口,而我们人脑更喜观平铺直叙的思维方式.当 我们看到递归时,总想把递归平铺展 ...

  3. Python解决八皇后问题

    最近看Python看得都不用tab键了,哈哈.今天看了一个经典问题--八皇后问题,说实话,以前学C.C++的时候有这个问题,但是当时不爱学,没搞会,后来算法课上又碰到,只是学会了思想,应该是学回溯法的 ...

  4. Python 解决八皇后问题

    问题介绍 八皇后问题是一个以国际象棋为背景的问题:如何能够在 \(8\times8\) 的国际象棋棋盘上放置八个皇后,使得任何一个皇后都无法直接吃掉其他的皇后?为了达到此目的,任两个皇后都不能处于同一 ...

  5. 8皇后以及N皇后算法探究,回溯算法的JAVA实现,非递归,循环控制及其优化

    上两篇博客 8皇后以及N皇后算法探究,回溯算法的JAVA实现,递归方案 8皇后以及N皇后算法探究,回溯算法的JAVA实现,非递归,数据结构“栈”实现 研究了递归方法实现回溯,解决N皇后问题,下面我们来 ...

  6. LeetCode 回溯法 别人的小结 八皇后 递归

    #include <iostream> #include <algorithm> #include <iterator> #include <vector&g ...

  7. Python解决八皇后问题的代码【解读】

    八皇后问题 来自于西方象棋(现在叫 国际象棋,英文chess),详情可见百度百科. 在西方象棋中,有一种叫做皇后的棋子,在棋盘上,如果双方的皇后在同一行.同一列或同一斜线上,就会互相攻击. 八皇后问题 ...

  8. 使用java语言实现八皇后问题

    八皇后问题,在一个8X8的棋盘中,放置八个棋子,每个棋子的上下左右,左上左下,右上右下方向上不得有其他棋子.正确答案为92中,接下来用java语言实现. 解: package eightQuen; / ...

  9. C语言:试探算法解决“八皇后”问题

    #include <stdio.h> #define N 4 int solution[N], j, k, count, sols; int place(int row, int col) ...

随机推荐

  1. SQL server 中 COUNT DISTINCT 函数

    目的:统计去重后表中所有项总和. 直观想法: SELECT COUNT(DISTINCT *) FROM [tablename] 结果是:语法错误. 事实上,我们可以一同使用 DISTINCT 和 C ...

  2. bootstrap 兼容IE8设置

    <!--[if lt IE 9]> <script src="//cdn.bootcss.com/html5shiv/3.7.2/html5shiv.min.js" ...

  3. js实现a标签超链接提交form表单的方法

    <a class="regButton"    id="saveRegister" onclick="document.getElementBy ...

  4. php随机函数

    <?php function generate_password( $length = 6 ) { // 密码字符集,可任意添加你需要的字符 // $chars = 'abcdefghijklm ...

  5. web前端之 CSS引入第三方插件

    引入第三方图标插件 - fontawesome 官网地址:http://fontawesome.io/ 1.下载图标插件包 下载地址:https://codeload.github.com/FortA ...

  6. python学习之路-8 面向对象之进阶

    上篇内容回顾和补充 面向对象三大特性 封装 继承 多态 在python中没有多态的概念 变量的类型允许为多种数据类型称之为多态 # c#/java中的多态 # 伪代码 def func(int arg ...

  7. (转)Tomcat 7 访问 Manager 和 Host Manager

    配置好 Tomcat 7.0 后,在 tomcat-users.xml 中配置用户角色来访问 localhost:8080 的这样三个按钮总出现问题: Server Status Manager Ap ...

  8. iOS多线程及其感悟

    感觉每天都是匆匆忙忙的,每天似乎都是时间不够用一样,可是等真的想要动手敲代码的时候才发现,原来还有好多好多的知识点不是太熟练,所以,人不可以一直感觉自我良好, 有时间就是那种自我感觉良好的心态毁了自己 ...

  9. Hacker(七)----黑客常用术语和DOS命令

    掌握基本的黑客术语和DOS命令是一名黑客最基本的技能,黑客术语能够实现自己和其他人之间的正常交流.DOS命令就是DOS操作系统的命令,它是一种面向磁盘的操作命令.黑客在入侵目标主机的过程中经常会使用这 ...

  10. TNS-00512: Address already in use-TNS-12542: TNS:address already in use

    监听启动或是停止时提示如下错误:TNS-12542: TNS:address already in use TNS-12560: TNS:protocol adapter error TNS-0051 ...