写点什么

Google SoC 系列:使用 Ruby 实现约束规划

  • 2007-06-26
  • 本文字数:2383 字

    阅读完需:约 8 分钟

Gecode/R 是一个得到 Google SoC 资助的 Ruby 项目,目的是使得 Ruby 语言的开发者同样可以使用运筹学中的约束规划(Constraint Programming)方法,项目的发起人 Andreas Launila 对于约束规划这样解释道:

约束规划是一种声明式的规划范式(Declarative Programming Paradigm),开发者描述需要何种解决方案,而不是告诉程序如何进行具体的计算。当使用约束规划时,开发者可以试图为问题建立模型,并将模型集中提交给问题处理程序。问题处理程序通过检索问题域中的所有可行解,并利用模型中的约束条件来去除这些解中的部分,而不需要真正去访问到这部分的可能解。 一个常见的例子就是数独游戏(Soduko),通过约束规划解决数独问题的过程是:为问题处理程序填写约束规则(比方说每一行的数字必须是各不相同的),然后寻找令所有约束条件同时满足的解决方案。

当问及现实世界中是否有使用约束规划解决问题的例子时,Andreas 解释道:

约束规划是用于处理 NP 难解组合问题的常用方法,在这种情况下,一般只存在有限的选择,但需要根据特定条件进行搜索。现实世界中的例子包括时序安排,周期分配以及人员工作时间设置等问题。

Gecode/R 实际上是建立在 C++ 约束规划函数库 Gecode基础之上的 Ruby 语言封装,关于为何使用 Ruby 语言来实现 Gecode 的功能,Andreas 这样解释道:

对我来说,约束规划是在我常用工具箱中一个非常有用的工具。当使用约束规划来处理恰当的问题类型时,会为我们节省大量的时间和工作。由于可以进行快速编码实现,当更好的算法存在并且运行效率并非是首要考虑的因素(或者性能上的差别比起在研究和实现算法上需要花的额外时间是可以忽略不计)时,约束规划可以用来很快的解决问题。

一个有趣的话题是关于如何使用 Ruby 语言来定义约束规划:

我的目的是为了创建一个前端,而不仅仅是一个绑定。我称之为类库而不仅是一种 DSL(Domain Specific Language,领域特定语言),尽管两者之间的边界并非有十分严格的界定。因为项目即将开始,至今还没有定义精确的语法规则。下面简单的代码段给出项目的大致指导方向,但这不一定是项目最终使用的语法规则。代码段示例解决了“send+more=money”这个经典的问题。 通过这样的途径,比较方法和算术方法被用于表示线性约束,并且通过对数组进行拓展来表达特殊约束来用于分支选择。变量在创建时需要记录,以便于跟踪他们。其中一个不足之处是符号“!=”没有被定义为一个方法,所以使用不等式时的语法与其他语言并不相同。

下面是程序代码编写的样例:

# 经典的 send+more=money 问题。<br></br>class SendMoreMoneyProblem < Gecode::Space<br></br> def initialize<br></br> # 设置变量,在 0..9 之间的 8 个字母.<br></br> s,e,n,d,m,o,r,y = letters = IntVar.array(8, 0..9)<br></br> # 设置约束条件.<br></br> constrain equation_row(s, e, n, d) +<br></br> equation_row(m, o, r, e) ==<br></br> equation_row(m, o, n, e, y)<br></br> constrain s.not_equal(0) # 并不是写成"s != 0" 因为我们无法重定义符号 != .<br></br> constrain m.not_equal(0)<br></br> letters.all_distinct<br></br> # 选择搜索解决方案时要使用的启发式分支.<br></br> letters.branch_using(:variable => :min_size, :value => :min)<br></br> end<p>private</p><br></br> # 使用线性方程有更好的方法。取出变量中的一个数字,<br></br> # 并且计算线性组合就像变量是基于 10 个字符的阿拉伯数字。<br></br> # 例如:x,y,z 成为 100*x + 10*y + z .<br></br> def equation_row(*variables)<br></br> variables.inject(0){ |result, variable| variable + 10*result }<br></br> end<br></br>end<br></br># 打印出问题的第一种解决方案 <br></br>p SendMoreMoneyProblem.new.solutions(:first)如果想了解更多细节,可以查看维基百科上面“send+more=money”词条相关的文章

当使用 Ruby 语言定义业务逻辑或者数据时,常有可能使用内部 DSL来编写更加简练并且可读性更强的代码。Andreas 关于代码可能的样式这样解释道:

另一个使得代码看起来更像 DSL 的方法是在编码中跳过类声明之类的。这个方法在快速处理单一样例的问题时更加出色,但是会妨碍与其它 Ruby 代码协同来使用约束规划。

下面是使用 DSL 方法的代码样例:

find_first_solution_to define_problem do |p|<br></br> # 设置变量,在 0..9 之间的 8 个字母。<br></br> s,e,n,d,m,o,r,y = letters = p.create_int_vars(8, 0..9)<br></br> # 设置约束。<br></br> p.add 1000*s + 100*e + 10*n + d +<br></br> 1000*m + 100*o + 10*r + e ==<br></br> 10000*m + 1000*o + 100*n + 10*e + y<br></br> p.add s.not_equal(0)<br></br> p.add m.not_equal(0)<br></br> p.add all_distinct(letters)<br></br> # 选择启发式分支。<br></br> p.branch_on letters, :variable => :min_size, :value => :min<br></br>end由于 Gecode/R 是在本地类库基础之上的绑定实现,这样就存在一个潜在的瓶颈问题。Andreas 非常自信的认为这不会是一个影响项目发展的问题:

在 Gecode 中已经实现了通用约束的传播机制,所以这些约束的实际传播过程已在此处完成。如果用户设置自定义的传播机制,则需要在 C 语言和 Ruby 语言之间进行来回切换,目前我还不清楚这样是否存在性能问题。但 Gecod 的设计可以使其能很好地与外界进行交互,并且可以很好的与 Java 语言协同工作,所以如果有人说 Gecode/R 在本地类库上实现绑定将会成为瓶颈,我肯定会对此表示惊讶。

Gecode/R 是建立在 RubyForge 上的开放源代码项目,这同时也是吸引对此感兴趣开发者加入的优秀站点。约束规范相关的API 接口和语法的可以在链接中的Wiki 页面深入了解。Andreas 同时也维护着建立在O’Reilly Ruby 站点上的Gecode/R 项目相关博客

查看英文原文: Google SoC Series: Constraint programming with Ruby

2007-06-26 19:301316
用户头像

发布了 74 篇内容, 共 13.9 次阅读, 收获喜欢 3 次。

关注

评论

发布
暂无评论
发现更多内容

百誉控股完成对Electra New Energy Vehicle Inc控股权的合并

财见

和鲸科技荣耀入选2024 H1 「中国最具价值 AGI 创新机构 TOP 50」

ModelWhale

AI 云原生 AGI

和鲸101计划夏令营火热进行中!北中医助阵医学数据探索,快来学习

ModelWhale

AI 数据科学 实践 Workshop 夏令营

JeecgBoot 中如何对敏感信息进行脱敏处理?

JEECG低代码

数据加密 数据脱敏 JeecgBoot jeecg

Docker镜像源失效,教你将源换成国内

JEECG低代码

Docker docker image

伙伴活动|AI硬件大潮来袭,深圳的创客们在哪里?

声网

开源可视化报表,JimuReport 积木报表 v1.7.8 版本发布

JEECG低代码

积木报表 jimureport 报表工具

Docker Compose 一键快速部署 RocketMQ

JEECG低代码

Docker RocketMQ

JeecgBoot 代码生成器使用入门

JEECG低代码

代码生成器 jeecg-boot JeecgBoot jeecg

TikTok直播卡顿、封号频发?搭建TikTok直播网络解决!

Ogcloud

tiktok直播 tiktok直播专线 tiktok直播网络 tiktok直播带货 TikTok跨境直播

数据分析的好帮手,小浣熊办公助手!

百里丶落云

AI 办公小浣熊 代码小浣熊 小浣熊家族 小浣熊

互联网不行了?程序员不吃香了?机会在哪里?

伤感汤姆布利柏

【信创国产化】JeecgBoot低代码适配达梦和人大金仓

JEECG低代码

信创 JeecgBoot 国产信创数据库

数据中台真的适合你的企业吗?

RestCloud

数据中台 BI 数仓 ETL

移动图形工作站那家好用?

青椒云云电脑

图形工作站 移动图形工作站

Sam Altman 新公司将打造 AI 健康教练;全新大模型架构 TTT 超越 Transformer 丨 RTE 开发者日报

声网

东南亚TikTok直播专线

Ogcloud

tiktok直播 tiktok直播专线 tiktok直播网络 tiktok直播带货 TikTok跨境直播

空窗期太长?这么说就对了!

王磊

Java

linux进程管理

不在线第一只蜗牛

Linux 运维

JeecgBoot 集成宝兰德 AppServer 部署方案

JEECG低代码

信创 低代码平台 JeecgBoot 信创国产化

乘用车1-5月销量同比增长7%,火山引擎数据飞轮能帮车企盘活下半年增长吗?

新消费日报

10个JavaScript One-Liners让初学者看起来很专业

OpenTiny社区

JavaScript 开源 前端 OpenTiny

【信创】JeecgBoot集成东方通TongRDS

JEECG低代码

信创 低代码平台 jeecg-boot JeecgBoot 信创国产化

Altair 签署协议收购 Metrics,扩大其在 EDA 行业的影响力

Altair RapidMiner

人工智能 仿真 altair

JeecgBoot v3.7.0 all 版本发布

JEECG低代码

低代码 低代码开发平台 JeecgBoot jeecg

图片链接一网打尽:利用JD商品详情API获取商品高清图

技术冰糖葫芦

API 安全 API 文档 API 开发 API 协议

百度智能云将大模型引入网络故障定位的智能运维实践

Baidu AICLOUD

智能运维 网络运维

Google SoC系列:使用Ruby实现约束规划_Ruby_Werner Schuster_InfoQ精选文章