How do I ignore weekends using the Java calendar?

前端 未结 2 1531
灰色年华
灰色年华 2020-12-11 07:27

I\'m trying to get the number of minutes between two particular time instances by ignoring weekends. This is what I\'ve done.

public static final List

        
相关标签:
2条回答
  • 2020-12-11 07:56

    JodaTime is the way to go, so @KatjaChristiansen is on the right track. If you need to use the Java Calendar, my solution would look like this:

    private static final long MILLIS_OF_WEEK = TimeUnit.DAYS.toMillis(7);
    private static final long MILLIS_OF_WORKWEEK = TimeUnit.DAYS.toMillis(5);
    
    public static int getMinsBetween(Date d1, Date d2, boolean onlyBusinessDays) {
        long duration = d2.getTime() - d1.getTime();
        if (onlyBusinessDays) {
            Date sat = toSaturdayMidnight(d1);
            long timeBeforeWeekend = Math.max(sat.getTime() - d1.getTime(), 0);
            if (duration > timeBeforeWeekend) {
                Date mon = toMondayMidnight(d2);
                long timeAfterWeekend = Math.max(d2.getTime() - mon.getTime(), 0);
                long numberOfWeekends = Math.max((duration / MILLIS_OF_WEEK) - 1, 0);
                duration = numberOfWeekends * MILLIS_OF_WORKWEEK + timeBeforeWeekend + timeAfterWeekend;
            }
        }
        return (int) TimeUnit.MILLISECONDS.toMinutes(duration);
    }
    
    private static Date toMondayMidnight(Date date) {
        Calendar cal = Calendar.getInstance();
        cal.setTime(date);
        switch (cal.get(Calendar.DAY_OF_WEEK)) {
            case Calendar.SATURDAY:
            case Calendar.SUNDAY:
                cal.add(Calendar.DAY_OF_MONTH, 7);
        }
        cal.set(Calendar.DAY_OF_WEEK, Calendar.MONDAY);
        toMidnight(cal);
        return cal.getTime();
    }
    
    private static Date toSaturdayMidnight(Date date) {
        Calendar cal = Calendar.getInstance();
        cal.setTime(date);
        cal.set(Calendar.DAY_OF_WEEK, Calendar.SATURDAY);
        toMidnight(cal);
        return cal.getTime();
    }
    
    private static void toMidnight(Calendar cal) {
        cal.set(Calendar.HOUR_OF_DAY, 0);
        cal.set(Calendar.MINUTE, 0);
        cal.set(Calendar.SECOND, 0);
        cal.set(Calendar.MILLISECOND, 0);
    }
    

    These tests pass:

    @Test
    public void testWithinSameDay() {
        assertMinsBetween(30, "2013-01-03 9:00", "2013-01-03 9:30");
    }
    
    @Test
    public void testOverWeekend() {
        assertMinsBetween(60, "2013-01-04 23:30", "2013-01-07 0:30");
    }
    
    @Test
    public void testWeekendStart() {
        assertMinsBetween(30, "2013-01-05 23:30", "2013-01-07 0:30");
    }
    
    @Test
    public void testTwoWeeks() {
        assertMinsBetween((int) TimeUnit.DAYS.toMinutes(10), "2013-01-08 23:30", "2013-01-22 23:30");
    }
    
    @Test
    public void testTwoWeeksAndOneDay() {
        assertMinsBetween((int) TimeUnit.DAYS.toMinutes(11), "2013-01-08 23:30", "2013-01-23 23:30");
    }
    
    @Test
    public void testOneWeekMinusOneDay() {
        assertMinsBetween((int) TimeUnit.DAYS.toMinutes(4), "2013-01-09 23:30", "2013-01-15 23:30");
    }
    
    private void assertMinsBetween(int expected, String start, String end) {
        try {
            assertEquals(expected, getMinsBetween(FORMAT.parse(start), FORMAT.parse(end), true));
        }
        catch (ParseException e) {
            throw new IllegalStateException(e);
        }
    }
    
    0 讨论(0)
  • 2020-12-11 08:10

    I would highly recommend using Joda-Time for anything concerning date manipulations in Java, because it comes with a lot of helpful functions to make the code less complicated.

    This code uses JodaTime:

    public static final List<Integer> NON_WORKING_DAYS;
    static {
        List<Integer> nonWorkingDays = new ArrayList<Integer>();
        nonWorkingDays.add(DateTimeConstants.SATURDAY);
        nonWorkingDays.add(DateTimeConstants.SUNDAY);
        NON_WORKING_DAYS = Collections.unmodifiableList(nonWorkingDays);
    }
    
    public static Minutes getMinsBetween(DateTime d1, DateTime d2,
            boolean onlyBusinessDays) {
    
        BaseDateTime startDate = onlyBusinessDays && !isBusinessDay(d1) ?
                    new DateMidnight(d1) : d1;
        BaseDateTime endDate = onlyBusinessDays && !isBusinessDay(d2) ?
                    new DateMidnight(d2) : d2;
    
        Minutes minutes = Minutes.minutesBetween(startDate, endDate);
    
        if (onlyBusinessDays) {
            DateTime d = new DateTime(startDate);
    
            while (d.isBefore(endDate)) {
                if (!isBusinessDay(d)) {
                    Duration dayDuration = new Duration(d, d.plusDays(1));
                    minutes = minutes.minus(int) dayDuration.getStandardMinutes());
                }
                d = d.plusDays(1);
            }
        }
        return minutes;
    }
    
    private static boolean isBusinessDay(DateTime dateToCheck) {
        return !NON_WORKING_DAYS.contains(dateToCheck.dayOfWeek().get());
    }
    

    When this code is tested, it gives the following results:

    DateTime d1 = new DateTime(2013, 1, 4, 18, 0); // a Friday, 6 pm
    DateTime d2 = new DateTime(2013, 1, 7, 6, 0);  // the following Monday, 6 am
    
    Minutes minutes = getMinsBetween(d1, d2, true);
    System.out.println(minutes.toStandardHours().getHours()); // outputs "12" (in hours)
    
    d1 = new DateTime(2013, 1, 5, 12, 0); // a Saturday, 12 pm
    d2 = new DateTime(2013, 1, 6, 12, 0); // the following Sunday, 12 pm
    
    minutes = getMinsBetween(d1, d2, true);
    System.out.println(minutes.toStandardHours().getHours()); // outputs "0" (in hours)
    
    d1 = new DateTime(2013, 1, 5, 12, 0); // a Saturday, 12 pm
    d2 = new DateTime(2013, 1, 7, 6, 0);  // the following Monday, 6 am
    
    minutes = getMinsBetween(d1, d2, true);
    System.out.println(minutes.toStandardHours().getHours()); // outputs "6" (in hours)
    

    I just tested a case where the month changes over the weekend: From Friday, March 29th (6pm) to Monday, April 1st (6am):

    d1 = new DateTime(2013, 3, 29, 18, 0);
    d2 = new DateTime(2013, 4, 1, 6, 0);
    
    minutes = getMinsBetween(d1, d2, true);
    System.out.println(minutes.toStandardHours().getHours());
    

    The result is 12 hours, so it works for the month change.


    My first solution wasn't handling daylight saving times correctly. We have to determine the duration of each actual day when subtracting the minutes because days with a change in daylight saving time will not be 24h:

    if (!isBusinessDay(d)) {
        Duration dayDuration = new Duration(d, d.plusDays(1));
        minutes = minutes.minus(int) dayDuration.getStandardMinutes());
    }
    
    0 讨论(0)
提交回复
热议问题