创建dateRange Scala,Joda,Java

时间:2013-07-17 03:47:05

标签: java scala jodatime

我花了好几个小时试图让下一段代码工作。

import org.joda.time.{DateTime, Period}


def dateRange(from: DateTime, to: DateTime, step: Period): Iterator[DateTime]      =Iterator.iterate(from)(_.plus(step)).takeWhile(!_.isAfter(to))

val range = {
dateRange(new DateTime(2012, 06, 30).minusYears(5), new DateTime(2000, 06, 30),new Period.months(6))
}

我正在尝试设置一个日期范围数组,以6个月为增量从2000年到2012年逐步完成。我面临的问题是以下错误。

Exception in thread "main" java.lang.IllegalArgumentException: No instant converter found for type: scala.Tuple3
at    org.joda.time.convert.ConverterManager.getInstantConverter(ConverterManager.java:165)
at org.joda.time.base.BaseDateTime.<init>(BaseDateTime.java:169)
at org.joda.time.DateTime.<init>(DateTime.java:241)
at tester.MomentumAlgo$class.$init$(MomentumAlgo.scala:154)
at tester.RunMomentumAlgo$$anon$1.<init>(RunMomentumAlgo.scala:86)
at tester.RunMomentumAlgo$.main(RunMomentumAlgo.scala:86)
at tester.RunMomentumAlgo.main(RunMomentumAlgo.scala)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:601)
at com.intellij.rt.execution.application.AppMain.main(AppMain.java:120)

我似乎与上一个Period.months()部分有关,但是我不知道如何修复它。我不知道Tuple3错误。

如果有人能给我一个不同的解决方案,那也会很棒。我想要一份2000至2012年的日期列表,每6个月一次。

欢迎提出任何问题。我认为这将是一段很常见的代码,但网上没有太多关于它的内容。

提前致谢。

6 个答案:

答案 0 :(得分:13)

解决方法是定义这样的日期:

val date = new DateTime().withYear(2013).withMonthOfYear(7).withDayOfMonth(16)

然后REPL中的整个序列变为:

scala> import org.joda.time.{DateTime, Period}
import org.joda.time.{DateTime, Period}

scala> def dateRange(from: DateTime, to: DateTime, step: Period): Iterator[DateTime]      =Iterator.iterate(from)(_.plus(step)).takeWhile(!_.isAfter(to))
dateRange: (from: org.joda.time.DateTime, to: org.joda.time.DateTime, step: org.joda.time.Period)Iterator[org.joda.time.DateTime]

scala> val from = new DateTime().withYear(2012).withMonthOfYear(6).withDayOfMonth(30).minusYears(5)
from: org.joda.time.DateTime = 2007-06-30T21:46:05.536-07:00

scala> val to = new DateTime().withYear(2000).withMonthOfYear(6).withDayOfMonth(30)
to: org.joda.time.DateTime = 2000-06-30T21:46:26.186-07:00

scala> val range = dateRange(from, to, new Period().withMonths(6))
range: Iterator[org.joda.time.DateTime] = non-empty iterator

scala> range.toList
res4: List[org.joda.time.DateTime] = List(
2000-06-30T21:46:26.186-07:00,
2000-12-30T21:46:26.186-08:00,
2001-06-30T21:46:26.186-07:00,
2001-12-30T21:46:26.186-08:00,
2002-06-30T21:46:26.186-07:00,
2002-12-30T21:46:26.186-08:00,
2003-06-30T21:46:26.186-07:00,
2003-12-30T21:46:26.186-08:00,
2004-06-30T21:46:26.186-07:00,
2004-12-30T21:46:26.186-08:00,
2005-06-30T21:46:26.186-07:00,
2005-12-30T21:46:26.186-08:00,
2006-06-30T21:46:26.186-07:00,
2006-12-30T21:46:26.186-08:00)

另外,我无法重现这一点,如我的评论中所述。似乎REPL和编译器中的行为是不同的。

答案 1 :(得分:5)

DateTime doesn't have一个带有三个int参数的构造函数,因此new DateTime(2012, 06, 30)以元组DateTime(Object)作为参数调用(2012, 06, 30)构造函数。文档说:

  

构造一个代表日期时间的Object实例。

     

如果对象暗示了一个年表(例如GregorianCalendar),那么将使用该年表。否则,使用ISO默认值。因此,如果传入GregorianCalendar,则使用的年表将是GJ,但如果在年表中传递日期将是ISO。

     

已识别的对象类型在ConverterManager中定义,包括ReadableInstantStringCalendarDateString格式由ISODateTimeFormat.dateTimeParser()描述。

不出所料,ConverterManager不知道如何处理Scala元组,这会导致异常。

  

如果有人能给我一个不同的解决方案,那也会很棒。我想要一份2000至2012年的日期列表,每6个月一次。

如果你真的想要日期,那么最好使用的类型是LocalDate(顺便说一下,它确实有你想要的构造函数)。如果您希望在这些日期开始时DateTime,那么您需要考虑使用的时区。

答案 2 :(得分:5)

我需要类似的东西。这就是我想出的:

import org.joda.time.{Period, DateTime}

class DateRange(val start: DateTime, val end: DateTime, val step: Period, inclusive: Boolean) extends Iterable[DateTime] {
    override def iterator: Iterator[DateTime] = new DateRangeIterator

    class DateRangeIterator extends Iterator[DateTime] {
        var current = start

        override def hasNext: Boolean = current.isBefore(end) || (inclusive && current == end)

        override def next(): DateTime = {
            val returnVal = current
            current = current.withPeriodAdded(step, 1)
            returnVal
        }
    }
}

示例用法:

val startOfDay: DateTime = new DateTime().withTimeAtStartOfDay()
val endOfDay: DateTime = startOfDay.plusDays(1)
val dateRange = new DateRange(startOfDay, endOfDay, Period.hours(1), false)
for (d <- dateRange) println(d)

输出:

2015-03-16T00:00:00.000-05:00
2015-03-16T01:00:00.000-05:00
2015-03-16T02:00:00.000-05:00
2015-03-16T03:00:00.000-05:00
2015-03-16T04:00:00.000-05:00
2015-03-16T05:00:00.000-05:00
2015-03-16T06:00:00.000-05:00
2015-03-16T07:00:00.000-05:00
2015-03-16T08:00:00.000-05:00
2015-03-16T09:00:00.000-05:00
2015-03-16T10:00:00.000-05:00
2015-03-16T11:00:00.000-05:00
2015-03-16T12:00:00.000-05:00
2015-03-16T13:00:00.000-05:00
2015-03-16T14:00:00.000-05:00
2015-03-16T15:00:00.000-05:00
2015-03-16T16:00:00.000-05:00
2015-03-16T17:00:00.000-05:00
2015-03-16T18:00:00.000-05:00
2015-03-16T19:00:00.000-05:00
2015-03-16T20:00:00.000-05:00
2015-03-16T21:00:00.000-05:00
2015-03-16T22:00:00.000-05:00
2015-03-16T23:00:00.000-05:00

答案 3 :(得分:4)

好的,这是完整的工作代码。

import org.joda.time.{Period, DateTime}

object runme {

  def main(args:Array[String]) {

  def dateRange(from: DateTime, to: DateTime, step: Period): Iterator[DateTime]
  =Iterator.iterate(from)(_.plus(step)).takeWhile(!_.isAfter(to))

  val range = { dateRange(new DateTime(2000, 06, 30,0,0,0,0).minusYears(5) ,new DateTime(2013, 06, 30,0,0,0,0),new Period(0,6,0,0,0,0,0,0))}

  range.foreach(u => { 
    print(u.getYear)
    print(u.getMonthOfYear)
    println(u.getDayOfMonth)
  })

 }
}

我认为我的主要问题是在DateTime()函数之后没有足够的数字(即毫秒等),这意味着编译器没有收到它想要的所有参数。如Alexey Romanov所述

然后打印所需范围的日期,并可用作迭代器。

希望能帮助他人。

感谢@Brian和其他人的帮助

答案 4 :(得分:1)

我偶然发现了这个问题,并提出了这个简单的解决方案。

def generateDateRange(from: LocalDate, to: LocalDate): List[Date] = {
    if(from.compareTo(to) > 0) Nil
    from :: generateDateRange(from.plusDays(1), to)
}

我们通过将范围的开头放在一天之后开始的新范围上来递归地建立日期列表。

答案 5 :(得分:0)

我对 hucko 的代码做了一些小改动

private def generateDateRange(from: LocalDate, to: LocalDate): List[LocalDate] = {
    if (from.compareTo(to) > 0) Nil
    else from :: generateDateRange(from.plusDays(1), to)
}