连接:7×7 Skyscrapersjava
C#答案(缘由:懒,可是彻底能够转成C++):bajdcc/learnstlgit
题目(机翻):github
在7乘7格的网格中,你只想在每一个广场上放置一个摩天大楼,只有一些线索:数组
你能写一个能解决这个难题的程序吗?函数
举个例子,6x6的:测试
6x6 摩天大楼 例优化
(有人说看不懂规则3和4,我就以图来讲明一下)3d
我根据上图制做了一个假城市(bajdcc/UnityTest),以下:code
咱们看线索中的左侧(3,4,4):
输入:int[]
输入的数组下标
输出:int[][]
花了两天思考这个系列的问题,最终固然是写出来了(光想没用)。
这个题目感受跟数独很像,我最早想的思路就是一步步推理,直到解决问题,然而,4x4大小的推理规则在6x6大小的问题中,可能须要补充一些东西或有些东西不实用了,这就麻烦了。
老实说,作数独题的时候,我不怎么用穷举法,我是对当前局面进行推理,利用排除法一步步肯定答案,但用代码实现推理就变得困难了,虽然这样作是最优的(由于根据不作多余计算)。
推理不能用,那就老老实实用穷举法吧,虽然穷举法有点low,但只要能过就行。
题目中透露了一些可供优化的信息:各行各列中数是惟一的。这就意味着复杂度从O(n^n)降低到O(n!)。这跟八皇后有点像,但又有所不一样,所以我就用回溯法作。
肯定用回溯作,也感受能够借鉴八皇后的思路,还不能立刻开工,有几个问题:
先解决全排列问题:抄了下https://github.com/bajdcc/jProlog/blob/master/src/com/bajdcc/rt/gen/array/RtNorepArray.java#L43 哈哈。
回溯的顺序应该是怎样呢?本身试喽。
先生成全排列,而后计算从左边向右看到的楼层数(假设为Sky函数,Sky:=int->int),将结果存到字典中。
会发现,当n=7时,sky(7)=1,可能性最低,都算出来后,排序。
sky(7)<sky(6)<sky(5)<sky(1)<sky(4)<sky(3)<sky(2)
展开顺序的话,我确定先找n=7的状况的,由于这时的不肯定性最低(解只有一个),至关于能够肯定一个格子的值了。
目前的思路是:找到值=7的数,就直接敲定一排解(1~7);再找值=6的数,这时答案不惟一,回溯;继续找……
这样运行程序的话,确定是耗时很长的,缘由是什么?
咱们优化程序的目标是回溯时尽能够减小回溯次数,那就必须在决定一个展开顺序:先展开不肯定性小的,再展开不肯定性较大的。
有一种状况,能够减小不肯定性:若是一排格子的两侧都有数字(都不是零),那其实能够将它们放到一块儿回溯。例: 5 | x x x x x x x | 2,这样能够极大下降可能性。事实证实这样作有一点效果。
思路有了:
注意点:
代码比较啰嗦,460行,不贴了(别人写得都很短简直了)。