Date类和Calendar类

public class DateTest01 {

    public static void main(String[] args) {
        //获取并输出当前年月日星期
        Date d = new Date();
        //输出:Tue Jul 04 15:00:46 CST 2017
        System.out.println(d);
        System.out.println("------------------------");
        System.out.println("使用Date类指定时间");
        System.out.println("------------------------");
        @SuppressWarnings("deprecation")
        //输出:Mon Feb 01 00:00:00 CST 3892
        Date d2 = new Date(1992,1,1);
        System.out.println(d2);
        /*
         * 使用带参数的构造方法,可以构造指定日期的Date类对象,Date类中年份的参数应该是实际需要代表的年份减去1900,实际需要代表的月份减去1以后的值
         * */
        //如果要输出1992年1月1日应该是年份-1900,月份-1,日正常输入,然而这个构造方法是过时了的
        @SuppressWarnings("deprecation")
        Date d3 = new Date(92,0,1);
        System.out.println(d3);
    }

}

  Date类中的大部分方法已经过时了,使用Calendar类。Calendar类是一个抽象类,在实际使用时实现特定的子类的对象,创建对象的过程对程序员来说是透明的,只需要使用getInstance方法创建即可。

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

public class CalendarTest01 {

    public static void main(String[] args) {
        //获取当前时间
         Calendar c1 =  Calendar.getInstance() ;
         //下面输出的并不是我们想要的
         System.out.println(c1);
         //把 Calendar类中的日期信息提取出来到Date类中并输出
         Date d = c1.getTime();
         System.out.println(d);

    }

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

public class CalendarTest01 {

    public static void main(String[] args) {
         Calendar c1 =  Calendar.getInstance() ;
//         Calendar类中年份的数值直接书写,月份的值为实际的月份值减1,日期的值就是实际的日期值。
         c1.set(1992,0,1);
         Date d = c1.getTime();
         System.out.println(d);
//         如果只设定某个字段,例如日期的值,则可以使用如下set方法:public void set(int field,int value)
//         Calendar.YEAR——年份
//         Calendar.MONTH——月份
//         Calendar.DATE——日期
//         Calendar.DAY_OF_MONTH——日期,和上面的字段完全相同
//         Calendar.HOUR——12小时制的小时数
//         Calendar.HOUR_OF_DAY——24小时制的小时数
//         Calendar.MINUTE——分钟
//         Calendar.SECOND——秒
//         Calendar.DAY_OF_WEEK——星期几
//         后续的参数value代表,设置成的值。例如:把日期设置为10
         c1.set(Calendar.DATE,10);
    }

}
import java.util.Calendar;

public class CalendarTest01 {

    public static void main(String[] args) {
        Calendar c1 = Calendar.getInstance();
        //注意,如果没有设置某个元素,会出现随机数,比如没有设置秒,而你要get(秒)输出的就会是随机数
        c1.set(1992, 0, 1, 00, 30,10);
        // 年份
        int year = c1.get(Calendar.YEAR);
        // 月份
        int month = c1.get(Calendar.MONTH) + 1;
        // 日期
        int date = c1.get(Calendar.DATE);
        // 小时
        int hour = c1.get(Calendar.HOUR_OF_DAY);
        // 分钟
        int minute = c1.get(Calendar.MINUTE);
        // 秒
        int second = c1.get(Calendar.SECOND);
        // 星期几
        int day = c1.get(Calendar.DAY_OF_WEEK);
        System.out.println("年份:" + year);
        System.out.println("月份:" + month);
        System.out.println("日期:" + date);
        System.out.println("小时:" + hour);
        System.out.println("分钟:" + minute);
        System.out.println("秒:" + second);
        System.out.println("星期:" + day);
    }

}
import java.util.Calendar;

public class CalendarTest01 {

    public static void main(String[] args) {
        Calendar c1 = Calendar.getInstance();
        //abstract  void add(int field, int amount) 根据日历的规则,为给定的日历字段添加或减去指定的时间量。
        //比如:当前时间增加100天以后的日期
        c1.add(Calendar.DATE,100);
        // 年份
        int year = c1.get(Calendar.YEAR);
        // 月份
        int month = c1.get(Calendar.MONTH) + 1;
        // 日期
        int date = c1.get(Calendar.DATE);
        System.out.println("年份:" + year);
        System.out.println("月份:" + month);
        System.out.println("日期:" + date);
        //没有add方法增加100天:年份:2017 月份:7 日期:4
        //有add方法增加100天:年份:2017 月份:10 日期:12

    }

}
Date d = new Date();
Calendar c6 = Calendar.getInstance();
//Calendar类型的对象转换为Date对象
Date d1 = c6.getTime();
//Date类型的对象转换为Calendar对象
Calendar c7 = Calendar.getInstance();
c7.setTime(d);
Calendar对象和相对时间之间的互转
Calendar c8 = Calendar.getInstance();
long t = 1252785271098L;
//将Calendar对象转换为相对时间
long t1 = c8.getTimeInMillis();
//将相对时间转换为Calendar对象
Calendar c9 = Calendar.getInstance();
c9.setTimeInMillis(t1);
import java.util.Calendar;
//计算两个日期差
public class CalendarTest01 {

    public static void main(String[] args) {
        //设置两个日期
        //日期:2009年3月11号
        Calendar c1 = Calendar.getInstance();
        c1.set(2009, 2, 11);
        //日期:2010年4月1号
        Calendar c2 = Calendar.getInstance();
        c2.set(2010, 3, 1);
        //转换为相对时间
        long t1 = c1.getTimeInMillis();
        long t2 = c2.getTimeInMillis();
        //计算天数 386
        long days = (t2 - t1)/(24 * 60 * 60 * 1000);
        System.out.println(days);
        }
}
public class CalendarTest01 {

    public static void main(String[] args) {
        // 获得当前时间
        Calendar c = Calendar.getInstance();
        // 设置代表的日期为1号
        c.set(Calendar.DATE, 1);
        // 获得1号是星期几
        int start = c.get(Calendar.DAY_OF_WEEK);
        // 获得当前月的最大日期数
        int maxDay = c.getActualMaximum(Calendar.DATE);
        // 输出标题
        System.out.println("星期日 星期一 星期二 星期三 星期四 星期五 星期六");
        // 输出开始的空格
        for (int i = 1; i < start; i++) {
            System.out.print(" ");
        }
        // 输出该月中的所有日期
        for (int i = 1; i <= maxDay; i++) {
            // 输出日期数字
            System.out.print(" " + i);
            // 输出分隔空格
            System.out.print(" ");
            if (i < 10) {
                System.out.print(‘ ‘);
            }
            // 判断是否换行
            if ((start + i - 1) % 7 == 0) {
                System.out.println();
            }
        }
        // 换行
        System.out.println();
    }
}
    //1.Calendar 转化 String
    Calendar calendat = Calendar.getInstance();
    SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");
    String dateStr = sdf.format(calendar.getTime());

    //2.String 转化Calendar
    String str="2010-5-27";
    SimpleDateFormat sdf= new SimpleDateFormat("yyyy-MM-dd");
    Date date =sdf.parse(str);
    Calendar calendar = Calendar.getInstance();
    calendar.setTime(date);

    //3.Date 转化String
    SimpleDateFormat sdf= new SimpleDateFormat("yyyy-MM-dd");
    String dateStr=sdf.format(new Date());

    //4.String 转化Date
    String str="2010-5-27";
    SimpleDateFormat sdf= new SimpleDateFormat("yyyy-MM-dd");
    Date birthday = sdf.parse(str);

    //5.Date 转化Calendar
    Calendar calendar = Calendar.getInstance();
    calendar.setTime(new java.util.Date());

    //6.Calendar转化Date
    Calendar calendar = Calendar.getInstance();
    java.util.Date date =calendar.getTime(); 

时间格式化SimpleDateFormat

SimpleDateFormat 是 Java 中一个非常常用的类,该类用来对日期字符串进行解析和格式化输出,但如果使用不小心会导致非常微妙和难以调试的问题,因为 DateFormat 和 SimpleDateFormat 类不都是线程安全的,在多线程环境下调用 format() 和 parse() 方法应该使用同步代码来避免问题。下面我们通过一个具体的场景来一步步的深入学习和理解SimpleDateFormat类。
在程序中应当尽量少的创建SimpleDateFormat 实例,因为创建这么一个实例需要耗费很大的代价。在一个读取数据库数据导出到excel文件的例子当中,每次处理一个时间信息的时候,就需要创建一个SimpleDateFormat实例对象,然后再丢弃这个对象。大量的对象就这样被创建出来,占用大量的内存和 jvm空间。代码如下:

import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;

public class DateUtil {

    public static  String formatDate(Date date)throws ParseException{
         SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
        return sdf.format(date);
    }

    public static Date parse(String strDate) throws ParseException{
         SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
        return sdf.parse(strDate);
    }
}

创建一个静态的simpleDateFormat实例,然后放到一个DateUtil类(如下)中,在使用时直接使用这个实例进行操作,也并不能行

import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;

public class DateUtil {
    private static final  SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");

    public static  String formatDate(Date date)throws ParseException{
        return sdf.format(date);
    }

    public static Date parse(String strDate) throws ParseException{

        return sdf.parse(strDate);
    }
}

大部分的时间里面都会工作得很好。但当你在生产环境中使用一段时间之后,你就会发现这么一个事实:它不是线程安全的。在正常的测试情况之下,都没有问题,但一旦在生产环境中一定负载情况下时,这个问题就出来了。他会出现各种不同的情况,比如转化的时间不正确,比如报错,比如线程被挂死等等

import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;

public class DateUtil {

    private static final  SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");

    public static  String formatDate(Date date)throws ParseException{
        return sdf.format(date);
    }

    public static Date parse(String strDate) throws ParseException{

        return sdf.parse(strDate);
    }
}
import java.text.ParseException;
import java.util.Date;

public class DateUtilTest {

    public static class TestSimpleDateFormatThreadSafe extends Thread {
        @Override
        public void run() {
            while(true) {
                try {
                    this.join(2000);
                } catch (InterruptedException e1) {
                    e1.printStackTrace();
                }
                try {
                    System.out.println(this.getName()+":"+DateUtil.parse("2013-05-24 06:02:20"));
                } catch (ParseException e) {
                    e.printStackTrace();
                }
            }
        }
    }

    public static void main(String[] args) {
        for(int i = 0; i < 3; i++){
            new TestSimpleDateFormatThreadSafe().start();
        }

    }
}

说明:Thread-1和Thread-0报java.lang.NumberFormatException: multiple points错误,直接挂死,没起来;Thread-2 虽然没有挂死,但输出的时间是有错误的,比如我们输入的时间是:2013-05-24 06:02:20 ,当会输出:Mon May 24 06:02:20 CST 2021 这样的灵异事件。
原因:
  相比于共享一个变量的开销要比每次创建一个新变量要小很多。上面的优化过的静态的SimpleDateFormat版,之所在并发情况下回出现各种灵异错误,是因为SimpleDateFormat和DateFormat类不是线程安全的。我们之所以忽视线程安全的问题,是因为从SimpleDateFormat和DateFormat类提供给我们的接口上来看,实在让人看不出它与线程安全有何相干。只是在JDK文档的最下面有如下说明:
  SimpleDateFormat中的日期格式不是同步的。推荐(建议)为每个线程创建独立的格式实例。如果多个线程同时访问一个格式,则它必须保持外部同步。
SimpleDateFormat继承了DateFormat,在DateFormat中定义了一个protected属性的 Calendar类的对象:calendar。只是因为Calendar累的概念复杂,牵扯到时区与本地化等等,Jdk的实现中使用了成员变量来传递参数,这就造成在多线程的时候会出现错误。
calendar.setTime(date)这条语句改变了calendar,稍后,calendar还会用到(在subFormat方法里),而这就是引发问题的根源。想象一下,在一个多线程环境下,有两个线程持有了同一个SimpleDateFormat的实例,分别调用format方法:
  线程1调用format方法,改变了calendar这个字段。
  中断来了。
  线程2开始执行,它也改变了calendar。
  又中断了。
  线程1回来了,此时,calendar已然不是它所设的值,而是走上了线程2设计的道路。如果多个线程同时争抢calendar对象,则会出现各种问题,时间不对,线程挂死等等。
  分析一下format的实现,我们不难发现,用到成员变量calendar,唯一的好处,就是在调用subFormat时,少了一个参数,却带来了这许多的问题。其实,只要在这里用一个局部变量,一路传递下去,所有问题都将迎刃而解。
  这个问题背后隐藏着一个更为重要的问题--无状态:无状态方法的好处之一,就是它在各种环境下,都可以安全的调用。衡量一个方法是否是有状态的,就看它是否改动了其它的东西,比如全局变量,比如实例的字段。format方法在运行过程中改动了SimpleDateFormat的calendar字段,所以,它是有状态的。
  这也同时提醒我们在开发和设计系统的时候注意下一下三点:
  1.自己写公用类的时候,要对多线程调用情况下的后果在注释里进行明确说明
  2.对线程环境下,对每一个共享的可变变量都要注意其线程安全性
  3.我们的类和方法在做设计的时候,要尽量设计成无状态的
解决方案:

//需要的时候创建新实例
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;

public class DateUtil {

    public static  String formatDate(Date date)throws ParseException{
        //创建局部实例
        SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
        return sdf.format(date);
    }

    public static Date parse(String strDate) throws ParseException{
        SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
        return sdf.parse(strDate);
    }
}
//使用同步代码块,当线程较多时,当一个线程调用该方法时,其他想要调用此方法的线程就要block,多线程并发量大的时候会对性能有一定的影响
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;

public class DateSyncUtil {

    private static SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");

    public static String formatDate(Date date)throws ParseException{
        synchronized(sdf){
            return sdf.format(date);
        }
    }

    public static Date parse(String strDate) throws ParseException{
        synchronized(sdf){
            return sdf.parse(strDate);
        }
    }
}
//使用ThreadLocal存储DateFormat,推荐使用

import java.text.DateFormat;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;

public class ConcurrentDateUtil {

    private static ThreadLocal<DateFormat> threadLocal = new ThreadLocal<DateFormat>() {
        @Override
        protected DateFormat initialValue() {
            return new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
        }
    };

    public static Date parse(String dateStr) throws ParseException {
        return threadLocal.get().parse(dateStr);
    }

    public static String format(Date date) {
        return threadLocal.get().format(date);
    }
}
//使用ThreadLocal存储DateFormat,推荐使用
import java.text.DateFormat;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;

public class ThreadLocalDateUtil {
    private static final String date_format = "yyyy-MM-dd HH:mm:ss";
    private static ThreadLocal<DateFormat> threadLocal = new ThreadLocal<DateFormat>(); 

    public static DateFormat getDateFormat()
    {
        DateFormat df = threadLocal.get();
        if(df==null){
            df = new SimpleDateFormat(date_format);
            threadLocal.set(df);
        }
        return df;
    }  

    public static String formatDate(Date date) throws ParseException {
        return getDateFormat().format(date);
    }

    public static Date parse(String strDate) throws ParseException {
        return getDateFormat().parse(strDate);
    }   }

来源:http://www.cnblogs.com/peida/archive/2013/05/31/3070790.html

时间: 2024-10-28 11:15:26

Date类和Calendar类的相关文章

Java Date类和Calendar类的一个控制台打印日期的小程序

Java Date类和Calendar类的一个打印日期的小程序,可以直接用. package com.boy.Idate.calendar; import java.text.ParseException; import java.text.SimpleDateFormat; import java.util.Calendar; import java.util.Date; /** * 控制台可视化日历 * @author 田硕 */ public class VisualCalendar {

JAVA的Date类与Calendar类【转】

Date类 在JDK1.0中,Date类是唯一的一个代表时间的类,但是由于Date类不便于实现国际化,所以从JDK1.1版本开始,推荐使用Calendar类进行时间和日期处理.这里简单介绍一下Date类的使用. 1.使用Date类代表当前系统时间 Date d = new Date(); System.out.println(d); 使用Date类的默认构造方法创建出的对象就代表当前时间,由于Date类覆盖了toString方法,所以可以直接输出Date类型的对象,显示的结果如下: Sun Ma

Date类和Calendar类应用到计算活了多少天和判断闰年与平年

在javaSE阶段,Date类和Calendar类以后会经常用到 这两个类当中的一些常用方法 通过两个demo 进行学习和练习 要求如下:让用户自己输入yyyy-MM-dd 格式的年月日 然后得出到今天为止活了多少天 ,这里面用到SimpleDateFormat Scanner 代码如下 import java.text.ParseException;import java.text.SimpleDateFormat;import java.util.Date;import java.util.

处理时间的类 —— System类、Date类 、SimpleDateFormat类 与 Calendar类

在我们以往的编程中,就有过通过运行前和运行后时间差来判断时间复杂度的例子,再扯得远一点,我们在C语言中制造随机数的操作,也要用到有关时间的函数.而且,在我们未来的编程中,也会时不时要用到能够读取当前时间的方法,所以,本篇博文所讲解的类,其实还是挺常用的,所以本人专门写一篇博文来为同学们展示下这个类的常用方法的用法 System类: 概述: System 类包含一些有用的类字段和方法.它不能被实例化. 说真的,System类与时间相关的主要原因就是它的一个方法,那么,现在,本人来展示下这个类的常用

JAVA的Date类与Calendar类

Date类 在JDK1.0中,Date类是唯一的一个代表时间的类,但是由于Date类不便于实现国际化,所以从JDK1.1版本开始,推荐使用Calendar类进行时间和日期处理.这里简单介绍一下Date类的使用. 1.使用Date类代表当前系统时间 Date d = new Date(); System.out.println(d); 使用Date类的默认构造方法创建出的对象就代表当前时间,由于Date类覆盖了toString方法,所以可以直接输出Date类型的对象,显示的结果如下: Sun Ma

Java学习关于时间操作的应用类--Date类、Calendar类及其子类

Date类 Date类封装了当期时间和日期.与Java1.0定义的原始版的Date类相比,Date类发生了本质的变化.在Java1.1发布时,原始版Date类定义的许多功能被移进Calendar类和DateFormat类中.原始版Date类中的许多方法已经不赞成使用. Date类支持的构造函数: Date() Date(long   millisec) 第一个构造函数使用当前日期和时间初始化对象.第二个构造函数接收一个参数,该参数等于自1970年1月1日午夜以来经历的毫秒数. Date对象创建以

JAVA Date类与Calendar类【转】

Date类 在JDK1.0中,Date类是唯一的一个代表时间的类,但是由于Date类不便于实现国际化,所以从JDK1.1版本开始,推荐使用Calendar类进行时间和日期处理.这里简单介绍一下Date类的使用. 1.使用Date类代表当前系统时间 Date d = new Date(); System.out.println(d); 使用Date类的默认构造方法创建出的对象就代表当前时间,由于Date类覆盖了toString方法,所以可以直接输出Date类型的对象,显示的结果如下: Sun Ma

Java 中常用的类:包括基本类型的包装类、Date 类、SimpleDateFormat 类、 Calendar 类、 Math 类

JAVA中的包装类 包装类里没有String,它是引用数据类型 基本类型是不能调用方法的,而其包装类具有很多方法 包装类主要提供了两大类方法: 1. 将本类型和其他基本类型进行转换的方法 2. 将字符串和本类型及包装类互相转换的方法 基本类型 对应的包装类 byte Byte short Short int Integer long Long float Float double Double char Character boolean Boolean Integer m=new Intege

Date、DateFormat类、Calendar类、基本类型包装类、System类、Math类、Arrays类、大数据运算类

1.Date (1)概述: 时间的原点:公元1970年 一月一日,午夜0:00:00 对应的毫秒值就是0 注意:时间和日期的计算,必须依赖毫秒值 ----------------Date()获取当前时间 import java.util.Date; public class Demo06boke { public static void main(String[] args) { Date sdf=new Date(); System.out.println(sdf); } } Date(lon