Leetcode N-Queens I

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.."]
]

写非递归回溯法要点:
1 记录好是否发生递归状态了
2 递归之后不能重复之前试过的解决方案
本题利用一个vector<int> backtrackCol(n,-1)实现了这两个功能
3 清楚什么时候需要回溯
本题是if (col == n) 这个条件就回溯了
4 回溯,就需要重置填过了的内容
本题需要重置两个数组: vector<int> backtrackCol和board
Iterative算法大概比recurrence算法难度高上三倍吧,而效率并没有明显区别。

下面是非递归算法12 queens用时也是12秒左右,和递归没什么区别。

可以打印的程序在下面连接:
http://blog.csdn.net/kenden23/article/details/14455915

vector<vector<string> > solveNQueens(int n) 
	{
		vector<string> board(n,string(n, '.'));
		vector<vector<string> > rs;
		int row = 0;
		vector<int> backtrackCol(n, -1);
		while (row >= 0)
		{
			int col = 0;
			if (backtrackCol[row] != -1)
			{
				col = backtrackCol[row]+1;
				board[row][col-1] = '.';
			}
			for (; col < n; col++)
			{
				if (isLegal(board, row, col))
				{
					board[row][col] = 'Q';
					backtrackCol[row] = col;
					if (row == n-1) 
					{
						rs.push_back(board);
						backtrackCol[row] = -1;
						board[row][col] = '.';
						row--;
						break;
					}
					row++;
					break;
				}
			}
			if (col == n)
			{
				backtrackCol[row] = -1;
				row--;
			}
		}
		return rs;
	}
	bool isLegal(const vector<string> &board, int row, int col)
	{
		for (int i = 0; i < row; i++)
		{
			if (board[i][col] == 'Q') return false;
		}
		for (int i = row - 1, j = col - 1; i>=0 && j>=0; i--, j--)
		{
			if (board[i][j] == 'Q') return false;
		}
		for (int i = row - 1, j = col + 1; i>=0 && j<board.size(); i--, j++)
		{
			if (board[i][j] == 'Q') return false;
		}
		return true;
	}

下面是递归回溯法,比上面简单多了。

vector<vector<string> > solveNQueens(int n) 
	{
		vector<string> board(n,string(n, '.'));
		vector<vector<string> > rs;
		solNQueens(rs, board);
		return rs;
	}
	void solNQueens(vector<vector<string> > &rs, vector<string> &board, int row=0)
	{
		if (row == board.size()) rs.push_back(board);

		for (int col = 0; col < board.size(); col++)
		{
			if (isLegal(board, row, col))
			{
				board[row][col] = 'Q';
				solNQueens(rs, board, row+1);
				board[row][col] = '.';
			}	
		}
	}

	bool isLegal(const vector<string> &board, int row, int col)
	{
		for (int i = 0; i < row; i++)
		{
			if (board[i][col] == 'Q') return false;
		}
		for (int i = row - 1, j = col - 1; i>=0 && j>=0; i--, j--)
		{
			if (board[i][j] == 'Q') return false;
		}
		for (int i = row - 1, j = col + 1; i>=0 && j<board.size(); i--, j++)
		{
			if (board[i][j] == 'Q') return false;
		}
		return true;
	}


递归回溯:

class Solution128 {
public:
	vector<vector<string> > solveNQueens(int n) 
	{
		vector<vector<string> > rs;
		vector<string> s(n, string(n, '.'));
		solve(rs, s, n);
		return rs;
	}

	void solve(vector<vector<string> > &rs, vector<string> &s, int n, int r = 0)
	{
		if (r == n)
		{
			rs.push_back(s);
			return;
		}

		for (int c = 0; c < n; c++)
		{
			if (isLegal(s, r, c))
			{
				s[r][c] = 'Q';
				solve(rs, s, n, r+1);
				s[r][c] = '.';
			}
		}
	}
	bool isLegal(vector<string> &s, int r, int c)
	{
		for (int i = r - 1, j = c - 1, k = c + 1; i >= 0 ; i--, j--, k++)
		{
			if (s[i][c] == 'Q' 
				|| j>=0 && s[i][j] == 'Q' 
				|| k < s.size() && s[i][k] == 'Q')
				return false;
		}
		return true;
	}
};














  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
对于计算机专业的学生而言,参加各类比赛能够带来多方面的益处,具体包括但不限于以下几点: 技能提升: 参与比赛促使学生深入学习和掌握计算机领域的专业知识与技能,如编程语言、算法设计、软件工程、网络安全等。 比赛通常涉及实际问题的解决,有助于将理论知识应用于实践中,增强问题解决能力。 实践经验: 大多数比赛都要求参赛者设计并实现解决方案,这提供了宝贵的动手操作机会,有助于积累项目经验。 实践经验对于计算机专业的学生尤为重要,因为雇主往往更青睐有实际项目背景的候选人。 团队合作: 许多比赛鼓励团队协作,这有助于培养学生的团队精神、沟通技巧和领导能力。 团队合作还能促进学生之间的知识共享和思维碰撞,有助于形成更全面的解决方案。 职业发展: 获奖经历可以显著增强简历的吸引力,为求职或继续深造提供有力支持。 某些比赛可能直接与企业合作,提供实习、工作机会或奖学金,为学生的职业生涯打开更多门路。 网络拓展: 比赛是结识同行业人才的好机会,可以帮助学生建立行业联系,这对于未来的职业发展非常重要。 奖金与荣誉: 许多比赛提供奖金或奖品,这不仅能给予学生经济上的奖励,还能增强其成就感和自信心。 荣誉证书或奖状可以证明学生的成就,对个人品牌建设有积极作用。 创新与研究: 参加比赛可以激发学生的创新思维,推动科研项目的开展,有时甚至能促成学术论文的发表。 个人成长: 在准备和参加比赛的过程中,学生将面临压力与挑战,这有助于培养良好的心理素质和抗压能力。 自我挑战和克服困难的经历对个人成长有着深远的影响。 综上所述,参加计算机领域的比赛对于学生来说是一个全面发展的平台,不仅可以提升专业技能,还能增强团队协作、沟通、解决问题的能力,并为未来的职业生涯奠定坚实的基础。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值