当前位置:首页 > Java技术 > 正文内容

java 中计算任意2个日期之间的工作天数

canca18年前 (2008-06-05)Java技术349

对于任意2个日期比如:date_start=2006-10-1、date_end=2006-10-14 ,首先计算这连个日期之间的时间间隔(天数),然后分别对date_start 和date_end 取得它们下一个星期一的日期,这样就可以得到一个新的可以整除7的完整日期间隔(这个新的日期间隔已经把星期几的问题剔出掉了),换一种说法就是我们可以得到,这两个新的日期之间的周数,拿这个周数乘以5就是工作日期了(tmpWorkingDays)。但是这个日期并不是我们所要的日期,接下来我们要做的就是计算date_start,date_end这两个日期对于根据它们所产生的新的日期之间的时间偏移量,date_start的偏移量(date_start_change)是需要加的,而date_end的这个偏移量(date_end_change)是需要减去的。 最后我们只要用tmpWorkingDays+date_start_change-date_end_change就是我们所要求的实际工作日了。以下是所有实现代码(两个日期跨年也没有问题)。

主要思路:

对于任意2个日期比如:date_start=2006-10-1、date_end=2006-10-14 ,首先计算这连个日期之间的时间间隔(天数),然后分别对date_start 和date_end 取得它们下一个星期一的日期,这样就可以得到一个新的可以整除7的完整日期间隔(这个新的日期间隔已经把星期几的问题剔出掉了),换一种说法就是我们可以得到,这两个新的日期之间的周数,拿这个周数乘以5就是工作日期了(tmpWorkingDays)。但是这个日期并不是我们所要的日期,接下来我们要做的就是计算date_start,date_end这两个日期对于根据它们所产生的新的日期之间的时间偏移量,date_start的偏移量(date_start_change)是需要加的,而date_end的这个偏移量(date_end_change)是需要减去的。 最后我们只要用tmpWorkingDays+date_start_change-date_end_change就是我们所要求的实际工作日了。以下是所有实现代码(两个日期跨年也没有问题)。

希望我的方法能对大家有用,应为这个算法没有考虑到节假日的问题,我以后会像办法处理,呵呵,今天就到这里了。

package com.date.utlit;

import java.text.SimpleDateFormat;
import java.util.Calendar;
import java.util.Date;
import java.util.GregorianCalendar;
/**
* 计算任意2个日期内的工作日(没有考虑到国定假日)
* @author user
*
*/
public class DateCal {

/**
    * @param args
    */
public static void main(String[] args) {
    try {

     String strDateStart = "2006-10-1";
     String strDateEnd = "2006-10-14";
     SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");
     Date date_start = sdf.parse(strDateStart);
     Date date_end = sdf.parse(strDateEnd);
     DateCal app = new DateCal();
     Calendar cal_start = Calendar.getInstance();
     Calendar cal_end = Calendar.getInstance();
     cal_start.setTime(date_start);
     cal_end.setTime(date_end);
     System.out.println("星期-->" + app.getChineseWeek(cal_start)
       + " 日期-->" + cal_start.get(Calendar.YEAR) + "-"
       + (cal_start.get(Calendar.MONTH) + 1) + "-"
       + cal_start.get(Calendar.DAY_OF_MONTH));
     System.out.println("星期-->" + app.getChineseWeek(cal_end) + " 日期-->"
       + cal_end.get(Calendar.YEAR) + "-"
       + (cal_end.get(Calendar.MONTH) + 1) + "-"
       + cal_end.get(Calendar.DAY_OF_MONTH));
     System.out.println("工作日为-->"
       + app.getWorkingDay(cal_start, cal_end));
     System.out.println("休息日-->"+app.getHolidays(cal_start, cal_end));
    } catch (Exception e) {
     // TODO: handle exception
    }
}

public int getDaysBetween(java.util.Calendar d1, java.util.Calendar d2) {
    if (d1.after(d2)) { // swap dates so that d1 is start and d2 is end
     java.util.Calendar swap = d1;
     d1 = d2;
     d2 = swap;
    }
    int days = d2.get(java.util.Calendar.DAY_OF_YEAR)
      - d1.get(java.util.Calendar.DAY_OF_YEAR);
    int y2 = d2.get(java.util.Calendar.YEAR);
    if (d1.get(java.util.Calendar.YEAR) != y2) {
     d1 = (java.util.Calendar) d1.clone();
     do {
      days += d1.getActualMaximum(java.util.Calendar.DAY_OF_YEAR);
      d1.add(java.util.Calendar.YEAR, 1);
     } while (d1.get(java.util.Calendar.YEAR) != y2);
    }
    return days;
}

/**
    * 计算2个日期之间的相隔天数
    * @param d1
    * @param d2
    * @return
    */
public int getWorkingDay(java.util.Calendar d1, java.util.Calendar d2) {
    int result = -1;
    if (d1.after(d2)) { // swap dates so that d1 is start and d2 is end
     java.util.Calendar swap = d1;
     d1 = d2;
     d2 = swap;
    }

    int betweendays = getDaysBetween(d1, d2);

  
    int charge_date = 0;
    int charge_start_date = 0;//开始日期的日期偏移量
    int charge_end_date = 0;//结束日期的日期偏移量
     // 日期不在同一个日期内
     int stmp;
     int etmp;
     stmp = 7 - d1.get(Calendar.DAY_OF_WEEK);
     etmp = 7 - d2.get(Calendar.DAY_OF_WEEK);
     if (stmp != 0 && stmp != 6) {// 开始日期为星期六和星期日时偏移量为0
      charge_start_date = stmp - 1;
     }
     if (etmp != 0 && etmp != 6) {// 结束日期为星期六和星期日时偏移量为0
      charge_end_date = etmp - 1;
     }
//    }
    result = (getDaysBetween(this.getNextMonday(d1), this.getNextMonday(d2)) / 7)
      * 5 + charge_start_date - charge_end_date;
    //System.out.println("charge_start_date>" + charge_start_date);
    //System.out.println("charge_end_date>" + charge_end_date);
    //System.out.println("between day is-->" + betweendays);
    return result;
}

public String getChineseWeek(Calendar date) {
    final String dayNames[] = { "星期日", "星期一", "星期二", "星期三", "星期四", "星期五",
      "星期六" };

    int dayOfWeek = date.get(Calendar.DAY_OF_WEEK);

    // System.out.println(dayNames[dayOfWeek - 1]);
    return dayNames[dayOfWeek - 1];

}

/**
    * 获得日期的下一个星期一的日期
    *
    * @param date
    * @return
    */
public Calendar getNextMonday(Calendar date) {
    Calendar result = null;
    result = date;
    do {
     result = (Calendar) result.clone();
     result.add(Calendar.DATE, 1);
    } while (result.get(Calendar.DAY_OF_WEEK) != 2);
    return result;
}

/**
    *
    * @param d1
    * @param d2
    * @return
    */
public int getHolidays(Calendar d1,Calendar d2){
    return this.getDaysBetween(d1, d2)-this.getWorkingDay(d1, d2);
  
}

}

扫描二维码推送至手机访问。

版权声明:本文由Ant.Master's Blog发布,如需转载请注明出处。

本文链接:https://iant.work/post/425.html

标签: Java技术
分享给朋友:

“java 中计算任意2个日期之间的工作天数” 的相关文章

对象序列化与反序列化

    序列化,并不是JAVA独有的。因此,在这里我用比较通俗的话说了。序列化就是把一个对象转换成有规则的二进制流。而反序列化就是把有规则的二进制数据重整成一个对象。其好处不难看见:1.可以把一个对象保存在一个文件里。例如,下载软件。当您关闭了软件,下次再打开...

Socket与ServerSocket的问题

//服务器端:import java.io.BufferedReader;import java.io.IOException;import java.io.InputStreamReader;import java.io.PrintStream;import java.net.ServerSock...

Java语言的反射机制

    由于项目的需要,在项目中要实现即插即用的方式,也就是说可以动态地加载包,不用设置CLASSPATH路径。当项目发布时,不可能要用户来设置环境变量吧!因此,就要用到JAVA的反射机制了。昨天,我是在研究JAVA的JNI技术。由于没有时间,所以过几天才写JNI。今天...

IM技术(1)

    做项目了,NetCL今天开工了,在这些日子里,我会将自己研究的内容写下来。做个记录,以下是我在网上搜到的。关于管理用户状态的解决方案,当然,我都有一个方案。不过对客户端的任务有点重吧,我方法是客户端从服务器端获到一个用户在线状态后,接着就与服务器无关了。好友离线...

JAVA内部类终极实例

最近心情不好,不想说太多东西了!电脑坏了,我现在又病了. class ClassFactory{ private final static String userName = "Hello,My name is CAnca."; public static Thread in =...

Java代理详话简说

        Java的代理,使项目实现低藕合成为可能。Spring框架中的AOP,内部就是用Java代理来实现。因此,认识JAVA代理原理对学习学Spring AOP是有密切关系的。    &nb...

发表评论

访客

◎欢迎参与讨论,请在这里发表您的看法和观点。