使用 Z3 求解器寻找冰冻湖上的路径

使用 z3 求解器寻找冰冻湖上的路径

本文将详细介绍如何使用 Z3 定理证明器在 Python 中解决冰冻湖寻路问题。我们将详细讲解如何将问题转化为 Z3 可以理解的约束条件,并提供完整的代码示例,帮助读者理解如何使用 Z3 找到从起点到终点的安全路径。本文重点在于如何正确建模问题,以及如何使用 Z3 的 API 来表达约束和求解。

问题描述

给定一个由 1(安全)和 0(不安全)组成的矩阵,代表一个冰冻湖。目标是找到一条从起始位置到目标位置的安全路径,即路径上的所有单元格都必须是安全的(值为 1)。 只能在相邻的单元格之间移动(上、下、左、右)。

解决方案

解决此问题的关键在于正确地将问题建模为 Z3 可以理解的约束。我们不应该为矩阵中的每个单元格创建符号变量,而是应该为路径本身创建变量。这允许我们直接约束路径的有效性。

1. 定义路径变量

首先,我们需要定义表示路径的变量。 由于我们不知道路径的长度,我们可以假设最坏的情况是路径包含矩阵中的所有单元格。 因此,我们可以为路径中的每个可能的位置创建整数变量,表示其行和列坐标。

from z3 import *def find_path(matrix, start, end):    # Define the dimensions of the matrix    rows = len(matrix)    cols = len(matrix[0])    # symbolic look-up into the matrix:    def lookup(x, y):        val = 0        for r in range(rows):            for c in range(cols):                val = If(And(x == r, y == c), matrix[r][c], val)        return val    # Create a path, there are at most rows*cols elements    path = []    for r in range(rows):        for c in range(cols):            path.append([FreshInt(), FreshInt()])

在这里,path 是一个列表,其中每个元素都是包含两个 FreshInt() 变量的列表,分别代表行和列的索引。 FreshInt() 创建一个新的整数变量,其名称与之前创建的任何变量不同。 lookup 函数用于查找矩阵中给定坐标的值。

2. 添加约束

接下来,我们需要添加约束来确保路径有效。 这包括以下内容:

路径的第一个元素必须是起始位置。路径中的每个后续元素必须与前一个元素相邻。路径中的每个元素必须是安全的(值为 1)。路径必须最终到达目标位置。路径中的所有位置都是唯一的。

    s = Solver()    # assert that the very first element of the path is the start position:    s.add(path[0][0] == start[0])    s.add(path[0][1] == start[1])    # for each remaining path-element, make sure either we reached the end, or it's a valid move    prev = path[0]    done = False    for p in path[1:]:        valid1 = And(p[0] >= 0, p[0] = 0, p[1] < cols)  # Valid coords        valid2 = Or( And(p[0] == prev[0]-1, p[1] == prev[1])     #    Go up                   , And(p[0] == prev[0]+1, p[1] == prev[1])     # or Go down                   , And(p[0] == prev[0],   p[1] == prev[1]+1)   # or Go right                   , And(p[0] == prev[0],   p[1] == prev[1]-1))  # or Go left        valid3 = lookup(p[0], p[1]) == 1 # The cell is safe        # Either we're done, or all valid conditions must hold        s.add(Or(done, And(valid1, valid2, valid3)))        prev = p        # We're done if p is the end position:        done = Or(done, And(p[0] == end[0], p[1] == end[1]))    # Make sure the path is unique:    for i in range(len(path)):        for j in range(len(path)):            if j <= i:                continue            s.add(Or(path[i][0] != path[j][0], path[i][1] != path[j][1]))

代码解释:

s = Solver() 创建一个 Z3 求解器实例。s.add(path[0][0] == start[0]) 和 s.add(path[0][1] == start[1]) 约束路径的第一个元素为起始位置。循环遍历路径中的其余元素,并添加约束以确保每个元素都与前一个元素相邻,并且是安全的。valid1 确保坐标有效。valid2 确保移动是有效的(上、下、左、右)。valid3 确保单元格是安全的。s.add(Or(done, And(valid1, valid2, valid3))) 添加约束,要求要么已经到达终点,要么所有有效条件都成立。done = Or(done, And(p[0] == end[0], p[1] == end[1])) 检查当前位置是否是终点。最后的嵌套循环添加约束,确保路径中的所有位置都是唯一的。

3. 求解并提取路径

最后,我们使用 Z3 求解器来查找满足所有约束的路径。 如果找到这样的路径,我们将从模型中提取它。

    # Compute the path:    if s.check() == sat:        model = s.model()        walk = []        for p in path:            cur = [model[p[0]].as_long(), model[p[1]].as_long()]            walk.append(cur)            if (cur[0] == end[0] and cur[1] == end[1]):                break        return walk    else:        return None

代码解释:

s.check() == sat 检查求解器是否找到满足所有约束的解。model = s.model() 获取模型,该模型包含变量的赋值。循环遍历路径,并从模型中提取每个位置的坐标。如果当前位置是终点,则停止提取路径。返回提取的路径。

4. 完整代码和示例用法

from z3 import *def find_path(matrix, start, end):    # Define the dimensions of the matrix    rows = len(matrix)    cols = len(matrix[0])    # symbolic look-up into the matrix:    def lookup(x, y):        val = 0        for r in range(rows):            for c in range(cols):                val = If(And(x == r, y == c), matrix[r][c], val)        return val    # Create a path, there are at most rows*cols elements    path = []    for r in range(rows):        for c in range(cols):            path.append([FreshInt(), FreshInt()])    s = Solver()    # assert that the very first element of the path is the start position:    s.add(path[0][0] == start[0])    s.add(path[0][1] == start[1])    # for each remaining path-element, make sure either we reached the end, or it's a valid move    prev = path[0]    done = False    for p in path[1:]:        valid1 = And(p[0] >= 0, p[0] = 0, p[1] < cols)  # Valid coords        valid2 = Or( And(p[0] == prev[0]-1, p[1] == prev[1])     #    Go up                   , And(p[0] == prev[0]+1, p[1] == prev[1])     # or Go down                   , And(p[0] == prev[0],   p[1] == prev[1]+1)   # or Go right                   , And(p[0] == prev[0],   p[1] == prev[1]-1))  # or Go left        valid3 = lookup(p[0], p[1]) == 1 # The cell is safe        # Either we're done, or all valid conditions must hold        s.add(Or(done, And(valid1, valid2, valid3)))        prev = p        # We're done if p is the end position:        done = Or(done, And(p[0] == end[0], p[1] == end[1]))    # Make sure the path is unique:    for i in range(len(path)):        for j in range(len(path)):            if j <= i:                continue            s.add(Or(path[i][0] != path[j][0], path[i][1] != path[j][1]))    # Compute the path:    if s.check() == sat:        model = s.model()        walk = []        for p in path:            cur = [model[p[0]].as_long(), model[p[1]].as_long()]            walk.append(cur)            if (cur[0] == end[0] and cur[1] == end[1]):                break        return walk    else:        return None# Example usagematrix = [[1, 1, 1, 0],          [1, 0, 1, 0],          [1, 0, 1, 0],          [1, 0, 0, 0]]start = (3, 0)end = (2, 2)path = find_path(matrix, start, end)if path:    print("Valid path found:")    for cell in path:        print(f"({chr(ord('A') + cell[0])}{cell[1] + 1})")else:    print("No valid path found.")

注意事项和总结

建模是关键: 使用 Z3 解决问题的关键在于正确地将问题建模为约束。 在这个问题中,为路径创建变量而不是为矩阵中的每个单元格创建变量,可以更有效地表达约束。性能: Z3 的性能取决于问题的复杂性。 对于较大的矩阵,可能需要调整约束或使用更高级的技术来提高性能。唯一性约束: 确保路径中的所有位置都是唯一的,这对于避免循环至关重要。

通过使用 Z3 定理证明器,我们可以有效地找到冰冻湖上的安全路径。 这种方法可以推广到其他寻路问题,只需根据特定问题的要求调整约束即可。

以上就是使用 Z3 求解器寻找冰冻湖上的路径的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1369435.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月14日 09:39:52
下一篇 2025年12月14日 09:40:11

相关推荐

  • SymPy牛顿法求解根:符号变量与数值变量混淆的ValueError解析与修正

    本文深入探讨了在SymPy中使用牛顿法求解多项式根时常见的ValueError: First variable cannot be a number错误。该错误源于函数内部局部数值变量与全局符号变量的混淆,导致SymPy的求导操作接收到数值而非符号变量。教程将详细分析错误根源,并提供修正后的代码示例…

    好文分享 2025年12月14日
    000
  • Python Z3 应用:基于约束求解的网格安全路径查找

    本文详细介绍了如何利用 Python Z3 约束求解器解决网格路径查找问题。通过将路径建模为一系列符号变量,并施加移动规则、安全区域限制以及路径唯一性等约束,Z3 能够有效地找到从起点到终点的有效路径,避开障碍物。教程提供了完整的代码示例和详细解释,帮助读者理解 Z3 在此类问题中的应用。 引言:基…

    2025年12月14日
    000
  • 使用管道将大型 C 结构体直接传递给 Python

    本教程旨在指导开发者如何通过管道将 C 语言结构体数据直接传递到 Python 脚本中进行处理。我们将详细介绍如何在 C 代码中使用 fwrite 将结构体数据写入标准输出,然后在 Python 中使用 subprocess 模块捕获输出,并利用 ctypes 模块将字节流解析为 Python 中的…

    2025年12月14日
    000
  • C语言结构体数据通过管道高效传输至Python:ctypes与二进制流处理教程

    本教程详细介绍了如何通过标准输出管道将C语言结构体数组的二进制数据高效传输至Python,并利用c++types模块进行精确解析。文章从C端的数据准备、二进制写入,到Python端的进程调用、数据捕获与结构化解析,提供了完整的代码示例。特别强调了C语言中正确引入头文件(如stdio.h)的重要性,并…

    2025年12月14日
    000
  • 使用 Allure-Behave 在 Python 中生成测试报告

    Allure-Behave 是一个强大的工具,它允许您在 Python 的 Behave 测试框架中无缝集成 Allure 报告功能。通过简单的配置,您可以生成包含详细测试结果、步骤、附件和历史记录的报告,从而极大地提高测试结果的可视化和分析效率。 安装 Allure-Behave 首先,您需要安装…

    2025年12月14日
    000
  • Python Behave自动化测试集成Allure报告生成指南

    本教程详细介绍了如何在Python的Behave自动化测试框架中集成Allure报告,实现测试结果的可视化。通过配置behave.ini文件或使用命令行参数,利用allure-behave插件的格式化器,无需复杂的代码即可自动生成高质量的Allure测试报告,有效解决传统手动生成或文档缺失的问题,提…

    2025年12月14日
    000
  • 在Python中使用Allure-Behave生成测试报告

    本文详细介绍了如何在Python项目中使用Allure-Behave集成Behave测试框架,以自动化生成美观且功能丰富的Allure测试报告。通过配置Behave的格式化器(formatter),您可以轻松地将Allure报告的生成过程无缝嵌入到测试运行中,无需复杂的代码修改或手动调用报告生成函数…

    2025年12月14日
    000
  • 从包含特殊字符的字典中读取字符串值(Python)

    本文旨在解决在Python中从包含特殊字符(如斜杠)的字典中读取字符串值时可能遇到的问题。通过json.loads()方法,将JSON格式的字符串转换为Python字典对象,从而安全、便捷地访问和操作字典中的数据。本文提供详细的代码示例和解释,帮助开发者理解和应用此方法,避免常见的错误。 在Pyth…

    2025年12月14日
    000
  • Dropbox Python API:团队与个人文件访问策略详解

    本教程详细阐述了如何使用Dropbox Python API正确访问Dropbox Business团队环境下的个人和团队文件。针对不同需求,文章提供了两种核心策略:通过精简API权限直接访问特定用户文件,以及利用团队范围和 as_user 方法以管理员身份管理团队成员文件,并辅以代码示例和关键注意…

    2025年12月14日
    000
  • 优化Dropbox Python API访问:正确管理个人与团队文件权限

    本教程详细阐述如何使用Dropbox Python API有效访问个人和团队文件。核心在于根据所需访问级别(个人用户或团队管理)正确配置OAuth作用域。通过选择合适的权限,开发者可以避免常见的认证错误,实现对特定用户文件或整个团队资源的精确控制。 在使用dropbox python api与dro…

    2025年12月14日
    000
  • 高效拆分PDF并精确保留目录结构(PyMuPDF教程)

    本教程详细介绍了如何使用PyMuPDF库(fitz)高效地将大型PDF文档按指定页面范围拆分为多个独立文件,并确保每个拆分后的PDF都能正确地包含其对应的、且符合PyMuPDF规范的目录(Table of Contents, TOC)。文章深入探讨了PyMuPDF的TOC结构规则,提供了修正不规范T…

    2025年12月14日
    000
  • 分割PDF并动态生成目录(TOC)的PyMuPDF专业指南

    本教程详细介绍了如何使用PyMuPDF库高效地按页码范围分割PDF文件,并为每个分割后的文件动态生成并维护对应的目录(TOC)。文章重点阐述了PyMuPDF中TOC结构的严格规则,包括层级(level)的合法性检查与调整策略,特别是通过添加“虚拟”条目来确保TOC的正确性,从而实现分割PDF后TOC…

    2025年12月14日
    000
  • Ren’Py中对话打字音效与停顿同步的实现教程

    本教程旨在解决Ren’Py游戏中角色对话时打字音效与文本停顿不同步的问题。通过详细阐述type_sound函数的实现原理,并重点介绍如何利用Ren’Py内置的{w}标签来创建与音效完美匹配的定时停顿,确保打字音效在对话暂停时也能同步停止,从而提升游戏体验的沉浸感。 在ren&…

    2025年12月14日
    000
  • Ren’Py对话打字音效同步:解决停顿播放问题

    本教程详细介绍了如何在Ren’Py游戏中实现与角色对话同步的打字音效,并重点解决在对话停顿时音效持续播放的问题。通过利用Ren’Py的{w=X}标签,开发者可以确保打字音效在文本显示时播放,并在对话暂停时自动停止,从而提供更自然、沉浸式的用户体验。 实现Ren’P…

    2025年12月14日
    000
  • 使用Ren’Py制作打字音效教程

    本文将介绍如何在Ren’Py游戏中实现打字音效,使音效与对话文本的显示速度同步。我们将探讨如何使用Ren’Py提供的功能,结合代码示例,解决音效持续播放的问题,并提供一种有效的暂停对话方法,确保音效与文本的节奏保持一致,从而提升游戏的沉浸感。 实现打字音效 在Ren&#821…

    2025年12月14日
    000
  • Ren’Py 中实现打字音效的精确控制

    本文旨在解决 Ren’Py 游戏中实现打字音效时,音效播放与文本显示速度不匹配的问题。通过使用正确的暂停标签,可以确保音效在对话停顿时也能同步暂停,从而实现更自然、更具沉浸感的打字音效效果。 在 Ren’Py 游戏中,为对话添加打字音效可以显著提升游戏的沉浸感和真实感。然而,…

    2025年12月14日
    000
  • 基于PyMuPDF实现PDF按页码范围分割并保留目录

    本文档旨在提供一个使用PyMuPDF库,根据指定的页码范围分割PDF文件,并保留或重建分割后PDF文件的目录(Table of Contents, TOC)的详细教程。我们将深入探讨PyMuPDF库提供的get_toc()和set_toc()方法,并提供相应的代码示例,帮助读者理解如何正确处理和更新…

    2025年12月14日
    000
  • Python多进程在Windows下动态类型创建与传递的解决方案

    本文探讨了在Windows环境下使用Python多进程时,动态创建的类无法被子进程正确序列化和反序列化的问题。通过分析错误原因,本文提供了一种解决方案,确保动态创建的类可以在父进程中定义,并在子进程中安全地使用,同时避免重复创建带来的性能损耗。 在Windows下使用Python的multiproc…

    2025年12月14日
    000
  • Ren’Py 实现打字音效同步教程

    本文旨在提供一种在 Ren’Py 游戏中实现与文本同步的打字音效的解决方案。通过巧妙地利用 Ren’Py 的文本标签和自定义 Python 函数,可以精确控制音效的播放,使其与屏幕上文本的显示速度保持一致,从而增强游戏的沉浸感。文章将详细介绍实现步骤,并提供示例代码,帮助开发…

    2025年12月14日
    000
  • 提取 HTML 文本的 BeautifulSoup 教程

    本文旨在指导读者如何使用 Python 的 BeautifulSoup 库从 HTML 文档中提取纯文本数据。通过结合 requests 库获取网页内容,并利用 BeautifulSoup 的 get_text() 方法,可以有效地去除 HTML 标签,获取干净、可用的文本信息,从而方便进行数据分析…

    2025年12月14日
    000

发表回复

登录后才能评论
关注微信