【回溯算法】【Python实现】符号三角形问题

news2025/1/10 11:39:23

文章目录

    • @[toc]
      • 问题描述
      • 回溯法
      • 时间复杂性
      • `Python`实现

问题描述

  • 下图是由 14 14 14个“ + + +”和 14 14 14个“ − - ”组成的符号三角形, 2 2 2个同号下面都是” + + +“, 2 2 2个异号下面都是“ − -

在这里插入图片描述

  • 在一般情况下,符号三角形的第一行有 n n n个符号,符号三角形问题要求对于给定的 n n n,计算有多少个不同的符号三角形,使其所含的“ + + +”和“ − - ”的个数相同

回溯法

  • 对于符号三角形问题,用 n n n元组 x [ 1 : n ] x[1 : n] x[1:n]表示符号三角形的第一行的 n n n个符号,由于 x [ i ] x[i] x[i]是二值的,所以在用回溯法解符号三角形问题时,可以用一棵完全二叉树来表示其解空间
  • 在符号三角形的第一行的前 i i i个符号 x [ 1 : i ] x[1 : i] x[1:i]确定后,就确定了一个由 i ( i + 1 ) / 2 i (i + 1) / 2 i(i+1)/2个符号组成的符号三角形,下一步确定 x [ i + 1 ] x[i + 1] x[i+1]的值后,只要在前面已确定的符号三角形的右边加一条边,就可以扩展为 x [ 1 : i + 1 ] x[1 : i + 1] x[1:i+1]相应的符号三角形
  • 最终由 x [ 1 : n ] x[1 : n] x[1:n]所确定的符号三角形中包含的“ + + +”个数与“ − - ”个数同为 n ( n + 1 ) / 4 n (n + 1) / 4 n(n+1)/4,因此在回溯搜索过程中,可用当前符号三角形所包含的“ + + +”个数与” − - “个数均不超过 n ( n + 1 ) / 4 n (n + 1) / 4 n(n+1)/4作为可行性约束,用于剪去不满足约束的子树
  • i = n i = n i=n时,算法搜索至叶节点,得到一个新的“ + + +”个数与“ − - ”个数相同的符号三角形,当前已找到符号三角形数 s u m sum sum 1 1 1
  • i < n i < n i<n时,当前扩展结点 Z Z Z是解空间中的内部结点,对当前扩展结点 Z Z Z的每个儿子结点,计算其相应的符号三角形中“ + + +”个数与“ − - ”个数,并以深度优先的方式递归地对可行子树进行搜索,或剪去不可行子树
  • 对于给定的 n n n,当 n ( n + 1 ) / 2 n (n + 1) / 2 n(n+1)/2为奇数时,显然不存在所包含的“ + + +”个数与“ − - ”个数相同的符号三角形,此时可以通过简单的判断加以处理

时间复杂性

  • 更新符号三角形矩阵需要 O ( n ) O(n) O(n)时间,在最坏情况下,有 O ( 2 n ) O(2^{n}) O(2n)个结点需要更新符号三角形矩阵
  • 所以解符号三角形问题的回溯算法所需的计算时间为 O ( n 2 n ) O(n 2^{n}) O(n2n)

Python实现

def symbol_triangle(n):
    if (n * (n + 1) // 2) % 2:
        return 0

    half = n * (n + 1) // 4

    count = 0  # 记录符合条件的符号三角形数量

    # 初始化符号三角形矩阵
    path = [[''] * n for _ in range(n)]

    def backtrack(row, col, path, plus_count, minus_count):
        nonlocal count

        # 边界条件: 当列数等于 n 时, 表示已经生成了符号三角形的一种排列
        if col == n:
            if plus_count == minus_count:
                count += 1

                print(path)

            return

        # 尝试当前位置为 +
        path[row][col] = '+'
        plus_count += 1

        # 更新符号三角形矩阵
        cur_col = col
        for i in range(1, cur_col + 1):
            if path[i - 1][cur_col - 1] == path[i - 1][cur_col]:
                path[i][cur_col - 1] = '+'
                plus_count += 1
            else:
                path[i][cur_col - 1] = '-'
                minus_count += 1

            cur_col -= 1

        # 检查是否满足条件, 继续生成下一行的符号
        if plus_count <= half and minus_count <= half:
            backtrack(row, col + 1, path, plus_count, minus_count)

        # 恢复回溯之前状态
        cur_col = col
        for i in range(cur_col + 1):
            if path[i][cur_col] == '+':
                path[i][cur_col] = ''
                plus_count -= 1
            else:
                path[i][cur_col] = ''
                minus_count -= 1

            cur_col -= 1

        # 尝试当前位置为 -
        path[row][col] = '-'
        minus_count += 1

        # 更新符号三角形矩阵
        cur_col = col
        for i in range(1, cur_col + 1):
            if path[i - 1][cur_col - 1] == path[i - 1][cur_col]:
                path[i][cur_col - 1] = '+'
                plus_count += 1
            else:
                path[i][cur_col - 1] = '-'
                minus_count += 1

            cur_col -= 1

        # 检查是否满足条件, 继续生成下一行的符号
        if plus_count <= half and minus_count <= half:
            backtrack(row, col + 1, path, plus_count, minus_count)

    backtrack(0, 0, path, 0, 0)

    return count


n = 4

print('满足条件的符号三角形如下:')

count = symbol_triangle(n)

print(f'符号三角形数量: {count}')
满足条件的符号三角形如下:
[['+', '+', '-', '+'], ['+', '-', '-', ''], ['-', '+', '', ''], ['-', '', '', '']]
[['+', '+', '-', '-'], ['+', '-', '+', ''], ['-', '-', '', ''], ['+', '', '', '']]
[['+', '-', '+', '+'], ['-', '-', '+', ''], ['+', '-', '', ''], ['-', '', '', '']]
[['+', '-', '+', '-'], ['-', '-', '-', ''], ['+', '+', '', ''], ['+', '', '', '']]
[['-', '+', '-', '+'], ['-', '-', '-', ''], ['+', '+', '', ''], ['+', '', '', '']]
[['-', '-', '+', '+'], ['+', '-', '+', ''], ['-', '-', '', ''], ['+', '', '', '']]
符号三角形数量: 6

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1660310.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

MySQL慢查询SQL优化

一、慢查询日志 描述&#xff1a;通过慢查询日志等定位那些执行效率较低的SQL语句 查看 # 慢查询是否开启 show variables like slow_query_log%; # 慢查询超时时间 show variables like long_query_time%;执行SQL 开启慢查询日志 set global slow_query_log ON;设置慢查…

python数据分析——数据的选择和运算

数据的选择和运算 前言一、数据选择NumPy的数据选择一维数组元素提取示例 多维数组行列选择、区域选择示例 花式索引与布尔值索引布尔索引示例一示例二 花式索引示例一示例二 Pandas数据选择Series数据获取DataFrame数据获取列索引取值示例一示例二 取行方式示例loc() 方法示例…

一篇文章,系统性聊聊Java注解

你好&#xff01; 这类系统性聊聊***知识点的文章&#xff0c;是希望给大家带来对某个技术的全貌认识&#xff0c;如果大家喜欢&#xff0c;后续可以陆续更新此系列 下面&#xff0c;开始今天的分享 在之前&#xff0c;我们已经分享过注解相关的三个面试题&#xff0c; 今天的…

【原创教程】步进MC_HOME回原点模式

我们所用软件:西门子TIA Portal V16编程软件 我们所用硬件:S7-1200系列:CPU1212C;雷赛科技DM542驱动器;西门子TP900 comfort触摸屏 我们的硬件接线 MC_HOME的模式: 一般情况下,西门子PLC的运动控制在使能绝对位置定位之前必须执行“回原点”或是“寻找参考点”。 Pos…

多层感知机(MLP)示例

通过5个条件判定一件事情是否会发生&#xff0c;5个条件对这件事情是否发生的影响力不同&#xff0c;计算每个条件对这件事情发生的影响力多大&#xff0c;写一个多层感知机模型pytorch程序,最后打印5个条件分别的影响力。 一 在多层感知机&#xff08;MLP&#xff09;模型中&…

RabbitMQ部署指南.md

RabbitMQ部署指南 1.单机部署 我们在Centos7虚拟机中使用Docker来安装。 1.1.下载镜像 方式一&#xff1a;在线拉取 docker pull rabbitmq:3.8-management方式二&#xff1a;从本地加载 在课前资料已经提供了镜像包&#xff1a; 上传到虚拟机中后&#xff0c;使用命令加载…

超详细的胎教级Stable Diffusion使用教程(一)

这套课程分为五节课&#xff0c;会系统性的介绍sd的全部功能和实操案例&#xff0c;让你打下坚实牢靠的基础 一、为什么要学Stable Diffusion&#xff0c;它究竟有多强大&#xff1f; 二、三分钟教你装好Stable Diffusion 三、小白快速上手Stable Diffusion 四、Stable dif…

【网络安全入门】新手如何参加护网行动?一篇带你零基础入门到精通

前言 “没有网络安全就没有国家安全”。 当前&#xff0c;网络安全已被提升到国家战略的高度&#xff0c;成为影响国家安全、社会稳定至关重要的因素之一。 一、网络安全行业特点 行业发展空间大&#xff0c;岗位非常多 网络安全行业产业以来&#xff0c;随即新增加了几十个…

Django 管理员登录安全 OTP双因素认证

目前安全双因素 最基本的&#xff0c;django管理员 默认直接登录的。 本项目环境:Django 2.0.13django-otp 0.9.3 1 安装pip3 install django-otp0.9.3 2 配置文件 vim api_statistics/settings.py INSTALLED_APPS里增加django_otp,django_otp.plugins.otp_totp,MIDDLEWARE…

出海企业必备神器:海外云手机的秘密你了解多少?

在全球化日益加速的今天&#xff0c;出海企业面临着前所未有的机遇与挑战。如何才能以一个较低的成本打入海外市场&#xff0c;实现业务的高速增长&#xff1f;答案就是——海外云手机&#xff01; 一、为何海外云手机成为出海企业的必备神器&#xff1f; 突破地域限制&#xf…

videosapi、强大的微信管理api

接口地址&#xff1a;videosapi.com 同步个人收藏夹 {"appId": "{{appid}}","syncKey": "" } 出参 {"ret": 200,"msg": "操作成功","data": {"syncKey": "CAESCAgBEJyi9e4…

Python扫描发送邮件

一、具体要求 编写 python 代码&#xff0c;扫描指定的目录下的所有文件&#xff0c;将这些扫描的文本内容邮 件发送到指定邮箱( 自己的 qq 邮箱 ) 示例如下&#xff1a; 使用QQ邮箱SMTP服务器需要获取QQ邮箱授权码&#xff0c;获取方式可参考以下链接&#xff1a; 电子邮件 二…

【千帆平台】使用AppBuilder零代码创建应用,然后通过OpenAPI方式调用应用

欢迎来到《小5讲堂》 这是《千帆平台》系列文章&#xff0c;每篇文章将以博主理解的角度展开讲解。 温馨提示&#xff1a;博主能力有限&#xff0c;理解水平有限&#xff0c;若有不对之处望指正&#xff01; 目录 前言创建API密钥调用文档调用说明API服务域名通信协议字符编码公…

企业短信群发被拦截的原因

企业短信群发作为一种高效的营销手段&#xff0c;被越来越多的企业所采用。也经常会面临被拦截的问题&#xff0c;这给企业带来了沉重的损失。本文将探讨企业群发被拦截的原因&#xff0c;并寻找解决办法。 频率过高 频率过高是导致企业群发被拦截的常见原因之一。当企业的频率…

淘宝扭蛋机小程序开发:探索未知,扭出惊喜

一、引领潮流&#xff0c;创新体验 在淘宝这片购物海洋中&#xff0c;我们总是寻求为您带来不一样的惊喜和乐趣。今天&#xff0c;我们正式推出淘宝扭蛋机小程序&#xff0c;将传统扭蛋的趣味与电商购物的便捷完美结合&#xff0c;为您带来前所未有的购物新体验。 二、扭蛋机…

摩菲Murphy显示器显示表 总线编程器维修PV780B

Murphy仪器维修包括&#xff1a;摩菲数字显示器&#xff1b;摩菲监视仪表&#xff1b;摩菲CAN总线控制器等维修 维修故障包括&#xff1a;黑屏、指示灯无显示&#xff0c;触摸屏上电无反应&#xff0c; 上电蓝屏、白屏&#xff0c;通电几分钟后屏幕变为蓝屏&#xff0c;主板故…

vue2后台管理项目

一:项目准备 1)拉取模板代码 远程仓库复制到本地仓库. 2)安装后的项目 路径 code 文件夹 会打开vscode的文件夹. 3)安装vetur和eslint插件可以保存时自动修改不规范的地方. 4)App内有一级路由,路由组件导入如果是layout架子,会导入的是文件夹下的index.js没有则导入index.v…

副业兼职没那么难,视频号带货,1天稳定500,适合新手操作

向大家推荐一个项目&#xff1a;视频号书单号带货玩法。我已经实践了一段时间&#xff0c;并成功售出了1200多单&#xff0c;赚取了2万多元。这个项目表现相当出色&#xff0c;强烈推荐给大家&#xff01; 周周近财&#xff1a;让网络小白少花冤枉钱&#xff0c;赚取第一桶金 …

使用Three.js开发一个3D案例Demo

使用Three.js开发一个3D案例 最近在找工作&#xff0c;发现好多招聘要求都需要会Three.js&#xff0c;以前接触比较多的是2D开发&#xff0c;也就是平面开发&#xff0c;用到的做多的技术就是d3.js&#xff0c;现在3D开发已经成为了大势所趋&#xff0c;所以就学习下Three.js。…

我国结肠镜市场规模逐渐扩大 国产化率有所增长

我国结肠镜市场规模逐渐扩大 国产化率有所增长 结肠镜是专用于检查结肠内部的医疗器械&#xff0c;属于内窥镜的细分品类之一。结肠镜具有特殊的设计&#xff0c;可以检查结肠壁的状态、寻找肠道中的异常病变和息肉等&#xff0c;在筛查结肠癌、检测炎症性肠病等方面发挥着十分…