Calendar类经常用法 日期间的转换 set方法有巨坑

Posted yutingliuyl

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Calendar类经常用法 日期间的转换 set方法有巨坑相关的知识,希望对你有一定的参考价值。

       今天发现项目的工具类方法有个bug,并且还能迷惑你的bug,刚開始也是非常迷惑,由于这个bug之前出现过,可是过了两天就自己好了。今天又出现了。哦对,今天是

2017年3月31日,之所以说今天的日期,就是跟bug有关,且看以下代码

calendarInstance.set(Calendar.MONTH, calendarInstance.get(Calendar.MONTH) + 1 );

        这句代码意思非常明确。获取当前日期的下一月。

正常来说,比方不是今天,是3月30日,这句代码运行一点问题没有,得到4月30日,可是今天是3月31日,得到的日期是4月31日,而4月没有31日,顺延到5月1日,那么,这个不定时出现的bug就出现了。还非常easy迷惑你,由于第二天bug自己主动恢复了。一点问题没有。摊手。。

        正确使用方法应该是以下,得到4月30日

calendarInstance.add(Calendar.MONTH, 1 );

顺便列一下Calendar的使用方法

Calendar类的静态方法getInstance()能够初始化一个日历对象:

Calendar now = Calendar.getInstance();

能够使用以下三个方法把日历定到不论什么一个时间:

set(int year ,int month,int date)
set(int year ,int month,int date,int hour,int minute)
set(int year ,int month,int date,int hour,int minute,int second)

假设想获得年份、月份、小时等信息能够使用:

    Now.get(Calendar.Month) 这个方案 0表示一月,1表示二月
    get(Calendar.DAY_OF_MONTH)获得这个月的第几天
    get(Calendar.DAY_OF_WEEK)获得这个星期的第几天
    get(Calendar.DAY_OF_YEAR)获得这个年的第几天
    getTimeMillis()获得当前时间的毫秒表示

例如以下是Calendar类方法简单介绍

    abstract void add(int field, int amount) 依据日历的规则。为给定的日历字段加入或减去指定的时间量。


    boolean after(Object when) 推断此 Calendar 表示的时间是否在指定 Object 表示的时间之后,返回推断结果。
    boolean before(Object when) 推断此 Calendar 表示的时间是否在指定 Object 表示的时间之前,返回推断结果。


    void clear()将此 Calendar 的所日历字段值和时间值(从历元至如今的毫秒偏移量)设置成没有定义。


    void clear(int field) 将此 Calendar 的给定日历字段值和时间值(从历元至如今的毫秒偏移量)设置成没有定义。


    Object clone()创建并返回此对象的一个副本。
    int compareTo(Calendar anotherCalendar) 比較两个 Calendar 对象表示的时间值(从历元至如今的毫秒偏移量)。
    protected void complete()填充日历字段中全部未设置的字段。
    protected abstract void computeFields()将当前毫秒时间值 time 转换为 fields[] 中的日历字段值。
    protected abstract void computeTime()将 fields[] 中的当前日历字段值转换为毫秒时间值 time。


    boolean equals(Object obj) 将此 Calendar 与指定 Object 比較。
    int get(int field)返回给定日历字段的值。
    int getActualMaximum(int field)给定此 Calendar 的时间值,返回指定日历字段可能拥有的最大值。
    int getActualMinimum(int field)给定此 Calendar 的时间值,返回指定日历字段可能拥有的最小值。


    static Locale[] getAvailableLocales()返回全部语言环境的数组。此类的 getInstance 方法能够为其返回本地化的实例。
    String getDisplayName(int field, int style, Locale locale) 返回给定 style 和 locale 下的日历 field 值的字符串表示形式。


    Map<String,Integer> getDisplayNames(int field, int style, Locale locale) 返回给定 style 和 locale 下包括日历 field 全部名称的 Map 及其对应字段值。
    int getFirstDayOfWeek()获取一星期的第一天;比如,在美国,这一天是 SUNDAY。而在法国,这一天是 MONDAY。
    abstract int getGreatestMinimum(int field)返回此 Calendar 实例给定日历字段的最高的最小值。
    static Calendar getInstance() 使用默认时区和语言环境获得一个日历。


    static Calendar getInstance(Locale aLocale) 使用默认时区和指定语言环境获得一个日历。


    static Calendar getInstance(TimeZone zone) 使用指定时区和默认语言环境获得一个日历。
    static Calendar getInstance(TimeZone zone, Locale aLocale) 使用指定时区和语言环境获得一个日历。
    abstract int getLeastMaximum(int field) 返回此 Calendar 实例给定日历字段的最低的最大值。


    abstract int getMaximum(int field) 返回此 Calendar 实例给定日历字段的最大值。
    int getMinimalDaysInFirstWeek()获取一年中第一个星期所需的最少天数。比如。假设定义第一个星期包括一年第一个月的第一天,则此方法将返回 1。
    abstract int getMinimum(int field) 返回此 Calendar 实例给定日历字段的最小值。


    Date getTime()返回一个表示此 Calendar 时间值(从历元至如今的毫秒偏移量)的 Date 对象。
    long getTimeInMillis()返回此 Calendar 的时间值,以毫秒为单位。
    TimeZone getTimeZone()获得时区。


    int hashCode()返回该此日历的哈希码。
    protected int internalGet(int field)返回给定日历字段的值。


    boolean isLenient()推断日期/时间的解释是否为宽松的。


    boolean isSet(int field) 确定给定日历字段是否已经设置了一个值,当中包含由于调用 get 方法触发内部字段计算而导致已经设置该值的情况。
    abstract void roll(int field, boolean up) 在给定的时间字段上加入或减去(上/下)单个时间单元,不更改更大的字段。
    void roll(int field, int amount) 向指定日历字段加入指定(有符号的)时间量,不更改更大的字段。
    void set(int field, int value) 将给定的日历字段设置为给定值。
    void set(int year, int month, int date) 设置日历字段 YEAR、MONTH 和 DAY_OF_MONTH 的值。
    void set(int year, int month, int date, int hourOfDay, int minute) 设置日历字段 YEAR、MONTH、DAY_OF_MONTH、HOUR_OF_DAY 和 MINUTE 的值。
    void set(int year, int month, int date, int hourOfDay, int minute, int second) 设置字段 YEAR、MONTH、DAY_OF_MONTH、HOUR、MINUTE 和 SECOND 的值。
    void setFirstDayOfWeek(int value) 设置一星期的第一天是哪一天;比如,在美国,这一天是 SUNDAY,而在法国。这一天是 MONDAY。
    void setLenient(boolean lenient) 指定日期/时间解释是否是宽松的。


    void setMinimalDaysInFirstWeek(int value) 设置一年中第一个星期所需的最少天数,比如。假设定义第一个星期包括一年第一个月的第一天,则使用值 1 调用此方法。


    void setTime(Date date) 使用给定的 Date 设置此 Calendar 的时间。
    void setTimeInMillis(long millis) 用给定的 long 值设置此 Calendar 的当前时间值。


    void setTimeZone(TimeZone value) 使用给定的时区值来设置时区。
    String toString() 返回此日历的字符串表示形式

Calendar的经常用法演示样例

1、计算某一月份的最大天数

Calendar time=Calendar.getInstance();
time.clear();
time.set(Calendar.YEAR,year);
time.set(Calendar.MONTH,i-1);//注意,Calendar对象默认一月为0            
int day=time.getActualMaximum(Calendar.DAY_OF_MONTH);//本月份的天数

注:在使用set方法之前。必须先clear一下。否则非常多信息会继承自系统当前时间

2、Calendar和Date的转化

(1) Calendar转化为Date

Calendar cal=Calendar.getInstance();
Date date=cal.getTime();

(2) Date转化为Calendar

Date date=new Date();
Calendar cal=Calendar.getInstance();
cal.setTime(date);

3、格式化输出日期时间

Date date=new Date();
SimpleDateFormat df=new SimpleDateFormat("yyyy-MM-dd hh:mm:ss");
System.out.println(df.format(date));

4、计算一年中的第几星期

(1)计算某一天是一年中的第几星期

Calendar cal=Calendar.getInstance();
cal.set(Calendar.YEAR, 2006);
cal.set(Calendar.MONTH, 8);
cal.set(Calendar.DAY_OF_MONTH, 3);
int weekno=cal.get(Calendar.WEEK_OF_YEAR);

(2)计算一年中的第几星期是几号

SimpleDateFormat df=new SimpleDateFormat("yyyy-MM-dd");
Calendar cal=Calendar.getInstance();
cal.set(Calendar.YEAR, 2006);
cal.set(Calendar.WEEK_OF_YEAR, 1);
cal.set(Calendar.DAY_OF_WEEK, Calendar.MONDAY);
System.out.println(df.format(cal.getTime()));

输出:

2006-01-02

5、add()和roll()的使用方法

(1) add()方法

SimpleDateFormat df=new SimpleDateFormat("yyyy-MM-dd");
Calendar cal=Calendar.getInstance();
cal.set(Calendar.YEAR, 2006);
cal.set(Calendar.MONTH, 8);
cal.set(Calendar.DAY_OF_MONTH, 3);
cal.add(Calendar.DATE, -4);
Date date=cal.getTime();
System.out.println(df.format(date));
cal.add(Calendar.DATE, 4);
date=cal.getTime();
System.out.println(df.format(date));

输出:

    2006-08-30
    2006-09-03

(2)roll方法

cal.set(Calendar.YEAR, 2006);
cal.set(Calendar.MONTH, 8);
cal.set(Calendar.DAY_OF_MONTH, 3);
cal.roll(Calendar.DATE, -4);
date=cal.getTime();
System.out.println(df.format(date));
cal.roll(Calendar.DATE, 4);
date=cal.getTime();
System.out.println(df.format(date));

输出:

    2006-09-29
    2006-09-03

可见。roll()方法在本月内循环,一般使用add()方法。

6、计算两个随意时间中间的间隔天数

(1)传进Calendar对象

/
**计算两个时间之间相隔天数
 * @param startday  開始时间
 * @param endday 结束时间
 * @return
 */
public int getIntervalDays(Calendar startday,Calendar endday){
    //确保startday在endday之前
    if(startday.after(endday)){
        Calendar cal=startday;
        startday=endday;
        endday=cal;
    }
    //分别得到两个时间的毫秒数
    long sl=startday.getTimeInMillis();
    long el=endday.getTimeInMillis();

    long ei=el-sl;   
    //依据毫秒数计算间隔天数
    return (int)(ei/(1000*60*60*24));
}

(2)传进Date对象

/**计算两个时间之间相隔天数
 * @param startday  開始时间
 * @param endday 结束时间
 * @return
 */
public int getIntervalDays(Date startday,Date endday){
    //确保startday在endday之前
    if(startday.after(endday)){
        Date cal=startday;
        startday=endday;
        endday=cal;
    }
    //分别得到两个时间的毫秒数
    long sl=startday.getTime();
    long el=endday.getTime();

    long ei=el-sl;   
    //依据毫秒数计算间隔天数
    return (int)(ei/(1000*60*60*24));
}

同理,能够用同样的方法计算出随意两个时间相隔的小时数。分钟数,秒钟数等

注:以上方法是全然按时间计算。有时并不能令人惬意,如:

startday="2006-10-11 20:00:00" endday="2006-10-12 8:00:00"

计算结果为0,可是我们或许相让计算结果变为1,此时能够用例如以下方法实现:

在传參之前,先设定endday的时间,如:

endday.set(Calendar.HOUR_OF_DAY, 23);
endday.set(Calendar.MINUTE, 59);
endday.set(Calendar.SECOND, 59);
endday.set(Calendar.MILLISECOND, 59);

这样再传进去startday,endday,则结果就如我们所愿了。只是。假设嫌以上方法麻烦,能够參考下面方法:

(3)改进精确计算相隔天数的方法

    public int getDaysBetween (Calendar d1, 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(Calendar.DAY_OF_YEAR) - d1.get(Calendar.DAY_OF_YEAR);
        int y2 = d2.get(Calendar.YEAR);
        if (d1.get(Calendar.YEAR) != y2) {
            d1 = (Calendar) d1.clone();
            do {
                days += d1.getActualMaximum(Calendar.DAY_OF_YEAR);//得到当年的实际天数
                d1.add(Calendar.YEAR, 1);
            } while (d1.get(Calendar.YEAR) != y2);
        }
        return days;
    }



demo

package cn.outofmemory.codes.Date;

import java.util.Calendar;
import java.util.Date;

public class CalendarDemo {
  public static void main(String[] args) {
     Calendar calendar=Calendar.getInstance();
     calendar.setTime(new Date());
     System.out.println("如今时间是:"+new Date());
     String year=String.valueOf(calendar.get(Calendar.YEAR));
     String month=String.valueOf(calendar.get(Calendar.MONTH)+1);
     String day=String.valueOf(calendar.get(Calendar.DAY_OF_MONTH));
     String week=String.valueOf(calendar.get(Calendar.DAY_OF_WEEK)-1);
     System.out.println("如今时间是:"+year+"年"+month+"月"+day+"日,星期"+week);
     long year2009=calendar.getTimeInMillis();
     calendar.set(1989,9,26);//这里与真实的月份之间相差1
     long year1989=calendar.getTimeInMillis();
     long days=(year2009-year1989)/(1000*60*60*24);
     System.out.println("今天和1989年10月26日相隔"+days+"天,"+"也就是说我在这个漂亮的星球上已经幸福的生活了"+days+"天。");

  }
}



以上是关于Calendar类经常用法 日期间的转换 set方法有巨坑的主要内容,如果未能解决你的问题,请参考以下文章

Java常用的日期操作

日期时间类

日期时间类

Calendar类

Java经常使用日期操作具体解释

Calendar类常用需求方法