java calendar - 获取一周的最后5天(没有周末)

时间:2012-03-07 14:56:49

标签: java calendar

我正在处理一些问题。

我需要从数据库中获取一些数据

1)从最后5天获得5条记录(每天一条记录),但日期必须是一周的日子(现在是周末)

2)从该月获得20条记录(当前或过去,每天一条记录)

我正在与Java日历作斗争,但我仍然不知道如何获取日期。

有人能帮帮我吗?

4 个答案:

答案 0 :(得分:2)

应该工作:

private Calendar workingDaysBack(final Calendar from, final int count) {
    for (int daysBack = 0; daysBack < count; ++daysBack) {
        do {
            from.add(Calendar.DAY_OF_YEAR, -1);
        } while(isWeekend(from));
    }
    return from;
}

private boolean isWeekend(Calendar cal) {
    return cal.get(Calendar.DAY_OF_WEEK) == Calendar.SATURDAY ||
           cal.get(Calendar.DAY_OF_WEEK) == Calendar.SUNDAY;
}

答案 1 :(得分:1)

您可以通过在您想要考虑的每一天调整星期几参数来完成第一部分。取决于您在几周内运行时的行为 - 应该是当前周还是最后一周?在这里,我假设当前一周......

Calendar cal = Calendar.getInstance();

List<Date> last5Days = new LinkedList<Date>();
List<Integer> weekDays = Arrays.asList(Calendar.MONDAY,
        Calendar.TUESDAY, Calendar.WEDNESDAY, Calendar.THURSDAY,
        Calendar.FRIDAY);
for (int weekDay : weekDays) {
    cal.set(Calendar.DAY_OF_WEEK, weekDay);
    last5Days.add(cal.getTime());
}
System.out.println(last5Days);

从当月起20周。取决于您想要的周数,而不是跨月份边界。在这里,我已经到了第一个星期一,然后从那里继续。

List<Date> last20Days = new LinkedList<Date>();
// find first Monday in the month
for (int i = 1; i <= 31; i++) {
    cal.set(Calendar.DAY_OF_MONTH, i);
    if (cal.get(Calendar.DAY_OF_WEEK) == weekDays.get(0)) {
        break;
    }
}
for (int i = 1; i <= 31; i++) {
    if (last20Days.size() == 20) {
        break;
    }
    if (weekDays.contains(cal.get(Calendar.DAY_OF_WEEK))) {
        last20Days.add(cal.getTime());
    }
    cal.add(Calendar.DAY_OF_MONTH, 1);
}
System.out.println(last20Days);

答案 2 :(得分:1)

TL;博士

myLocalDate.with( 
    org.threeten.extra.Temporals.previousWorkingDay() 
)

使用java.time

其他Answers使用麻烦的旧日期时间类,例如Calendar,而不是现在的遗留类,取而代之的是java.time类。

要获得过去五天,首先需要“今天”。

LocalDate

LocalDate类表示没有时间且没有时区的仅限日期的值。

时区对于确定日期至关重要。对于任何给定的时刻,日期在全球范围内因地区而异。例如,在Paris France午夜后的几分钟是新的一天,而Montréal Québec中仍然是“昨天”。

continent/region的格式指定proper time zone name,例如America/MontrealAfrica/CasablancaPacific/Auckland。切勿使用诸如ESTIST之类的3-4字母缩写,因为它们不是真正的时区,不是标准化的,甚至不是唯一的(!)。

ZoneId z = ZoneId.of( "America/Montreal" );
LocalDate today = LocalDate.now( z );

DayOfWeek

要获取前几天,您可以致电minusDays逐步增加日期。

LocalDate yesterday = today.minusDays( 1 );
LocalDate dayBeforeYesterday = yesterday.minusDays( 1 );

您可以使用LocalDate枚举来测试每个DayOfWeek对象的星期几。该类预定义七个对象,一个星期中的每一天。请注意,这些是真实的对象,而不仅仅是字符串。这里根本不需要使用字符串。

Boolean isSaturday = myLocalDate.getDayOfWeek().equals( DayOfWeek.SATURDAY );
Boolean isSunday = myLocalDate.getDayOfWeek().equals( DayOfWeek.SUNDAY );

EnumSet

进行比较的更灵活,更灵活的方法是使用EnumSetSet的实现针对枚举进行了优化,执行速度极快且使用的内存非常少。

EnumSet<DayOfWeek> weekend = EnumSet.of( DayOfWeek.SATURDAY , DayOfWeek.SUNDAY );

使用contains来测试LocalDate对象的DayOfWeek

Boolean isWeekend = weekend.contains( myLocalDate.getDayOfWeek() );

ThreeTen-Extra TemporalAdjuster previousWorkingDay()

ThreeTen-Extra项目使用其他功能扩展了java.time类。特别是在这里,我们可以使用previousWorkingDay接口的TemporalAdjuster实现。

这个调整器是硬编码的,可以跳过周六和周日。如果您想要考虑其他星期值,可以编写自己的TemporalAdjuster实现。

java.time类使用不可变对象。因此,我们不是更改对象上的值(“mutate”),而是生成一个基于原始值的新对象。因此,我们在此处致电LocalDate::with,传递TemporalAdjuster,并在周六/周日跳过时生成代表先前日期的新LocalDate

LocalDate previousWorkingDay = 
    myLocalDate.with( 
        org.threeten.extra.Temporals.previousWorkingDay() 
    ) 
;

完整示例

所以,让我们把它们放在一起。

ZoneId z = ZoneId.of ( "America/Montreal" );
LocalDate today = LocalDate.now ( z );

Integer countDaysPrior = 5;
List<LocalDate> priorWeekdays = new ArrayList<> ( countDaysPrior ); // Set initial capacity to number of days to be collected.
LocalDate localDate = today; // Initialize to today. Exclusive, we want only prior days.
for ( int i = 1 ; i <= countDaysPrior ; i ++ ) {
    localDate = localDate.with ( org.threeten.extra.Temporals.previousWorkingDay () );  // Decrement each day into the past, skipping over Saturday/Sunday.
    priorWeekdays.add ( localDate );
}

System.out.println ( "Weekdays prior to " + today + " are: " + priorWeekdays );
  

在2017-02-27之前的工作日是:[2017-02-24,2017-02-23,2017-02-22,2017-02-21,2017-02-20]

我想你可以使用Lambdas和Streams使代码更短。

关于 java.time

java.time框架内置于Java 8及更高版本中。这些类取代了麻烦的旧legacy日期时间类,例如java.util.DateCalendar和&amp; SimpleDateFormat

现在位于Joda-Timemaintenance mode项目建议迁移到java.time类。

要了解详情,请参阅Oracle Tutorial。并搜索Stack Overflow以获取许多示例和解释。规范是JSR 310

您可以直接与数据库交换 java.time 对象。使用符合JDBC driver或更高版本的JDBC 4.2。不需要字符串,不需要java.sql.*类。

从哪里获取java.time类?

ThreeTen-Extra项目使用其他类扩展java.time。该项目是未来可能添加到java.time的试验场。您可以在此处找到一些有用的课程,例如IntervalYearWeekYearQuartermore

答案 3 :(得分:0)

只是一个开始

Calendar cal = Calendar.getInstance();//initilized with current date & time
Date date = cal.getTime();//returned the current date & time that cal holds
boolean isMonday = Calendar.MONDAY == cal.get(Calendar.DAY_OF_WEEK);//it will return an integer 

See API Doc for more