Loading [MathJax]/jax/output/CommonHTML/config.js
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
社区首页 >问答首页 >数独解递归解

数独解递归解
EN

Code Review用户
提问于 2017-02-20 17:00:50
回答 2查看 2.5K关注 0票数 8

下面是我在Python2.7中为数独解析器编写的代码。任何关于性能改进、代码错误或一般代码风格建议的建议都将受到欢迎。

我的主要想法是:

  1. 使用方法生成一些随机数据,然后使用check_invalid查看它是否是有效的Sudoku。
  2. 如果从1开始,它是一个有效的Sudoku,那么我将调用resolve_sudoku来填充有效值。

我发现我的代码有时运行很长时间而没有完成。你能找到什么代码错误吗?

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
import random
from collections import defaultdict
found = False
def check_invalid(matrix):
    # check for each row
    for row in range(len(matrix)):
        cur_row = set()
        for col in range(len(matrix[0])):
            if matrix[row][col] == 0:
                continue
            elif 1 <= matrix[row][col] <= 9:
                if matrix[row][col] in cur_row:
                    return False
                else:
                    cur_row.add(matrix[row][col])
            else:
                return False # invalid number
    # check each col
    for col in range(len(matrix[0])):
        cur_col = set()
        for row in range(len(matrix)):
            if matrix[row][col] == 0:
                continue
            elif 1 <= matrix[row][col] <= 9:
                if matrix[row][col] in cur_col:
                    return False
                else:
                    cur_col.add(matrix[row][col])
            else:
                return False # invalid number
    # check each 3*3 square
    for start_row in [0,3,6]:
        for start_col in [0,3,6]:
            cur_square = set()
            for row in range(start_row, start_row+3):
                for col in range(start_col, start_col + 3):
                    if matrix[row][col] == 0:
                        continue
                    elif 1 <= matrix[row][col] <= 9:
                        if matrix[row][col] not in cur_square:
                            cur_square.add(matrix[row][col])
                        else:
                            return False
                    else:
                        return False # invalid value
    return True

def resolve_sudoku(matrix, row_map, col_map, square_map, cur_row, cur_col):
    global found
    if found:
        return
    if cur_row == len(matrix):
        found = True
        for r in matrix:
            print r
    elif cur_col == len(matrix[0]):
        resolve_sudoku(matrix, row_map, col_map, square_map, cur_row+1, 0)
    elif matrix[cur_row][cur_col] != 0:
        resolve_sudoku(matrix, row_map, col_map, square_map, cur_row, cur_col+1)
    else:
        for val in range(1,10):
            square_x = cur_row / 3
            square_y = cur_col / 3
            if val in row_map[cur_row] or val in col_map[cur_col] or val in square_map[(square_x, square_y)]:
                continue
            else:
                row_map[cur_row].add(val)
                col_map[cur_col].add(val)
                square_map[(square_x, square_y)].add(val)
                matrix[cur_row][cur_col] = val
                resolve_sudoku(matrix, row_map, col_map, square_map, cur_row, cur_col+1)
                row_map[cur_row].remove(val)
                col_map[cur_col].remove(val)
                square_map[(square_x, square_y)].remove(val)
                matrix[cur_row][cur_col] = 0
                if found:
                    return
if __name__ == "__main__":
    matrix = []
    for row in range(9):
        cur_row = []
        for col in range(9):
            if random.random() < 0.1:
                cur_row.append(random.randint(1,9))
            else:
                cur_row.append(0)
        matrix.append(cur_row)
    for r in matrix:
        print r
    re = check_invalid(matrix)
    print re
    if re:
        # init for row map and col map
        row_map = defaultdict(set)
        col_map = defaultdict(set)
        square_map = defaultdict(set)
        for row in range(len(matrix)):
            for col in range(len(matrix[0])):
                square_x = row / 3
                square_y = row / 3
                if matrix[row][col] != 0:
                    row_map[row].add(matrix[row][col])
                    col_map[col].add(matrix[row][col])
                    square_map[(row, col)].add(matrix[row][col])
        resolve_sudoku(matrix, row_map, col_map, square_map, 0, 0)
EN

回答 2

Code Review用户

回答已采纳

发布于 2017-03-04 21:24:59

让我们把它一片片地拆开。

生成矩阵

首先,将“生成潜在sudoku矩阵”移动到一个单独的函数中。而且,我们可以通过使用清单理解来改进它:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
def generate_sudoku_matrix():
    """Generates a 9x9 matrix with random numbers from 1 to 9, leaving 0s as unsolved."""
    return [[0 if random.random() < 0.1 else random.randint(1, 9)
             for _ in range(9)]
            for _ in range(9)]

注意_用于丢弃变量的使用。

验证矩阵

首先,我将调用函数check_valid()而不是check_invalid(),因为在矩阵有效的情况下,您将返回True。不过,更好的名称可能是类似于is_valid_sudoku()

该函数本身过于复杂,而且也违反了干原理,因为您基本上对行、列和3x3平方重复相同的检查。

如果我们定义一个可重用的函数来验证一个“块”(行、列或正方形),该怎么办:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
def is_valid_block(block):
    """Returns true if a block (row, column or square) is a valid Sudoku block."""
    return len(block) == 9 and sum(block) == sum(set(block))

并将其应用于以下3种情况:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
from itertools import chain


def is_valid_sudoku(matrix):
    """Returns True if a Sudoku matrix is valid, False otherwise."""

    # check each row
    for row in matrix:
        if not is_valid_block(row):
            return False

    # check each column
    for col in zip(*matrix):
        if not is_valid_block(col):
            return False

    # check each 3x3 square
    for i in range(0, 9, 3):
        for j in range(0, 9, 3):
            square = list(chain(row[j:j + 3] for row in matrix[i:i + 3]))
            if not is_valid_block(square):
                return False

    return True

我们还可以更进一步,利用生成器和all()

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
def generate_blocks(matrix):
    """Generates rows, columns and 3x3 squares for a Sudoku matrix."""
    for row in matrix:
        yield row

    for col in zip(*matrix):
        yield col

    for i in range(0, 9, 3):
        for j in range(0, 9, 3):
            yield list(chain(*(row[j:j + 3] for row in matrix[i:i + 3])))


def is_valid_sudoku(matrix):
    """Returns True if a Sudoku matrix is valid, False otherwise."""
    return all(is_valid_block(block) for block in generate_blocks(matrix))

解数独

首先,将初始映射设置移动到单独的函数,以保持"main“函数的干净和可读性。

我不喜欢使用全局found变量处理退出函数的方式。它使得很难跟踪程序的流程,并且损害了模块性和关注点的分离。

另一个需要改进的地方是临时映射和矩阵修改和回滚。不过,我不确定复制数据结构以将其传递给下一个递归调用是否会更干净和更快。

我以前从未编写过sudoku求解器,但下面是我对实现蛮力解决方案的想法(我想这只是一种不同的方法,与代码评审无关):

  • 定义is_valid_solved_sudoku()函数,该函数将验证一个矩阵是否为有效求解的sudoku矩阵。该函数将与is_valid_sudoku()非常相似,除了检查每个“块”的方式--不再允许使用0
  • 然后,is_valid_solved_sudoku()正结果将是我们的递归基条件。
  • 然后,在每个递归调用中,我们找到下一个未解决的单元格(带有0值)的位置。
  • 对于每个row_index,0单元格的col_index,从相应的行、列和平方中获取一组不可能的值/排除数字。
  • 对每个数进行递归调用,不包括我们在上一步构造的一组排除数中的数字

代码风格及其他问题和建议:

  • 使用漂亮印刷pprint代替常规打印
  • 脚本“主”部分中未使用的square_xsquare_y变量(实际上可能是错误/错误,因为我希望您使用它们为square_map字典创建一个键)
  • 按PEP8 (参考文献)组织导入
  • 函数之间的两行空行(参考文献)
票数 4
EN

Code Review用户

发布于 2017-03-05 01:57:51

至于性能,这并不是因为您的算法(非常高效),而是其中的几个bug:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
def resolve_sudoku(matrix, row_map, col_map, square_map, cur_row, cur_col):
    global found
    if found:
        return
    if cur_row == len(matrix):
        found = True
        for r in matrix:
            print r
    elif cur_col == len(matrix[0]):
        resolve_sudoku(matrix, row_map, col_map, square_map, cur_row+1, 0)
    elif matrix[cur_row][cur_col] != 0:
        resolve_sudoku(matrix, row_map, col_map, square_map, cur_row, cur_col+1)
    else:
        for val in range(1,10):
            # square_x = cur_row / 3 # <- bug
            square_x = cur_col / 3
            # square_y = cur_col / 3 # <- bug
            square_y = cur_row / 3
            if val in row_map[cur_row] or val in col_map[cur_col] or val in square_map[(square_x, square_y)]:
                continue
            else:
                row_map[cur_row].add(val)
                col_map[cur_col].add(val)
                square_map[(square_x, square_y)].add(val)
                matrix[cur_row][cur_col] = val
                resolve_sudoku(matrix, row_map, col_map, square_map, cur_row, cur_col+1)
                row_map[cur_row].remove(val)
                col_map[cur_col].remove(val)
                square_map[(square_x, square_y)].remove(val)
                matrix[cur_row][cur_col] = 0
                if found:
                    return
票数 1
EN
页面原文内容由Code Review提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://codereview.stackexchange.com/questions/155890

复制
相关文章
成为更好程序员的8种途径
  是时候开始认真考虑一下如何升级你的开发技术了。让我们来认真地学习一下吧。   给自己设定一个提高开发技术的目标很容易,但是“想成为一名伟大的程序员”却不是一个容易实现的目标。首先,说“我想变得更好
用户1289394
2018/02/28
6920
成为更好程序员的8种途径
成为更好程序员的8种途径
▲ 是时候开始认真考虑一下如何升级你的开发技术了。让我们来认真地学习一下吧。 给自己设定一个提高开发技术的目标很容易,但是“想成为一名伟大的程序员”却不是一个容易实现的目标。首先,说“我想变得更好”,是建立在你认识到“更好”的样子基础之上。另外,有太多的人追求进步而不知道如何去实现。 因此,让我分享八个可实际操作的指导方针,你可以把它们作为提高编程技能的流程图。这些智慧都是伴随着计算机35年的发展沉淀下来的。 1.时刻提醒自己:学习 学习某件事的第一步是承认你不知道。这听起来很正常,但经验丰富的程序员还记得
企鹅号小编
2018/01/16
6090
2014,成为更好程序员的7个方法
英文原文:7 Ways to be a Better Programmer in 2014
Isaac Zhang
2019/09/11
4150
一名普通的程序员进阶成为伟大程序员有哪8种途径?
本文作者讲述了8种方式帮助你如何从一名普通的程序员进阶成为一名伟大的程序员,让我们就从此时此刻开始提高自己的开发技能吧。 是时候开始认真考虑一下如何升级你的开发技术了。让我们来认真地学习一下吧。 给自己设定一个提高开发技术的目标很容易,但是“想成为一名伟大的程序员”却不是一个容易实现的目标。首先,说“我想变得更好”,是建立在你认识到“更好”的样子基础之上。另外,有太多的人追求进步而不知道如何去实现。 因此,让我分享八个可实际操作的指导方针,你可以把它们作为提高编程技能的流程图。这些智慧都是伴随着计算机3
CSDN技术头条
2018/02/13
1.1K0
一名普通的程序员进阶成为伟大程序员有哪8种途径?
成为一名更好的程序员:如何阅读源代码
成为一名更好的程序员:如何阅读源代码 阅读源代码有许多益处。你会发现新的架构(construct)和库,与其他的代码维护者产生共鸣,但最重要的是学会如何组织代码,避免因内部极其复杂而变得不可维护。 但
用户1289394
2018/02/27
8640
成为一名更好的程序员:如何阅读源代码
如何成为优秀的程序员 如何成为优秀的程序员
无论你是多么优秀的程序员,无论你从事何种职业,基础都是最重要的,任何高深的理论,任何看似复杂的任务,都是通过基础一点点解决的,一个人只有将基础打牢,他才能更上一层楼。
程序那些事儿
2023/03/07
2640
如何成为优秀的程序员
如何成为优秀的程序员
【技术指南】成为更优秀开发者的10条途径
我读过好多“成为更优秀开发者的方法”的文章,它们大部分似乎写于10年前。但大部分仍然很明智,因此我在这篇文章中提取出我认为的最好的10条途径。你可以随时看看。 我们开始吧。 读他人的代码 —— Scott Hanselmann 读他人的代码,并从中学习。你会适时得到提升,因为你容易学到其他开发者是如何处理问题的。结对编程是提升自我的最好途径。你从另外一个开发者那里读代码,实时地看到他/她的思维过程。反之亦然。你们可以挑战彼此的观点,共同进步。(推荐阅读:《阅读优秀代码是提高开发人员修为的一种捷径》) 找人读
程序员互动联盟
2018/03/13
7790
【技术指南】成为更优秀开发者的10条途径
凭什么2016年就会成为更好的自己?
回顾那些适(ban)可(tu)而(er)止(fei)的坚持,其实也留下了不少宝贵的财富,比如花了几个小时下载的电子书,每天收藏的好文章,讲座时拍下的PPT照片,随手记下的灵感和创作,认真做的几本读书笔记......这些有价值的资料散落在各处,没有被好好地整理和收纳。当这些知识不能被结构化和加深理解记忆,也就成了一堆沾满灰尘的废品,想找的时候都无处可寻。
科技云报道
2022/04/12
3570
凭什么2016年就会成为更好的自己?
学会如何学习 - 成为更好的终身学习者
酥鱼我从小学到大学毕业,当了十六年的学生,工作又选择了程序员这个发展日新月异、需要持续学习的行业。
scarsu
2020/10/22
8220
学会如何学习 - 成为更好的终身学习者
如何成为优秀的程序员?
经过了5年多的专职钉马掌生活后,我开始问自己一个问题:我要一直这样干下去吗?能不能干点儿别的?我的性格跟其他乡下那些叼着雪茄、喝着小酒的铁匠不一样,我从来没有戴过牛仔帽或骑过公牛。我渴望的是知道更多的
程序员互动联盟
2018/03/16
6540
如何成为优秀的程序员?
如何成为出色的程序员
年轻,高潜 这种程序员是每个公司都喜欢的员工,首先他们年轻,同时潜力巨大。 前几天看过一篇文章,说的是如果你35岁了,还有多少几率成为富豪。 数据统计的结果是:几乎为0。 面对这个数据,每个抱有侥幸心理的人恐怕都会哑口无言,这个几乎为零对于大部分没有好的背景,好的积蓄,好的人脉,好的父母,好的机遇的人来说恐怕就是0,所以在你35岁之前,甚至30岁之前,就需要不断努力,而不是在某个你时间点坐在那里仍对未来抱有侥幸。 年轻代表无限未来,高潜则是有较好的基础,聪明的头脑,学起新东西来更快,对新事物的变化更加敏
春哥大魔王
2018/04/17
6360
如何成为出色的程序员
成为更好的 Swift 开发者的 10 个 Tips(译)
你是否已经用 Swift 开发了几个月,现在,你想成为一个更好的 Swift 开发者?少年,你来对了地方,我这里有一本失传多年的武林秘籍传授于你。 不要在意代码的格式,我想尽我所能保持代码的简洁。以至于可以是你方便的拷贝到playground 来进行代码验证。
Swift社区
2021/11/26
2580
成为更好的 Swift 开发者的 10 个 Tips(译)
你是否已经用 Swift 开发了几个月,现在,你想成为一个更好的 Swift 开发者?少年,你来对了地方,我这里有一本失传多年的武林秘籍传授于你。 不要在意代码的格式,我想尽我所能保持代码的简洁。以至于可以是你方便的拷贝到playground 来进行代码验证。
网罗开发
2021/02/01
4010
程序员如何认识学历高点的程序员,以更好的提升自己?
写过几年代码,程序员的能力并不能直接和高学历挂钩,毕竟学历代表你曾经的学习能力和成果,并不能直接意味着做编程一定是高手,只能讲在有个好的基础之上成为高手的概率变高了,但不是绝对的。程序员提升自己能力,除了自身牢牢打好技术基础,还需要找个技术氛围好的工作,其次就是加入一些技术圈子结识一些技术大佬。
程序员互动联盟
2019/05/08
4940
程序员如何认识学历高点的程序员,以更好的提升自己?
10 个 Vue 开发技巧,助力成为更好的工程师!
在组件中使用 $route 会使之与其对应路由形成高度耦合,从而使组件只能在某些特定的 URL 上使用,限制了其灵活性。
逆锋起笔
2020/06/17
1.8K0
我们需要更多的程序员,而不是更好的工具
我们需要更多的程序员,而不是更好的工具 我和他的年纪差不多,并且有着相似的初始经验——在TRS-80、TI-99/4A、然后是Windows PC上用BASIC编程。所以,我觉得我有这个资格对他的文章
用户1289394
2018/02/27
8240
我们需要更多的程序员,而不是更好的工具
通往云端的多条途径
如今,关于云计算仍然存在着许多误解和神话。这种混淆使许多人难以理解其潜力,以及如何让云计算实现其业务目标。而人们必须超越这些基本概念,并开始探索云交付和采用趋势的不同复杂性,以便充分获得云计算所提供的好处。
静一
2018/08/20
6880
9个Vue开发技巧助力成为更好的工程师
原文链接:https://juejin.im/post/5e8a9b1ae51d45470720bdfa
歪马
2020/04/16
4.2K0
成为伟大程序员的 10 个要点
最近我在接受采访时被问到我关于成为一名伟大程序员的见解。这是一个有趣的问题,我认为我们都可以是伟大的程序员,无论我们的天赋如何,如果我们遵循一些规则的话——我相信——这应该是常识。实际上,这些规则并不只适用于编程领域,也适合任何专业。
哲洛不闹
2018/09/14
4160
成为伟大程序员的 10 个要点
成为优秀程序员的10个技巧
我最近在接受采访时被问如何成为优秀程序员。这是一个有趣的问题,我认为如果我们应该遵循一些准则 - 我相信 - 无论我们的天赋如何,我们都可以成为伟大的程序员。事实上,这些准则不但适用于程序员,而且适用于所有专业人士。
我就静静地看
2018/09/11
6120

相似问题

选择和取消选择UITableViewCells - Swift

34

如何防止自定义UITableViewCells在取消选择时闪烁白色?

20

单击时不取消选择多个选择

20

取消时使拖放不返回

12

使用querystring时不返回URI段。

14
添加站长 进交流群

领取专属 10元无门槛券

AI混元助手 在线答疑

扫码加入开发者社群
关注 腾讯云开发者公众号

洞察 腾讯核心技术

剖析业界实践案例

扫码关注腾讯云开发者公众号
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
查看详情【社区公告】 技术创作特训营有奖征文