介绍

链接:7×7 Skyscrapers

C#答案(原因:懒,但是完全可以转成C++):bajdcc/learnstl

题目(机翻):

在7乘7格的网格中,你只想在每个广场上放置一个摩天大楼,只有一些线索:

  • 摩天大楼的高度在1到7之间

  • 一行或一列中的所有摩天大楼楼层数量各不相同
  • 你只知道从外面的一行或一列中看到的摩天大楼的数量
  • 高层摩天大楼阻挡了后面的下层摩天大楼,导致你看不到后面的大楼了

你能写一个能解决这个难题的程序吗?

举个例子,6x6的:

6x6 摩天大楼 例

(有人说看不懂规则3和4,我就以图来说明一下)

我根据上图制作了一个假城市(bajdcc/UnityTest),如下:

虚拟城市

我们看线索中的左侧(3,4,4):

规则3和4的说明

要求

输入:int[]

输入的数组下标

输出:int[][]

分析

花了两天思考这个系列的问题,最终当然是写出来了(光想没用)。

这个题目感觉跟数独很像,我最先想的思路就是一步步推理,直到解决问题,然而,4x4大小的推理规则在6x6大小的问题中,可能需要补充一些东西或有些东西不实用了,这就麻烦了。

老实说,做数独题的时候,我不怎么用穷举法,我是对当前局面进行推理,利用排除法一步步确定答案,但用代码实现推理就变得困难了,虽然这样做是最优的(因为根据不做多余计算)。
推理不能用,那就老老实实用穷举法吧,虽然穷举法有点low,但只要能过就行。

题目中透露了一些可供优化的信息:各行各列中数是唯一的。这就意味着复杂度从O(n^n)下降到O(n!)。这跟八皇后有点像,但又有所不同,因此我就用回溯法做。

进一步思考

确定用回溯做,也感觉可以借鉴八皇后的思路,还不能马上开工,有几个问题:

  1. 全排列的生成,这个嘛拿之前的代码粘贴下

  2. 回溯遍历的顺序,这个顺序大有文章

先解决全排列问题:抄了下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,这样可以极大降低可能性。事实证明这样做有一点效果。

思路有了:

  1. 将规则排好序(数大,对位匹配优先),回溯规则

  2. 保存现场,查看是否冲突,一是当前的测试值是否与该行/列冲突,二是测试值是否满足各行/列数唯一的原则
  3. 如果失败,回溯
  4. 如果成功,level+=1,再测试下一个规则
  5. 如果level==规则总数,意味着通过了所有规则,此时还不能确定解是否正确
  6. 规则有覆盖不到的行/列,此时要将值是零的格子进行填空

  7. 填空的办法是用排除法,然后不断遍历填空,如果最后填空失败,则解不正确,如果填空成功,则解正确

注意点:

  1. 坐标系问题,方向问题

  2. 判断条件是否完备
  3. 尽量用数组做(用list<>竟然比int[]快。。)
  4. 等等细节问题

代码比较啰嗦,460行,不贴了(别人写得都很短简直了)。

https://zhuanlan.zhihu.com/p/30713476备份。

【Codewars】7×7 摩天大楼的更多相关文章

  1. [CodeWars][JS]实现链式加法

    在知乎上看到这样一个问题:http://www.zhihu.com/question/31805304; 简单地说就是实现这样一个add函数: add(x1)(x2)(x3)...(xn) == x1 ...

  2. codewars 随手记

    1.ES6数组遍历语法糖=> 在C#Linq里曾经用过,因此也不是很陌生. var range = Array.apply(null, Array(x)).map((_, i) => ++ ...

  3. window 10 用composer安装 基于 Laravel 5.3 的 Coaster CMS Curl错误77解决

    安装教程来自 http://laravelacademy.org/post/6547.html 1.系统环境windows 10 nodejs3.9.2 composer1.2.4(国内镜像) Cmd ...

  4. [stm32] 一个简单的stm32vet6驱动的天马4线SPI-1.77寸LCD彩屏DEMO

    书接上文<1.一个简单的nRF51822驱动的天马4线SPI-1.77寸LCD彩屏DEMO> 我们发现用16MHz晶振的nRF51822驱动1.77寸的spi速度达不到要求 本节主要采用7 ...

  5. [nRF51822] 1、一个简单的nRF51822驱动的天马4线SPI-1.77寸LCD彩屏DEMO

    最近用nRF51822写了个天马4线SPI的1.77寸LCD彩屏驱动,效果如下: 屏幕的规格资料为:http://pan.baidu.com/s/1gdfkr5L 屏幕的驱动资料为:http://pa ...

  6. 【hiho一下第77周】递归-减而治之 (MS面试题:Koch Snowflake)

    本题是一道微软面试题,看起来复杂,解出来会发现其实是一个很简单的递归问题,但是这道题的递归思路是很值得我们反复推敲的. 原题为hihocoder第77周的题目. 描述 Koch Snowflake i ...

  7. NYOJ题目77开灯问题

    aaarticlea/png;base64,iVBORw0KGgoAAAANSUhEUgAAAsUAAAHXCAIAAADbX7BCAAAgAElEQVR4nO3dvVLrSMAm4L0Jci6E2B

  8. LeetCode39/40/22/77/17/401/78/51/46/47/79 11道回溯题(Backtracking)

    LeetCode 39 class Solution { public: void dfs(int dep, int maxDep, vector<int>& cand, int ...

  9. Scala 深入浅出实战经典 第77讲:模式匹配下的提取器动手构造实战

    王家林亲授<DT大数据梦工厂>大数据实战视频 Scala 深入浅出实战经典(1-87讲)完整视频.PPT.代码下载:百度云盘:http://pan.baidu.com/s/1c0noOt6 ...

随机推荐

  1. 【二】Drupal 入门之新建主题

    Drupal 的模板是以  *.tpl.php  命名的 php 文件 1.在Drupal中,默认模板路径为 moudles/system 这就是我们为什么还没有制作模板 Drupal 就能够正常显示 ...

  2. 实现 1像素border

    border-1px($color) position: relative &:after display: block position: absolute left: 0 bottom: ...

  3. 在云主机或vps上用bzr拉OpenERP7.0代码

    前面的文章讲过了用bzr来参与OpenERP开发的过程,其中很重要的一步就是创建本地分支.我在阿里云上建server和web的分支都没有问题,就是addons分支搞了30几次都在中途被kill了. 今 ...

  4. Oracle 12c CDB PDB

    先说基本用法: 先按11G之前进行 conn / as sysdba; create user test identifed by test; ORA-65096: 公用用户名或角色名无效. 查官方文 ...

  5. 公钥私钥与SSL的握手协议(转)

    一,公钥私钥1,公钥和私钥成对出现2,公开的密钥叫公钥,只有自己知道的叫私钥3,用公钥加密的数据只有对应的私钥可以解密4,用私钥加密的数据只有对应的公钥可以解密5,如果可以用公钥解密,则必然是对应的私 ...

  6. Oracle创建库

    oracle创建表空间 SYS用户在CMD下以DBA身份登陆: 在CMD中打sqlplus /nolog 然后再 conn / as sysdba --如果路径不存在则要创建路径 --创建临时表空间 ...

  7. no !/ in spec

    问题: 在学习hibernate的过程中涉及到hibernate.cfg.xml和hibernate.hbm.xml配置文件的编写,写完之后配置文件出现黄色感叹号,鼠标悬停在感叹号上时显示no !/ ...

  8. HTML:调用静态页面html 的几种方法

    今天做办公用品管理系统时,发现需要用到在一个静态页面html 中调用多个静态页面html的内容.查找资料总结了以下一些方法: 一.iframe引入的方法 代码如下: <!-- 部门--> ...

  9. Xcode8的调试技能Memory Graph 实战解决闭包引用循环问题

    Xcode8的调试技能又增加了一个黑科技:Memory Graph.简单的说就是可以在运行时将内存中的对象生成一张图. 那么通过一个实际项目来练习一下吧. 首先我们写了一个自定义UIView:MyVi ...

  10. Windows右键菜单设置与应用技巧

    http://blog.163.com/lee_honleon/blog/static/555251522008014258896/   需要修改注册表,挺麻烦的.最好还是把迅雷卸了重装,不会影响什么 ...