草庐IT

ruby-on-rails - 在时间表中对无限期重复的任务进行建模(类似日历的 Rails 应用程序)

coder 2025-05-24 原文

这是一个很大的绊脚石。警告:以下不是问题,而是对我的想法的解释。我的问题是——你有更好的方法吗?是否有一些我不熟悉的常用技术?看起来这是一个微不足道的问题。

所以你有任务模型。您可以创建任务、完成任务、销毁任务。然后你有经常性的任务。它就像常规任务一样,但它附加了一个重复规则。但是,任务可以无限期地重复——您可以在计划中提前一年,您应该会看到任务出现。

因此,当用户创建一个循环任务时,您不会想在未来百年内构建数千个任务,并将它们保存到数据库中,对吧?所以我开始思考 — 如何创建它们?

一种方法是在您查看日程安排时创建它们。因此,当用户提前一个月移动时,将创建任何重复性任务。当然,这意味着您不能再简单地使用任务的数据库记录。您曾经对任务执行的每个 SELECT 操作都必须在特定日期范围的上下文中,以便触发该日期范围内的重复任务以持续存在。这是一种维护和性能负担,但可行。

好的,但是原始任务呢?每个循环任务都与创建它的循环规则相关联,并且每个循环规则都需要知道启动循环的原始任务。后者很重要,因为您需要在用户浏览他们的日程安排时将原始任务克隆到新日期。我想也是可行的。

但是如果更新了原始任务会发生什么?这意味着现在当我们浏览日程表时,我们将创建从修改后的任务中克隆出来的重复性任务。这是不可取的。所有隐式持久化的重复任务都应该以添加重复时原始任务的方式显示。因此,我们需要单独存储原始任务的副本,并从中克隆,以便重复执行。

但是,当用户浏览日程中的任务时,我们如何知道在特定点是否需要创建新的重复任务?我们询问重复规则:“嘿,我应该坚持这一天的任务吗?”它说是或否。如果这一天已经有此重复的任务,我们不会创建一个。一切都很好,除了用户还可以简单地删除已自动保留的重复任务之一。在这种情况下,按照我们的逻辑,系统将重新创建已删除的任务。不好。所以这意味着我们需要继续存储任务,但将其标记为本次重复的已删除任务。嗯。

正如我在开头所说的,我想知道是否有人解决了这个问题并且可以在这里提供架构建议。一定要这么乱吗?我还缺少什么更优雅的东西吗?

更新:由于这个问题很难完美回答,我会赞成对设计/架构最有帮助的见解,它对这类问题有最好的帮助/权衡比。它不必包含所有细节。

最佳答案

我知道这是一个老问题,但我刚刚开始为我自己的应用研究这个问题,我发现 Martin Fowler 的这篇论文很有启发性:Recurring Events for Calendars

我的主要收获是使用他所谓的“时间表达式”来确定预订是否在特定日期范围内,而不是尝试将无限数量的事件(或您的情况下的任务)插入数据库。

实际上,对于您的用例,这可能意味着您使用名为 schedule 的“时间表达式”属性存储任务。 ice_cube recurrence gem能够将自身序列化为事件记录属性 like so :

class Task < ActiveRecord::Base
  include IceCube
  serialize :schedule, Hash

  def schedule=(new_schedule)
    write_attribute(:schedule, new_schedule.to_hash)
  end

  def schedule
    Schedule.from_hash(read_attribute(:schedule))
  end
end

Ice cube 看起来非常灵活,甚至允许您指定重复规则的异常(exception)情况。 (假设您只想删除任务的一次出现,而不是全部。)

问题是您无法真正查询数据库以查找属于特定日期范围内的任务,因为您只存储了创建任务的规则,而不是任务本身。对于我的情况,我正在考虑添加一个属性,如“next_recurrence_date”,它将用于进行一些基本的排序/过滤。您甚至可以使用它来将任务放入队列中,以便在下一个重复日期完成某件事。 (比如检查该日期是否已过,然后重新生成它。您甚至可以在下一个重复日期过后存储任务的“存档”版本。)

这解决了“如果任务更新了怎么办”的问题,因为任务在过去之前不会持久化。

无论如何,我希望这对试图为自己的应用考虑这一点的人有所帮助。

关于ruby-on-rails - 在时间表中对无限期重复的任务进行建模(类似日历的 Rails 应用程序),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5346839/

有关ruby-on-rails - 在时间表中对无限期重复的任务进行建模(类似日历的 Rails 应用程序)的更多相关文章

  1. ruby-on-rails - 使用 Ruby on Rails 进行自动化测试 - 最佳实践 - 2

    很好奇,就使用ruby​​onrails自动化单元测试而言,你们正在做什么?您是否创建了一个脚本来在cron中运行rake作业并将结果邮寄给您?git中的预提交Hook?只是手动调用?我完全理解测试,但想知道在错误发生之前捕获错误的最佳实践是什么。让我们理所当然地认为测试本身是完美无缺的,并且可以正常工作。下一步是什么以确保他们在正确的时间将可能有害的结果传达给您? 最佳答案 不确定您到底想听什么,但是有几个级别的自动代码库控制:在处理某项功能时,您可以使用类似autotest的内容获得关于哪些有效,哪些无效的即时反馈。要确保您的提

  2. ruby-on-rails - rails : keeping DRY with ActiveRecord models that share similar complex attributes - 2

    这似乎应该有一个直截了当的答案,但在Google上花了很多时间,所以我找不到它。这可能是缺少正确关键字的情况。在我的RoR应用程序中,我有几个模型共享一种特定类型的字符串属性,该属性具有特殊验证和其他功能。我能想到的最接近的类似示例是表示URL的字符串。这会导致模型中出现大量重复(甚至单元测试中会出现更多重复),但我不确定如何让它更DRY。我能想到几个可能的方向...按照“validates_url_format_of”插件,但这只会让验证干给这个特殊的字符串它自己的模型,但这看起来很像重溶液为这个特殊的字符串创建一个ruby​​类,但是我如何得到ActiveRecord关联这个类模型

  3. ruby-on-rails - 在 Rails 中将文件大小字符串转换为等效千字节 - 2

    我的目标是转换表单输入,例如“100兆字节”或“1GB”,并将其转换为我可以存储在数据库中的文件大小(以千字节为单位)。目前,我有这个:defquota_convert@regex=/([0-9]+)(.*)s/@sizes=%w{kilobytemegabytegigabyte}m=self.quota.match(@regex)if@sizes.include?m[2]eval("self.quota=#{m[1]}.#{m[2]}")endend这有效,但前提是输入是倍数(“gigabytes”,而不是“gigabyte”)并且由于使用了eval看起来疯狂不安全。所以,功能正常,

  4. ruby-on-rails - Ruby net/ldap 模块中的内存泄漏 - 2

    作为我的Rails应用程序的一部分,我编写了一个小导入程序,它从我们的LDAP系统中吸取数据并将其塞入一个用户表中。不幸的是,与LDAP相关的代码在遍历我们的32K用户时泄漏了大量内存,我一直无法弄清楚如何解决这个问题。这个问题似乎在某种程度上与LDAP库有关,因为当我删除对LDAP内容的调用时,内存使用情况会很好地稳定下来。此外,不断增加的对象是Net::BER::BerIdentifiedString和Net::BER::BerIdentifiedArray,它们都是LDAP库的一部分。当我运行导入时,内存使用量最终达到超过1GB的峰值。如果问题存在,我需要找到一些方法来更正我的代

  5. ruby-on-rails - unicode 字符串的长度 - 2

    在我的Rails(2.3,Ruby1.8.7)应用程序中,我需要将字符串截断到一定长度。该字符串是unicode,在控制台中运行测试时,例如'א'.length,我意识到返回了双倍长度。我想要一个与编码无关的长度,以便对unicode字符串或latin1编码字符串进行相同的截断。我已经了解了Ruby的大部分unicode资料,但仍然有些一头雾水。应该如何解决这个问题? 最佳答案 Rails有一个返回多字节字符的mb_chars方法。试试unicode_string.mb_chars.slice(0,50)

  6. ruby-on-rails - Ruby on Rails 迁移,将表更改为 MyISAM - 2

    如何正确创建Rails迁移,以便将表更改为MySQL中的MyISAM?目前是InnoDB。运行原始执行语句会更改表,但它不会更新db/schema.rb,因此当在测试环境中重新创建表时,它会返回到InnoDB并且我的全文搜索失败。我如何着手更改/添加迁移,以便将现有表修改为MyISAM并更新schema.rb,以便我的数据库和相应的测试数据库得到相应更新? 最佳答案 我没有找到执行此操作的好方法。您可以像有人建议的那样更改您的schema.rb,然后运行:rakedb:schema:load,但是,这将覆盖您的数据。我的做法是(假设

  7. ruby - 在 Ruby 程序执行时阻止 Windows 7 PC 进入休眠状态 - 2

    我需要在客户计算机上运行Ruby应用程序。通常需要几天才能完成(复制大备份文件)。问题是如果启用sleep,它会中断应用程序。否则,计算机将持续运行数周,直到我下次访问为止。有什么方法可以防止执行期间休眠并让Windows在执行后休眠吗?欢迎任何疯狂的想法;-) 最佳答案 Here建议使用SetThreadExecutionStateWinAPI函数,使应用程序能够通知系统它正在使用中,从而防止系统在应用程序运行时进入休眠状态或关闭显示。像这样的东西:require'Win32API'ES_AWAYMODE_REQUIRED=0x0

  8. ruby-on-rails - rails : "missing partial" when calling 'render' in RSpec test - 2

    我正在尝试测试是否存在表单。我是Rails新手。我的new.html.erb_spec.rb文件的内容是:require'spec_helper'describe"messages/new.html.erb"doit"shouldrendertheform"dorender'/messages/new.html.erb'reponse.shouldhave_form_putting_to(@message)with_submit_buttonendendView本身,new.html.erb,有代码:当我运行rspec时,它失败了:1)messages/new.html.erbshou

  9. ruby-on-rails - 由于 "wkhtmltopdf",PDFKIT 显然无法正常工作 - 2

    我在从html页面生成PDF时遇到问题。我正在使用PDFkit。在安装它的过程中,我注意到我需要wkhtmltopdf。所以我也安装了它。我做了PDFkit的文档所说的一切......现在我在尝试加载PDF时遇到了这个错误。这里是错误:commandfailed:"/usr/local/bin/wkhtmltopdf""--margin-right""0.75in""--page-size""Letter""--margin-top""0.75in""--margin-bottom""0.75in""--encoding""UTF-8""--margin-left""0.75in""-

  10. ruby-on-rails - Rails 3 中的多个路由文件 - 2

    Rails2.3可以选择随时使用RouteSet#add_configuration_file添加更多路由。是否可以在Rails3项目中做同样的事情? 最佳答案 在config/application.rb中:config.paths.config.routes在Rails3.2(也可能是Rails3.1)中,使用:config.paths["config/routes"] 关于ruby-on-rails-Rails3中的多个路由文件,我们在StackOverflow上找到一个类似的问题

随机推荐