jjzjj

跨时区的 Java Quartz-Scheduler

coder 2024-03-06 原文

我的服务器在欧洲/罗马时区运行 - 这是服务器上的默认时区 - 我需要根据用户的时区安排作业,所以,如果用户, 生活在太平洋/火奴鲁鲁时区,安排一个 CronTrigger 在每天下午 22:00 为他所在的地球区域触发我找到了这个解决方案:

CronTrigger trigger = newTrigger()
  .withIdentity("name", "group")
  .withSchedule(
    cronSchedule("0 0 22 ? * *").inTimeZone(TimeZone.getTimeZone("Pacific/Honolulu"))
  )
  .startNow()
  .build();

在我的服务器上这项工作在“我的”第二天早上 09:00 开始

除了保持更新时区(即 Timezone Updater Tool)之外,还有哪些特殊问题需要考虑?

如果我想为之前的工作定义.startAt()和.endAt(),这样的日期可以吗? 使用此过程可能的夏令时是安全的吗?

Calendar calTZStarts = new GregorianCalendar(TimeZone.getTimeZone("Pacific/Honolulu"));
calTZStarts.set(2013, Calendar.JANUARY, 10);

Calendar calTZEnds = new GregorianCalendar(TimeZone.getTimeZone("Pacific/Honolulu"));
calTZEnds.set(2013, Calendar.JANUARY, 30);

Calendar calStarts = Calendar.getInstance();
calStarts.set(Calendar.YEAR, calTZStarts.get(Calendar.YEAR));
calStarts.set(Calendar.MONTH, calTZStarts.get(Calendar.MONTH));
calStarts.set(Calendar.DAY_OF_MONTH, calTZStarts.get(Calendar.DAY_OF_MONTH));
calStarts.set(Calendar.HOUR_OF_DAY, calTZStarts.get(Calendar.HOUR_OF_DAY));
calStarts.set(Calendar.MINUTE, calTZStarts.get(Calendar.MINUTE));
calStarts.set(Calendar.SECOND, calTZStarts.get(Calendar.SECOND));
calStarts.set(Calendar.MILLISECOND, calTZStarts.get(Calendar.MILLISECOND));

Calendar calEnds = Calendar.getInstance();
calEnds.set(Calendar.YEAR, calTZEnds.get(Calendar.YEAR));
calEnds.set(Calendar.MONTH, calTZEnds.get(Calendar.MONTH));
calEnds.set(Calendar.DAY_OF_MONTH, calTZEnds.get(Calendar.DAY_OF_MONTH));
calEnds.set(Calendar.HOUR_OF_DAY, calTZEnds.get(Calendar.HOUR_OF_DAY));
calEnds.set(Calendar.MINUTE, calTZEnds.get(Calendar.MINUTE));
calEnds.set(Calendar.SECOND, calTZEnds.get(Calendar.SECOND));
calEnds.set(Calendar.MILLISECOND, calTZEnds.get(Calendar.MILLISECOND));

CronTrigger trigger = newTrigger()
  .withIdentity("name", "group")
  .withSchedule(
    cronSchedule("0 0 22 ? * *").inTimeZone(TimeZone.getTimeZone("Pacific/Honolulu"))
  )
  .startAt(calStarts.getTime())
  .endAt(calEnds.getTime())
  .build();

或者我必须设置简单的开始和结束使用:

Calendar calTZStarts = new GregorianCalendar();
calTZStarts.set(2013, Calendar.JANUARY, 10, 0, 0, 0);

Calendar calTZEnds = new GregorianCalendar();
calTZEnds.set(2013, Calendar.JANUARY, 30, 0, 0, 0);

CronTrigger trigger = newTrigger()
  .withIdentity("name", "group")
  .withSchedule(
    cronSchedule("0 0 22 ? * *").inTimeZone(TimeZone.getTimeZone("Pacific/Honolulu"))
  )
  .startAt(calTZStarts.getTime())
  .endAt(calTZEnds.getTime())
  .build();

然后作业在“太平洋/火奴鲁鲁”定义的日期正确开始/结束?

提前感谢您的每一个建议

最佳答案

我想我找到了解决方案,经过测试并且它一直有效,直到证明不是这样 ;)

恢复 我的服务器在特定时区(即欧洲/罗马)运行

如果 Pacific/Honolulu TZ 上的用户想要安排一个工作,该工作从 2013 年 1 月 27 日星期日下午 3:00 开始,结束于 2013 年 1 月 31 日星期四晚上 9:00,从 2 开始每天每五分钟触发一次:00PM 到 10:55PM (0 0/5 14-22 * * ?) 正确的方法如下:

  • 在 CronScheduleBuilder 的 inTimeZone 方法中设置用户时区
  • 通过从太平洋/火奴鲁鲁转换为欧洲/罗马来适应服务器时间的 startAt 和 endAt 日期

示例代码:

// Begin User Input
String userDefinedTZ = "Pacific/Honolulu"; // +11

int userStartYear = 2013;
int userStartMonth = Calendar.JANUARY;
int UserStartDayOfMonth = 27;
int userStartHour = 15;
int userStartMinute = 0;
int userStartSecond = 0;

int userEndYear = 2013;
int userEndMonth = Calendar.JANUARY;
int UserEndDayOfMonth = 31;
int userEndHour = 21;
int userEndMinute = 0;
int userEndSecond = 0;
// End User Input


Calendar userStartDefinedTime = Calendar.getInstance();
// set start schedule by user input
userStartDefinedTime.set(userStartYear, userStartMonth, UserStartDayOfMonth, userStartHour, userStartMinute, userStartSecond);

Calendar userEndDefinedTime = Calendar.getInstance();
// set end schedule by user input
userEndDefinedTime.set(userEndYear, userEndMonth, UserEndDayOfMonth, userEndHour, userEndMinute, userEndSecond);


CronTrigger trigger = newTrigger()
  .withIdentity("name", "group")
  .withSchedule(
    // define timezone for the CronScheduleBuilder
    cronSchedule("0 0/5 14-22 * * ?").inTimeZone(TimeZone.getTimeZone("Pacific/Honolulu"))
  )
  // adapt user start date to server timezone
  .startAt( convertDateToServerTimeZone(userStartDefinedTime.getTime(), userDefinedTZ) )
  // adapt user end date to server timezone
  .endAt( convertDateToServerTimeZone(userEndDefinedTime.getTime(), userDefinedTZ) )
  .build();

根据 tz 转换日期的实用程序:

public Calendar convertDateToServerTimeZone(Date dateTime, String timeZone) {
    Calendar userDefinedTime = Calendar.getInstance();
    userDefinedTime.setTime(dateTime);
    if(!TimeZone.getDefault().getID().equalsIgnoreCase(timeZone)) {
    System.out.println        ("original defined time: " + userDefinedTime.getTime().toString() + " on tz:" + timeZone);
    Calendar quartzStartDate = new GregorianCalendar(TimeZone.getTimeZone(timeZone));
    quartzStartDate.set(Calendar.YEAR, userDefinedTime.get(Calendar.YEAR));
    quartzStartDate.set(Calendar.MONTH, userDefinedTime.get(Calendar.MONTH));
    quartzStartDate.set(Calendar.DAY_OF_MONTH, userDefinedTime.get(Calendar.DAY_OF_MONTH));
    quartzStartDate.set(Calendar.HOUR_OF_DAY, userDefinedTime.get(Calendar.HOUR_OF_DAY));
    quartzStartDate.set(Calendar.MINUTE, userDefinedTime.get(Calendar.MINUTE));
    quartzStartDate.set(Calendar.SECOND, userDefinedTime.get(Calendar.SECOND));
    quartzStartDate.set(Calendar.MILLISECOND, userDefinedTime.get(Calendar.MILLISECOND));
    System.out.println("adapted time for " + TimeZone.getDefault().getID() + ": " + quartzStartDate.getTime().toString());
    return quartzStartDate;
    } else {
    return userDefinedTime;
    }
}

== 更新开始 2012-01-24 ==

基于 Quartz 的实用程序,使用 DateBuilder 转换基于 tz 的日期:

public Calendar convertDateToServerTimeZone(Date dateTime, String timeZone) {
    Calendar userDefinedTime = Calendar.getInstance();
    userDefinedTime.setTime(dateTime);
    if(!TimeZone.getDefault().getID().equalsIgnoreCase(timeZone)) {
      System.out.println("original defined time: " + userDefinedTime.getTime().toString() + " on tz:" + timeZone);

      Date translatedTime = DateBuilder.translateTime(userDefinedTime.getTime(), TimeZone.getDefault(), TimeZone.getTimeZone(timeZone));

      Calendar quartzStartDate = new GregorianCalendar();
      quartzStartDate.setTime(translatedTime);
      System.out.println("adapted time for " + TimeZone.getDefault().getID() + ": " + quartzStartDate.getTime().toString());
      return quartzStartDate;
    } else {
      return userDefinedTime;
    }
}

== 更新结束 2012-01-24 ==

所以在我的欧洲/罗马 Quartz 服务器上,这项工作计划从 CET 2013 年 1 月 28 日星期一 02:00:00 到 2013 年 CET 2013 年 2 月 1 日星期五 08:00:00 开始,每天每五分钟触发一次01:00AM 至 08:55PM

在为开始和结束时间构建日期时,在实例化日期之前还要指定时区(在 java.util.Calendar 或日期格式字符串或 org.quartz.DateBuilder 上)。然后 quartz 将日期存储为自 1970 年 1 月 1 日以来在该特定时区的 UTC 中的毫秒数 - 因此当服务器的时区发生变化时,触发器不受影响。

关于跨时区的 Java Quartz-Scheduler,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14273917/

有关跨时区的 Java Quartz-Scheduler的更多相关文章

  1. ruby - 如何每月在 Heroku 运行一次 Scheduler 插件? - 2

    在选择我想要运行操作的频率时,唯一的选项是“每天”、“每小时”和“每10分钟”。谢谢!我想为我的Rails3.1应用程序运行调度程序。 最佳答案 这不是一个优雅的解决方案,但您可以安排它每天运行,并在实际开始工作之前检查日期是否为当月的第一天。 关于ruby-如何每月在Heroku运行一次Scheduler插件?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/8692687/

  2. ruby - 没有轨道的 ActiveRecord 时区 - 2

    我在非Rails项目中使用ActiveRecord。在Rails中,我可以这样做:config.time_zone='EasternTime(US&Canada)'config.active_record.default_timezone='EasternTime(US&Canada)'但如果我不使用rails,我该如何设置时区? 最佳答案 ActiveRecord::Base.default_timezone='EasternTime(US&Canada)' 关于ruby-没有轨道的A

  3. ruby - 将 Logstash 中的时间戳时区转换为输出索引名称 - 2

    在我的场景中,Logstash收到的系统日志行的“时间戳”是UTC,我们在Elasticsearch输出中使用事件“时间戳”:output{elasticsearch{embedded=>falsehost=>localhostport=>9200protocol=>httpcluster=>'elasticsearch'index=>"syslog-%{+YYYY.MM.dd}"}}我的问题是,在UTC午夜,Logstash在外时区(GMT-4=>America/Montreal)结束前将日志发送到不同的索引,并且索引在20小时(晚上8点)之后没有日志,因为“时间戳”是UTC。我们已

  4. ruby - 如何设置mysql2时区选项以删除查询警告 - 2

    使用mysql2做查询总是得到警告/usr/local/lib/ruby/gems/1.9.1/gems/mysql2-0.2.6/lib/active_record/connection_adapters/mysql2_adapter.rb:463:warning::database_timezoneoptionmustbe:utcor:local-defaultingto:local我确实看到了时区选项Mysql2现在支持两个时区选项::database_timezone-thisisthetimezoneMysql2willassumefieldsarealreadystored

  5. ruby - 我可以覆盖 Ruby 中的系统时区吗? - 2

    我在Ubuntu12.04上,我可以看到:$cat/etc/timezoneAmerica/Phoenix相应地,Time将返回一个非UTC时区的时间:$irb>Time.now=>2013-03-2713:44:49-0700>Time.at0=>1969-12-3117:00:00-0700我可以使用TZ环境变量覆盖系统时区:$TZ=UTCirb>Time.now=>2013-03-2720:47:19+0000>Time.at0=>1970-01-0100:00:00+0000无论如何,我可以在Ruby进程中以编程方式进行此更改吗? 最佳答案

  6. ruby - 更改纯 ruby (不是铁轨)的时区 - 2

    我正在构建一个混合UTC/PST数据源的Sinatra站点,但将在PST中查看。所以我需要一种方法来轻松地将时间对象从UTC转换为PST。没有Rails,我无法访问Time.zone、in_time_zone等。我只需要更改Time对象的时区,例如2014-08-1421:44:17+0000=>2014-08-1414:44:17-0700。首先我尝试了这个:classTimedefto_pstself.utc+Time.zone_offset('PDT')endend但这会改变实际时间戳而不是区域。我需要time.to_i和time.strftime才能工作;所以我不能改变时间戳的

  7. ruby - 在 resque scheduler 中调度动态作业 - 2

    我正在尝试使用rails3和resquescheduler测试future的调度作业:以下是我正在使用的代码,但我收到NoMethodErrorset_schedule。Resque.set_schedule("1",{:cron=>"306**1",:class=>"Notify",:queue=>"username",:message=>'notificationmessage'})我尝试使用简单的入队Resque.enqueue(Notify,params[:message])并且效果很好。更新:以下是我遇到的错误:undefinedmethod`set_schedule'for

  8. ruby-on-rails - 在 rails 3.2 中存储带时区的时间戳 - 2

    我正在尝试将所有时间戳及其包含的时区存储在Rails应用程序中。我对ActiveRecord将它们转换为utc很好,但是我有多个应用程序访问同一个数据库,其中一些应用程序是根据时区要求实现的。所以我想做的是像往常一样让activerecord转换我的时间戳,然后将它们写入数据库,并在时间戳后附加字符串“America/Los_Angeles”或任何适当的时区。我目前在jruby1.7.8上运行rails3.2.13,它实现了ruby​​1.9.3api。我的数据库是postgres9.2.4,与activerecord-jdbcpostgresql-adaptergem连接。列类型是带

  9. ruby - 带 ActiveSupport 的默认时区(无 Rails) - 2

    如何在ActiveSupport中设置默认时区?这是发生了什么:irb-r'rubygems'ruby-1.8.7-p174>require'active_support'ruby-1.8.7-p174>require'active_support/time_with_zone'ruby-1.8.7-p174>Time.zoneruby-1.8.7-p174>nil如何将其默认设置为当前位置? 最佳答案 在rails中,它通过rails初始化程序在environment.rb中设置Rails::Initializer.rundo|c

  10. ruby - 如何使用 Ruby 和 MongoId 正确保存时区? - 2

    如果这有点菜鸟问题,请原谅:我有一个应用程序,用户可以在其中设置自己的个人资料中的时区。当有人添加阵容(应用特定术语)时,我执行以下操作:time=ActiveSupport::TimeZone.new(user.timezone).parse("Wednesday,26October,201113:30:00")#Thisoutputs:2011-10-2613:30:00+0200-validaccordingtotheuserselectedTZ然后我保存阵容:Lineup.create({:date=>time.gmtime,:uid=>user._id,:pid=>produ

随机推荐