- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
八皇后难题是将八个国际象棋皇后放置在一个8×8的棋盘上,使两个皇后之间不会相互威胁的问题。因此,解决方案要求没有两个皇后共享相同的行、列或对角线。八皇后难题是更一般的 n 皇后问题的一个例子,即在 n×n 棋盘上放置 n 个皇后,其中除了 n=2 或 n=3 之外的所有自然数 n 都存在解。
但是,有没有人可以帮我解决这个在递归方法中无限循环的问题?
ps:你可以直接复制/粘贴到playground试试,谢谢!
class ChessBoard {
var limit: Int
var queens = [Queen]()
init(limit: Int) {
self.limit = limit
}
// Check if (i,j) is a safe position for queen
func isSafeForQueen(atRow row: Int, col: Int) -> Bool {
for q in queens {
// not in same row
if q.row == row { return false }
// not in same column
if q.col == col { return false }
// not in same diagol line
if abs(q.row-row) == abs(q.col-col) { return false }
}
return true
}
// recursive method
func dropQueen(atRow r: Int, c: Int) {
// running into last row
if r == limit {
if queens.count < 8 {
queens.removeLast()
let q = queens.last!
dropQueen(atRow: q.row, c: q.col+1)
}
output() // if success, log out the positions
return
}
// running into last column of current row
if c == limit {
queens.removeLast()
let q = queens.last!
// if no position for queen at current row, then back to last row
dropQueen(atRow: r-1, c: q.col+1)
}
// if this postion is safe for queen, then drop the queen and try next row; if not, try next spot
if isSafeForQueen(atRow: r, col: c) {
let q = Queen(row: r, col: c)
queens.append(q)
dropQueen(atRow: r+1, c: c)
} else {
dropQueen(atRow: r, c: c+1)
}
}
func play() {
dropQueen(atRow: 0, c: 0) // game will start at(0,0)
}
func output() -> String {
var s = ""
for q in queens {
s += "(\(q.row),\(q.col))"
}
return s
}
}
struct Queen {
var row: Int
var col: Int
}
// Tesing:
let b = ChessBoard(limit: 8)
//b.play()
最佳答案
看来您在理解递归的概念时遇到了困难。递归并不意味着一切 都变成递归调用。在函数 dropQueen
中,您正在使用递归调用,就像它们是 goto 一样。
尤其是这个:
dropQueen(atRow: r-1, c: q.col+1)
这显然是一种回溯的尝试。下定决心;要么做回溯要么使用递归!在递归中,回到先前的决定是一个return
。如有必要,返回一个 bool 值让调用者知道递归调用是否找到了解决方案 (return true
) 或没有 (return false
)。如果您希望您的程序找到所有 可能的解决方案,而不仅仅是第一个,则不需要它。
另一个引起我注意的电话:
dropQueen(atRow: r, c: c+1)
递归调用在这里有点矫枉过正,而且令人困惑;使用循环扫描所有可能的列。顺便说一句,通过进行此更改,您会发现整个第二个函数参数变得多余。
这只给我们留下了一个回溯调用;这绰绰有余。
dropQueen(atRow: r+1, c: c)
正如 vacawama 所指出的,第二个参数 c: c
似乎是错误的。当您前进一行 (r+1
) 时,为什么要排除您放在棋盘上的最后一个皇后左边的所有内容?
如果你想做递归,那么想想你的递归函数应该做什么。通常,您希望它在已经被 r 个皇后占据的棋盘上放置 (8-r) 个皇后。您正在逐行工作,因此在您的方法中 r 只是当前行号。想一想如何根据第 r+1 行的解决方案来表达第 r 行问题的解决方案。为最后一行做一个特殊的异常(exception);一旦 r 等于极限,解决方案就很简单(需要放置零皇后);你完成了。
哦,请重命名函数 dropQueen
;该名称清楚地表明您正在以错误的心态从事此工作。到现在为止,您应该能够想出更合适的东西。
编辑:您为完成这项工作付出了相当大的努力,因此我将与您分享我的解决方案。它使用您的原始代码(如您的问题所示),但函数 dropQueen
完全重写(并且少了一个参数)。请注意它是多么简短;这是典型的递归。
func dropQueen(atRow r: Int) {
if queens.count < limit {
for col in 0...limit-1 {
if isSafeForQueen(atRow: r, col: col) {
let q = Queen(row: r, col: col)
queens.append(q)
dropQueen(atRow: r+1)
if queens.count == limit {
return
}
queens.removeLast()
}
}
}
}
作为奖励,如果将 return
替换为 println(output())
,则程序将打印所有 92 个解决方案。您可以在以下位置查看实际效果: http://swiftstub.com/923601919/
关于swift - 这是我解决八皇后难题的快速代码,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/27470703/
如何更改循环中变量的名称?比如 number1 、 number2 、 number3 、 number4 ? var array = [2,4,6,8] func ap ( number1: Int
我想设置 View 的背景颜色并在一定延迟后将其更改为另一种颜色。这是我的尝试方式: print("setting color 1") self.view.backgroundColor = UICo
我在使用 express-session 时遇到问题。 session 数据不会在请求之间持续存在。 正如您在下面的代码中看到的那样,/join 路由设置了一些 session 属性,但是当 /sur
我试图从叶渲染器获得一个非常简单的结果,用于快速 Steam 的 for 循环。 我正在上传叶文件 HTML,因为它不接受此处格式正确的代码 - 下面的pizza.swift代码- import
你们中有人有什么好的链接可以与我分享吗?我正在寻找一个 FAST 程序员编辑器,它可以非常快速地打开包含超过 100, 000 行代码的文件?我目前正在使用记事本自动取款机,打开一个 29000 行长
我现在正在处理眼动追踪数据,因此拥有一个巨大的数据集(想想数百万行),因此希望有一种快速的方法来完成此任务。这是它的简化版本。 数据告诉您眼睛在每个时间点正在查看的位置以及我们正在查看的每个文件。 X
我是新手,想为计时器或其他设备选择提示音。 如何打开此列表,以选择其中一种声音? Alert sound list 最佳答案 您将无法在应用中使用系统声音。 但是,您可以包括自己的声音文件,并将其显示
我编写了以下代码来构建具有顺序字符串的数组。 它的工作方式与我预期的一样,但我希望它能更快地运行。有没有更有效的方法在PowerShell中产生我想要的结果? 我是PowerShell的新手,非常感谢
我有一个包含一些非唯一行的矩阵,例如: x 尝试 y <- rle(apply(x, 1, paste, collapse = " ")) # y$lengths is the vector con
我的函数“keyboardWillShown”有问题。所以我想要的是菜单打开时,菜单正好出现在键盘上方。它可以在Iphone 8 plus,8、7、6上完美运行。但是,当我在模拟器上运行Iphone
我正在尝试通过Swift 5中的HTTP get方法从API提取数据。它在启动时成功加载了数据,但是当我刷新页面时,它说“索引超出范围”,这是因为数据是不再会在我的日志中读取,因此索引中没有任何内容。
我想做什么: 从我的数据库中获取时间戳并将其转换为用户的时区。 我的代码: let tryItNow = "\(model.timestampName)" let format = D
给定字体名称和字体大小,如何查找字符串的宽度(CGFloat)? (目标是将UIView的宽度设置为足以容纳字符串的宽度。) 我有两个字符串:一个重复“1”,重复36次,另一个重复“M”,重复36次。
我正在尝试解析此JSON ["Items": ( { AccountBalance = 0; AlphabetType = 3; Description = "\U0631\U
我在UINavigationBar内放置了一个UILabel。 我想根据navigationBar的高度增加该标签的字体大小。当navigationBar很大时,我希望字体大小更大;当滚动并缩小nav
我想将用户输入限制为仅有效数字并使用以下内容: func textView(_ textView: UITextView, shouldChangeTextIn range: NSRange, rep
目前我有一个包含超过 100.000 张图像的数据库,它们大小不一或类似,但我想为我的公司制作以下内容: 我插入/上传一张图片,系统返回最有可能相同的图片。我不知道使用什么算法,但它需要快速。我可以预
在我的 swift 项目中,我有一个按钮,我想在标签上打印按下该按钮的时间。 如何解决这个问题? 最佳答案 添加到DHEERAJ的答案中,您只需在func press(sender: UIButton
我必须发表评论,尝试在解析中导入数组。然而,有一个问题。 当我尝试从 Parse 加载数组时,我的输出是 ("Blah","Blah","Blah")这是一个元组...而不是一个数组 TT... 如何
我的应用程序有一个名为 MyDevice 的类,我用它来与硬件通信。该硬件是可选的,实例变量也是可选的: var theDevice:MyDevice = nil 然后,在应用程序中,我必须初始化设备
我是一名优秀的程序员,十分优秀!