localtime是非线程安全的(如何在Java 8中愉快地处理日期和时间)

2024-05-09 05:57:02 88

localtime是非线程安全的(如何在Java 8中愉快地处理日期和时间)

本文目录

如何在Java 8中愉快地处理日期和时间


JAVA 8 Time示例
import java.time.LocalDate;
import java.time.LocalDateTime;
import java.time.LocalTime;
import java.time.Month;
import java.time.temporal.ChronoUnit;
public class Java8TimeDemo {
//LocalDate和LocalTime和String一样,是final修饰的不变类型,线程安全且不能修改。
public static void main(String args) {
// 取当前日期:
LocalDate today = LocalDate.now();
System.out.println(today);
// 根据年月日取日期,月份从1开始到12 而不是java.util.date里的月份从0到11:
LocalDate birthday = LocalDate.of(2015, 10, 27);
long days = ChronoUnit.DAYS.between(birthday,today);
if(birthday.isBefore(today)){
System.out.println(“生日已过“+days+“天“);
}else{
System.out.println(“还差“+Math.abs(days)+“天过生日“);
}
int year = today.getYear();//年
System.out.println(year);
int month = today.getMonthValue();//月
System.out.println(month);
int day = today.getDayOfMonth();//日
System.out.println(day);
int dw = today.getDayOfWeek().getValue();//星期几
System.out.println(“星期“+dw);
int dd = today.lengthOfMonth();//本月多少天
System.out.println(dd);
boolean leapYear = today.isLeapYear();//是否是润年
System.out.println(leapYear?“是闰年“:“不是闰年“);
//LocalDate的实例。由于它是不可变类型,每次操作都会产生一个新的实例,而原有实例不收任何影响。
LocalDate date = LocalDate.of(1998, Month.NOVEMBER, 01);
date = date.withYear(2200); //设置年
System.out.println(date);
date = date.plusMonths(2); //增加2个月 plus
System.out.println(date);
date = date.minusDays(1);//减去1天
System.out.println(date);
LocalTime now = LocalTime.now(); // 包含毫秒数
System.out.println(now);
now = LocalTime.now().withNano(0); //清除毫秒数:
System.out.println(now);
LocalDateTime dateTime = LocalDateTime.now();
System.out.println(dateTime);
}
}

posix标准io是不是线程安全的额


localtime
localtime返回一个指向静态变量的指针,这样是线程不安全的,使用localtime_r代替
STL中的容器类
STL容器只保证线程读安全,不保证线程写安全
strtok
应用函数初次调用strtok时传递一个字串的地址,比如”aaa.bbb.dddd”,并将字串的地址保存在自己的静态变量中,当你将来再次调用strtok并传递NULL时(strtok的特殊用法,第二次调用时字串传NULL表示对第一次传进去的字串继续分隔,所以要先保存字串地址,这是有点怪异的实现),该函数就会引用保存好的字串地址。在多线程环境下,另一个线程也可能调用strtok,在这种环境下,另一个线程会在第一个线程不知道的情况下替换静态变量中的字串地址,这就会导致各种难以排除的错误出现。
asctime()
basename()
catgets()
crypt()
ctermid()//if passed a non-NULL argument
ctime()
dbm_clearerr()
dbm_close()
dbm_delete()
dbm_error()
dbm_fetch()
dbm_firstkey()
dbm_nextkey()
dbm_open()
dbm_store()
dirname()
dlerror()
drand48()
ecvt()//[POSIX.1-2001 only (function removed in POSIX.1-2008)]
encrypt()
endgrent()
endpwent()
endutxent()
fcvt()//[POSIX.1-2001 only (function removed in POSIX.1-2008)]
ftw()
gcvt()//[POSIX.1-2001 only (function removed in POSIX.1-2008)]
getc_unlocked()
getchar_unlocked()
getdate()
getenv()
genevn的实现原理为:
进程应当维护着一个全局变量
char
**environ,用来指向每一条环境变量的字符串,同时在原本的getenv的实现里,还应当维护着一个全局的字符串的缓冲区(暂且命名为envbuf),当线程调用getenv()时,系统找到对应的环境变量的字符串,拷贝到全局缓冲区envbuf中,再通过getenv()函数返回给调用者。
如果多个线程同时调用getenv()的话,可能就会出现对envbuf的异步访问,造成返回错误结果
getgrent()
getgrgid()
getgrnam()
gethostbyaddr()//[POSIX.1-2001 only (function removed in POSIX.1-2008)]
gethostbyname()//[POSIX.1-2001 only (function removed in POSIX.1-2008)]
gethostent()
getlogin()
getnetbyaddr()
getnetbyname()
getnetent()
getopt()
getprotobyname()
getprotobynumber()
getprotoent()
getpwent()
getpwnam()
getpwuid()
getservbyname()
getservbyport()
getservent()
getutxent()
getutxid()
getutxline()
gmtime()
hcreate()
hdestroy()
hsearch()
inet_ntoa()
l64a()
lgamma()
lgammaf()
lgammal()
localeconv()
localtime()
lrand48()
mrand48()
nftw()
nl_langinfo()
ptsname()
putc_unlocked()
putchar_unlocked()
putenv()
pututxline()
rand()
readdir()
setenv()
setgrent()
setkey()
setpwent()
setutxent()
strerror()
strsignal()//[Added in POSIX.1-2008]
system()//[Added in POSIX.1-2008]
system()中会调用fork()产生一个子进程,注意是进程;程序中所有的线程好像都被复制。。
这样,貌似不能在多线程环境中使用system()

tmpnam()//if passed a non-NULL argument
ttyname()
unsetenv()
wcrtomb()//if its final argument is NULL
wcsrtombs()//if its final argument is NULL
wcstombs()
wctomb()
参考文档:http://www.ccvita.com/506.html
结论:没什么事别用多线程,一不留神,死都不知道怎么死的..

localtime函数 在多线程中使用会core吗


从localtime函数的原型定义分析
struct tm *localtime(const time_t *timep);
localtime函数返回了一个struct tm指针,函数的说明中并未要求调用者将该指针free掉,证明tm对象并非是malloc或new出来的。那么tm一定是个全局变量或者是localtime函数中的局部静态变量。
这样的变量在多线程访问时会出问题,第一个线程在localtime返回之前首先改变了这个变量的值,第二个线程又进来改变这个变量,以此类推。这样第一个线程在真正返回时该值已经被多次改变,这是典型的函数重入问题。
因此localtime函数线程不安全。

如何在java8中愉快地处理日期和时间java8新增了localdate和localtim


怎么才能愉快地处理日期和时间?答案是:立刻升级到Java 8!
Java 8新增了LocalDate和LocalTime接口,为什么要搞一套全新的处理日期和时间的API?因为旧的java.util.Date实在是太难用了。
java.util.Date月份从0开始,一月是0,十二月是11,变态吧!java.time.LocalDate月份和星期都改成了enum,就不可能再用错了。
java.util.Date和SimpleDateFormatter都不是线程安全的,而LocalDate和LocalTime和最基本的String一样,是不变类型,不但线程安全,而且不能修改。
java.util.Date是一个“万能接口”,它包含日期、时间,还有毫秒数,如果你只想用java.util.Date存储日期,或者只存储时间,那么,只有你知道哪些部分的数据是有用的,哪些部分的数据是不能用的。在新的Java 8中,日期和时间被明确划分为LocalDate和LocalTime,LocalDate无法包含时间,LocalTime无法包含日期。当然,LocalDateTime才能同时包含日期和时间。
新接口更好用的原因是考虑到了日期时间的操作,经常发生往前推或往后推几天的情况。用java.util.Date配合Calendar要写好多代码,而且一般的开发人员还不一定能写对。
LocalDate
看看新的LocalDate怎么用:
// 取当前日期:
LocalDate today = LocalDate.now(); // -》 2014-12-24
// 根据年月日取日期,12月就是12:
LocalDate crischristmas = LocalDate.of(2014, 12, 25); // -》 2014-12-25
// 根据字符串取:
LocalDate endOfFeb = LocalDate.parse(“2014-02-28“); // 严格按照ISO yyyy-MM-dd验证,02写成2都不行,当然也有一个重载方法允许自己定义格式
LocalDate.parse(“2014-02-29“); // 无效日期无法通过:DateTimeParseException: Invalid date
日期转换经常遇到,比如:
// 取本月第1天:
LocalDate firstDayOfThisMonth = today.with(TemporalAdjusters.firstDayOfMonth()); // 2014-12-01
// 取本月第2天:
LocalDate secondDayOfThisMonth = today.withDayOfMonth(2); // 2014-12-02
// 取本月最后一天,再也不用计算是28,29,30还是31:
LocalDate lastDayOfThisMonth = today.with(TemporalAdjusters.lastDayOfMonth()); // 2014-12-31
// 取下一天:
LocalDate firstDayOf2015 = lastDayOfThisMonth.plusDays(1); // 变成了2015-01-01
// 取2015年1月第一个周一,这个计算用Calendar要死掉很多脑细胞:
LocalDate firstMondayOf2015 = LocalDate.parse(“2015-01-01“).with(TemporalAdjusters.firstInMonth(DayOfWeek.MONDAY)); // 2015-01-05
LocalTime
LocalTime只包含时间,以前用java.util.Date怎么才能只表示时间呢?答案是,假装忽略日期。
LocalTime包含毫秒:
LocalTime now = LocalTime.now(); // 11:09:09.240
你可能想清除毫秒数:
LocalTime now = LocalTime.now().withNano(0)); // 11:09:09
构造时间也很简单:
LocalTime zero = LocalTime.of(0, 0, 0); // 00:00:00
LocalTime mid = LocalTime.parse(“12:00:00“); // 12:00:00
时间也是按照ISO格式识别,但可以识别以下3种格式:
12:00
12:01:02
12:01:02.345
JDBC
最新JDBC映射将把数据库的日期类型和Java 8的新类型关联起来:
SQL -》 Java
--------------------------
date -》 LocalDate
time -》 LocalTime
timestamp -》 LocalDateTime
再也不会出现映射到java.util.Date其中日期或时间某些部分为0的情况了。

程序包java.time不存在怎么解决


  jdk版本太低,java.time.LocalTime是java 8新出的包。如果版本太低的话,比如jdk1.6,当然java.time不存在。换一个好版本的jdk!!
为什么需要新的日期/日期API?
  在开始研究Java 8日期/时间API之前,让我们先来看一下为什么我们需要这样一个新的API。在Java中,现有的与日期和时间相关的类存在诸多问题,其中有:
  Java的日期/时间类的定义并不一致,在java.util和java.sql的包中都有日期类,此外用于格式化和解析的类在java.text包中定义。
  java.util.Date同时包含日期和时间,而java.sql.Date仅包含日期,将其纳入java.sql包并不合理。另外这两个类都有相同的名字,这本身就是一个非常糟糕的设计。
  对于时间、时间戳、格式化以及解析,并没有一些明确定义的类。对于格式化和解析的需求,我们有java.text.DateFormat抽象类,但通常情况下,SimpleDateFormat类被用于此类需求。
  所有的日期类都是可变的,因此他们都不是线程安全的,这是Java日期类最大的问题之一。
  日期类并不提供国际化,没有时区支持,因此Java引入了java.util.Calendar和java.util.TimeZone类,但他们同样存在上述所有的问题。
  在现有的日期和日历类中定义的方法还存在一些其他的问题,但以上问题已经很清晰地表明:Java需要一个健壮的日期/时间类。这也是为什么Joda Time在Java日期/时间需求中扮演了高质量替换的重要角色。
  Java 8日期/时间API
  Java 8日期/时间API是JSR-310的实现,它的实现目标是克服旧的日期时间实现中所有的缺陷,新的日期/时间API的一些设计原则是:
  不变性:新的日期/时间API中,所有的类都是不可变的,这对多线程环境有好处。
  关注点分离:新的API将人可读的日期时间和机器时间(unix timestamp)明确分离,它为日期(Date)、时间(Time)、日期时间(DateTime)、时间戳(unix timestamp)以及时区定义了不同的类。
  清晰:在所有的类中,方法都被明确定义用以完成相同的行为。举个例子,要拿到当前实例我们可以使用now()方法,在所有的类中都定义了format()和parse()方法,而不是像以前那样专门有一个独立的类。为了更好的处理问题,所有的类都使用了工厂模式和策略模式,一旦你使用了其中某个类的方法,与其他类协同工作并不困难。
  实用操作:所有新的日期/时间API类都实现了一系列方法用以完成通用的任务,如:加、减、格式化、解析、从日期/时间中提取单独部分,等等。
  可扩展性:新的日期/时间API是工作在ISO-8601日历系统上的,但我们也可以将其应用在非IOS的日历上。
  
Java 8 日期/时间API包:
  java.time包:这是新的Java日期/时间API的基础包,所有的主要基础类都是这个包的一部分,如:LocalDate, LocalTime, LocalDateTime, Instant, Period, Duration等等。所有这些类都是不可变的和线程安全的,在绝大多数情况下,这些类能够有效地处理一些公共的需求。
  java.time.chrono包:这个包为非ISO的日历系统定义了一些泛化的API,我们可以扩展AbstractChronology类来创建自己的日历系统。
  java.time.format包:这个包包含能够格式化和解析日期时间对象的类,在绝大多数情况下,我们不应该直接使用它们,因为java.time包中相应的类已经提供了格式化和解析的方法。
  java.time.temporal包:这个包包含一些时态对象,我们可以用其找出关于日期/时间对象的某个特定日期或时间,比如说,可以找到某月的第一天或最后一天。你可以非常容易地认出这些方法,因为它们都具有“withXXX”的格式。
  java.time.zone包:这个包包含支持不同时区以及相关规则的类。

java里怎么自定义一个日期!


在旧版本的jdk中,Java的日期/时间类的定义并不一致,在java.util和java.sql的包中都有日期类,此外用于格式化和解析的类在java.text.SimpleDateFormat包中定义。

java.util.Date是在除了SQL语句的情况下面使用的。
java.sql.Date是针对SQL语句使用的,它只包含日期而没有时间部分

它们都有getTime方法返回毫秒数,自然就可以直接构建。 java.util.Date 是 java.sql.Date 的父类,前者是常用的表示时间的类,我们通常格式化或者得到当前时间都是用他,后者之后在读写数据库的时候用他,因为PreparedStament的setDate()的第2参数和ResultSet的getDate()方法的第2个参数都是java.sql.Date。

java.sql.Date转为java.util.Date
java.sql.Date date=new java.sql.Date();
java.util.Date d=new java.util.Date (date.getTime());

java.util.Date转为java.sql.Date
java.util.Date utilDate=new Date();
java.sql.Date sqlDate=new java.sql.Date(utilDate.getTime());
java.util.Date utilDate=new Date();
java.sql.Date sqlDate=new java.sql.Date(utilDate.getTime());
java.sql.Time sTime=new java.sql.Time(utilDate.getTime());
java.sql.Timestamp stp=new java.sql.Timestamp(utilDate.getTime());

这里所有时间日期都可以被SimpleDateFormat格式化format()
SimpleDateFormat f=new SimpleDateFormat(“yyyy-MM-dd hh:mm:ss“);
f.format(stp);
f.format(sTime);
f.format(sqlDate);
f.format(utilDate)
java.sql.Date sqlDate=java.sql.Date.valueOf(“ 2005-12-12“);
utilDate=new java.util.Date(sqlDate.getTime());

另类取得年月日的方法:
import java.text.SimpleDateFormat;
import java.util.*;
java.util.Date date = new java.util.Date();
//如果希望得到YYYYMMDD的格式SimpleDateFormat
sy1=new SimpleDateFormat(“yyyyMMDD“);
String dateFormat=sy1.format(date);
//如果希望分开得到年,月,日SimpleDateFormat
sy=new SimpleDateFormat(“yyyy“);
SimpleDateFormat sm=new SimpleDateFormat(“MM“);
SimpleDateFormat sd=new SimpleDateFormat(“dd“);
String syear=sy.format(date);
String smon=sm.format(date);
String sday=sd.format(date);

注意啦!!!在JAVA 8中有了日期的新的表示方式。在java.time包中。

     Java 8日期/时间( Date/Time)API是开发人员最受追捧的变化之一,Java从一开始就没有对日期时间处理的一致性方法,因此日期/时间API也是除Java核心API以外另一项倍受欢迎的内容。

为什么我们需要新的Java日期/时间API?

      在开始研究Java 8日期/时间API之前,让我们先来看一下为什么我们需要这样一个新的API。在Java中,现有的与日期和时间相关的类存在诸多问题,其中有:

1. Java的日期/时间类的定义并不一致,在java.util和java.sql的包中都有日期类,此外用于格式化和解析的类在java.text包中定义。

2. java.util.Date同时包含日期和时间,而java.sql.Date仅包含日期,将其纳入java.sql包并不合理。另外这两个类都有相同的名字,这本身就是一个非常糟糕的设计。

3. 对于时间、时间戳、格式化以及解析,并没有一些明确定义的类。对于格式化和解析的需求,我们有java.text.DateFormat抽象类,但通常情况下,SimpleDateFormat类被用于此类需求。

4. 所有的日期类都是可变的,因此他们都不是线程安全的,这是Java日期类最大的问题之一。

5. 日期类并不提供国际化,没有时区支持,因此Java引入了java.util.Calendar java.util.TimeZone类,但他们同样存在上述所有的问题。

       在现有的日期和日历类中定义的方法还存在一些其他的问题,但以上问题已经很清晰地表明:Java需要一个健壮的日期/时间类。这也是为什么Joda Time在Java日期/时间需求中扮演了高质量替换的重要角色。

Java 8日期/时间API是JSR-310的实现,它的实现目标是克服旧的日期时间实现中所有的缺陷,新的日期/时间API的一些设计原则是:

1. 不变性:新的日期/时间API中,所有的类都是不可变的,这对多线程环境有好处。

2. 关注点分离:新的API将人可读的日期时间和机器时间(unix timestamp)明确分离,它为日期(Date)、时间(Time)、日期时间(DateTime)、时间戳(unix timestamp)以及时区定义了不同的类。

3. 清晰:在所有的类中,方法都被明确定义用以完成相同的行为。举个例子,要拿到当前实例我们可以使用now()方法,在所有的类中都定义了format()和parse()方法,而不是像以前那样专门有一个独立的类。为了更好的处理问题,所有的类都使用了工厂模式和策略模式,一旦你使用了其中某个类的方法,与其他类协同工作并不困难。

4. 实用操作:所有新的日期/时间API类都实现了一系列方法用以完成通用的任务,如:加、减、格式化、解析、从日期/时间中提取单独部分,等等。

5. 可扩展性:新的日期/时间API是工作在ISO-8601日历系统上的,但我们也可以将其应用在非IOS的日历上。

Java日期/时间API包含以下相应的包。

1. java.time包:这是新的Java日期/时间API的基础包,所有的主要基础类都是这个包的一部分,如:LocalDate, LocalTime, LocalDateTime, Instant, Period, Duration等等。所有这些类都是不可变的和线程安全的,在绝大多数情况下,这些类能够有效地处理一些公共的需求。

2. java.time.chrono包:这个包为非ISO的日历系统定义了一些泛化的API,我们可以扩展AbstractChronology类来创建自己的日历系统。

3. java.time.format包:这个包包含能够格式化和解析日期时间对象的类,在绝大多数情况下,我们不应该直接使用它们,因为java.time包中相应的类已经提供了格式化和解析的方法。

4. java.time.temporal包:这个包包含一些时态对象,我们可以用其找出关于日期/时间对象的某个特定日期或时间,比如说,可以找到某月的第一天或最后一天。你可以非常容易地认出这些方法,因为它们都具有“withXXX”的格式。

5. java.time.zone包:这个包包含支持不同时区以及相关规则的类。

新旧API的对比图:

新API的示例代码:

public class TimeIntroduction {
    public static void testClock() throws InterruptedException {
        //时钟提供给我们用于访问某个特定 时区的 瞬时时间、日期 和 时间的。  
        Clock c1 = Clock.systemUTC(); //系统默认UTC时钟(当前瞬时时间 System.currentTimeMillis())  
        System.out.println(c1.millis()); //每次调用将返回当前瞬时时间(UTC)  
        Clock c2 = Clock.systemDefaultZone(); //系统默认时区时钟(当前瞬时时间)  
        Clock c31 = Clock.system(ZoneId.of(“Europe/Paris“)); //巴黎时区  
        System.out.println(c31.millis()); //每次调用将返回当前瞬时时间(UTC)  
        Clock c32 = Clock.system(ZoneId.of(“Asia/Shanghai“));//上海时区  
        System.out.println(c32.millis());//每次调用将返回当前瞬时时间(UTC)  
        Clock c4 = Clock.fixed(Instant.now(), ZoneId.of(“Asia/Shanghai“));//固定上海时区时钟  
        System.out.println(c4.millis());
        Thread.sleep(1000);
        System.out.println(c4.millis()); //不变 即时钟时钟在那一个点不动  
        Clock c5 = Clock.offset(c1, Duration.ofSeconds(2)); //相对于系统默认时钟两秒的时钟  
        System.out.println(c1.millis());
        System.out.println(c5.millis());
    }
    public static void testInstant() {
        //瞬时时间 相当于以前的System.currentTimeMillis()  
        Instant instant1 = Instant.now();
        System.out.println(instant1.getEpochSecond());//精确到秒 得到相对于1970-01-01 00:00:00 UTC的一个时间  
        System.out.println(instant1.toEpochMilli()); //精确到毫秒  
        Clock clock1 = Clock.systemUTC(); //获取系统UTC默认时钟  
        Instant instant2 = Instant.now(clock1);//得到时钟的瞬时时间  
        System.out.println(instant2.toEpochMilli());
        Clock clock2 = Clock.fixed(instant1, ZoneId.systemDefault()); //固定瞬时时间时钟  
        Instant instant3 = Instant.now(clock2);//得到时钟的瞬时时间  
        System.out.println(instant3.toEpochMilli());//equals instant1  
    }
    public static void testLocalDateTime() {
        //使用默认时区时钟瞬时时间创建 Clock.systemDefaultZone() --》即相对于 ZoneId.systemDefault()默认时区  
        LocalDateTime now = LocalDateTime.now();
        System.out.println(now);
//自定义时区  
        LocalDateTime now2 = LocalDateTime.now(ZoneId.of(“Europe/Paris“));
        System.out.println(now2);//会以相应的时区显示日期  
//自定义时钟  
        Clock clock = Clock.system(ZoneId.of(“Asia/Dhaka“));
        LocalDateTime now3 = LocalDateTime.now(clock);
        System.out.println(now3);//会以相应的时区显示日期  
//不需要写什么相对时间 如java.util.Date 年是相对于1900 月是从0开始  
//2013-12-31 23:59  
        LocalDateTime d1 = LocalDateTime.of(2013, 12, 31, 23, 59);
//年月日 时分秒 纳秒  
        LocalDateTime d2 = LocalDateTime.of(2013, 12, 31, 23, 59, 59, 11);
//使用瞬时时间 + 时区  
        Instant instant = Instant.now();
        LocalDateTime d3 = LocalDateTime.ofInstant(Instant.now(), ZoneId.systemDefault());
        System.out.println(d3);
//解析String---》LocalDateTime  
        LocalDateTime d4 = LocalDateTime.parse(“2013-12-31T23:59“);
        System.out.println(d4);
        LocalDateTime d5 = LocalDateTime.parse(“2013-12-31T23:59:59.999“);//999毫秒 等价于999000000纳秒  
        System.out.println(d5);
//使用DateTimeFormatter API 解析 和 格式化  
        DateTimeFormatter formatter = DateTimeFormatter.ofPattern(“yyyy/MM/dd HH:mm:ss“);
        LocalDateTime d6 = LocalDateTime.parse(“2013/12/31 23:59:59“, formatter);
        System.out.println(formatter.format(d6));
//时间获取  
        System.out.println(d6.getYear());
        System.out.println(d6.getMonth());
        System.out.println(d6.getDayOfYear());
        System.out.println(d6.getDayOfMonth());
        System.out.println(d6.getDayOfWeek());
        System.out.println(d6.getHour());
        System.out.println(d6.getMinute());
        System.out.println(d6.getSecond());
        System.out.println(d6.getNano());
//时间增减  
        LocalDateTime d7 = d6.minusDays(1);
        LocalDateTime d8 = d7.plus(1, IsoFields.QUARTER_YEARS);
//LocalDate 即年月日 无时分秒  
//LocalTime即时分秒 无年月日  
//API和LocalDateTime类似就不演示了  
    }
    public static void testZonedDateTime() {
        //即带有时区的date-time 存储纳秒、时区和时差(避免与本地date-time歧义)。  
//API和LocalDateTime类似,只是多了时差(如2013-12-20T10:35:50.711+08:00[Asia/Shanghai])  
        ZonedDateTime now = ZonedDateTime.now();
        System.out.println(now);
        ZonedDateTime now2 = ZonedDateTime.now(ZoneId.of(“Europe/Paris“));
        System.out.println(now2);
//其他的用法也是类似的 就不介绍了  
        ZonedDateTime z1 = ZonedDateTime.parse(“2013-12-31T23:59:59Z[Europe/Paris]“);
        System.out.println(z1);
    }
    public static void testDuration() {
        //表示两个瞬时时间的时间段  
        Duration d1 = Duration.between(Instant.ofEpochMilli(System.currentTimeMillis() - 12323123), Instant.now());
//得到相应的时差  
        System.out.println(d1.toDays());
        System.out.println(d1.toHours());
        System.out.println(d1.toMinutes());
        System.out.println(d1.toMillis());
        System.out.println(d1.toNanos());
//1天时差 类似的还有如ofHours()  
        Duration d2 = Duration.ofDays(1);
        System.out.println(d2.toDays());
    }
    public static void testChronology() {
        //提供对java.util.Calendar的替换,提供对年历系统的支持  
        Chronology c = HijrahChronology.INSTANCE;
        ChronoLocalDateTime d = c.localDateTime(LocalDateTime.now());
        System.out.println(d);
    }
    /**
     * 新旧日期转换
     */
    public static void testNewOldDateConversion(){
        Instant instant=new Date().toInstant();
        Date date=Date.from(instant);
        System.out.println(instant);
        System.out.println(date);
    }
    public static void main(String args) throws InterruptedException {
        testClock();
        testInstant();
        testLocalDateTime();
        testZonedDateTime();
        testDuration();
        testChronology();
        testNewOldDateConversion();
    }
}

为什么localtime线程不安全


从localtime函数的原型定义分析
struct tm *localtime(const time_t *timep);
localtime函数返回了一个struct tm指针,函数的说明中并未要求调用者将该指针free掉,证明tm对象并非是malloc或new出来的。那么tm一定是个全局变量或者是localtime函数中的局部静态变量。
这样的变量在多线程访问时会出问题,第一个线程在localtime返回之前首先改变了这个变量的值,第二个线程又进来改变这个变量,以此类推。这样第一个线程在真正返回时该值已经被多次改变,这是典型的函数重入问题。
因此localtime函数线程不安全。

localtime是非线程安全的(如何在Java 8中愉快地处理日期和时间)

本文编辑:admin

更多文章:


win7显卡驱动安装失败(win7系统安装显卡驱动失败怎么弄)

win7显卡驱动安装失败(win7系统安装显卡驱动失败怎么弄)

其实win7显卡驱动安装失败的问题并不复杂,但是又很多的朋友都不太了解win7系统安装显卡驱动失败怎么弄,因此呢,今天小编就来为大家分享win7显卡驱动安装失败的一些知识,希望可以帮助到大家,下面我们一起来看看这个问题的分析吧!本文目录wi

2024年7月13日 07:42

手机号自带的邮箱(手机邮箱是什么格式)

手机号自带的邮箱(手机邮箱是什么格式)

大家好,关于手机号自带的邮箱很多朋友都还不太明白,不过没关系,因为今天小编就来为大家分享关于手机邮箱是什么格式的知识点,相信应该可以解决大家的一些困惑和问题,如果碰巧可以解决您的问题,还望关注下本站哦,希望对各位有所帮助!本文目录手机邮箱是

2024年5月9日 14:40

windows+i(windows快捷键)

windows+i(windows快捷键)

本篇文章给大家谈谈windows+i,以及windows快捷键对应的知识点,文章可能有点长,但是希望大家可以阅读完,增长自己的知识,最重要的是希望对各位有所帮助,可以解决了您的问题,不要忘了收藏本站喔。本文目录windows快捷键Windo

2024年4月12日 20:30

西游大战僵尸3(西游大战僵尸的人物技能)

西游大战僵尸3(西游大战僵尸的人物技能)

本文目录西游大战僵尸的人物技能西游大战僵尸怎么幻化西游大战僵尸小游戏为什么到了20级后就不能再升级西游大战僵尸里的冰晶凤凰如何达到最强形态植物大战僵尸西游版3大英雄下载西游大战僵尸ce修改器使用方法植物大战僵尸的西游版攻略西游大战僵尸的人物

2024年7月6日 11:33

acdsee9 0免费下载(哪里可以免费下到ACDSee)

acdsee9 0免费下载(哪里可以免费下到ACDSee)

本文目录哪里可以免费下到ACDSeeACDSee中文的下载网址哪里可以免费下到ACDSeeACDSee v9.0 Build 50 官方简体中文注册特别版 http://www.quick8.cn/smallSoftDown/acdsee9

2024年7月1日 13:12

java是什么意思(java是什么意思)

java是什么意思(java是什么意思)

大家好,关于java是什么意思很多朋友都还不太明白,不过没关系,因为今天小编就来为大家分享关于java是什么意思的知识点,相信应该可以解决大家的一些困惑和问题,如果碰巧可以解决您的问题,还望关注下本站哦,希望对各位有所帮助!本文目录java

2024年7月14日 02:11

免费领皮肤的软件下载(免费领游戏皮肤的软件有哪些)

免费领皮肤的软件下载(免费领游戏皮肤的软件有哪些)

大家好,免费领皮肤的软件下载相信很多的网友都不是很明白,包括免费领游戏皮肤的软件有哪些也是一样,不过没有关系,接下来就来为大家分享关于免费领皮肤的软件下载和免费领游戏皮肤的软件有哪些的一些知识点,大家可以关注收藏,免得下次来找不到哦,下面我

2024年7月20日 19:50

dnf男法师用什么武器(DNF男魔法师.用什么武器好)

dnf男法师用什么武器(DNF男魔法师.用什么武器好)

本文目录DNF男魔法师.用什么武器好DNF男法师武器魔杖好还是法杖好请问DNF男法师用什么武器好、dnf男法师转职是冰洁师好还是爆破元素好啊分别都用什么武器好呢DNF男魔法师.用什么武器好首先看你是什么职业,如果是冰洁师就用魔杖,元素爆破就

2024年3月17日 05:50

春雨直播在线下载(春雨医生怎么看直播 春雨直播查看方法)

春雨直播在线下载(春雨医生怎么看直播 春雨直播查看方法)

大家好,关于春雨直播在线下载很多朋友都还不太明白,不过没关系,因为今天小编就来为大家分享关于春雨医生怎么看直播 春雨直播查看方法的知识点,相信应该可以解决大家的一些困惑和问题,如果碰巧可以解决您的问题,还望关注下本站哦,希望对各位有所帮助!

2024年7月17日 21:08

怎么注册趣头条自媒体账号?怎么入驻趣头条自媒体赚钱

怎么注册趣头条自媒体账号?怎么入驻趣头条自媒体赚钱

大家好,趣头条官网相信很多的网友都不是很明白,包括怎么注册趣头条自媒体账号也是一样,不过没有关系,接下来就来为大家分享关于趣头条官网和怎么注册趣头条自媒体账号的一些知识点,大家可以关注收藏,免得下次来找不到哦,下面我们开始吧!本文目录怎么注

2024年5月24日 00:46

可以搜索的输入法(手机什么输入法好用)

可以搜索的输入法(手机什么输入法好用)

其实可以搜索的输入法的问题并不复杂,但是又很多的朋友都不太了解手机什么输入法好用,因此呢,今天小编就来为大家分享可以搜索的输入法的一些知识,希望可以帮助到大家,下面我们一起来看看这个问题的分析吧!本文目录手机什么输入法好用什么手机输入法自带

2024年6月28日 13:42

电脑开不开机了怎么解决(电脑开不开机应该怎么办)

电脑开不开机了怎么解决(电脑开不开机应该怎么办)

其实电脑开不开机了怎么解决的问题并不复杂,但是又很多的朋友都不太了解电脑开不开机应该怎么办,因此呢,今天小编就来为大家分享电脑开不开机了怎么解决的一些知识,希望可以帮助到大家,下面我们一起来看看这个问题的分析吧!本文目录电脑开不开机应该怎么

2024年9月5日 00:35

手机说话直接变声的软件(手机变声器软件哪个好用)

手机说话直接变声的软件(手机变声器软件哪个好用)

大家好,关于手机说话直接变声的软件很多朋友都还不太明白,不过没关系,因为今天小编就来为大家分享关于手机变声器软件哪个好用的知识点,相信应该可以解决大家的一些困惑和问题,如果碰巧可以解决您的问题,还望关注下本站哦,希望对各位有所帮助!本文目录

2024年5月20日 20:52

胚胎6周停育,胚胎染色体查出是92XXYY,请问反映出什么问题?STAC 92xx @ 英特尔82801I(ICH9) 这个声卡驱动给我发个呗 急需求 谢谢了~~

胚胎6周停育,胚胎染色体查出是92XXYY,请问反映出什么问题?STAC 92xx @ 英特尔82801I(ICH9) 这个声卡驱动给我发个呗 急需求 谢谢了~~

本文目录胚胎6周停育,胚胎染色体查出是92XXYY,请问反映出什么问题STAC 92xx @ 英特尔82801I(ICH9) 这个声卡驱动给我发个呗 急需求 谢谢了~~IDT SigmaTel STAC 92XX HD Audio CODE

2024年5月7日 22:16

腾讯新闻直播(腾讯新闻直播节,资讯直播的星星之火能否燎原)

腾讯新闻直播(腾讯新闻直播节,资讯直播的星星之火能否燎原)

本文目录腾讯新闻直播节,资讯直播的星星之火能否燎原腾讯新闻的直播间是怎么开的,个人可以申请吗腾讯新闻直播节,资讯直播的星星之火能否燎原2016年随着智能手机配置的升级以及服务器云计算能力的发展,在线视频直播迎来了最大的风口,而受到用户需求驱

2024年1月24日 22:40

腾讯最新网页游戏(腾讯又出什么新游戏了,是什么)

腾讯最新网页游戏(腾讯又出什么新游戏了,是什么)

本文目录腾讯又出什么新游戏了,是什么谁知道腾讯新出的角色扮演类 网页游戏要新的 越新越好腾讯又出什么新游戏了,是什么腾讯09年新游戏:《丝路英雄》网页游戏、目前已经公测、《战地之王》网络游戏、目前还在内测、以后还会出飞行岛和飞车...估计得

2024年3月17日 12:25

喜欢蕾丝是一种怎样的体验?学明星穿衣搭配,如何打造完美的蕾丝风

喜欢蕾丝是一种怎样的体验?学明星穿衣搭配,如何打造完美的蕾丝风

本文目录喜欢蕾丝是一种怎样的体验学明星穿衣搭配,如何打造完美的蕾丝风蕾丝款式如何搭配蕾丝布料做什么衣服好承载着宅男少女们终极幻想的蕾丝,在服装设计中该如何巧妙运用如何找到个穿透明蕾丝衣性感的视频美女秀 扣扣蕾丝视频缓存文件在哪蕾丝视频5.3

2024年3月1日 23:20

中国十大基金公司排名(中国十大优秀基金公司)

中国十大基金公司排名(中国十大优秀基金公司)

各位老铁们好,相信很多人对中国十大基金公司排名都不是特别的了解,因此呢,今天就来为大家分享下关于中国十大基金公司排名以及中国十大优秀基金公司的问题知识,还望可以帮助大家,解决大家的一些困惑,下面一起来看看吧!本文目录中国十大优秀基金公司私募

2024年6月17日 00:28

手机学车模拟开车软件(考驾照下载什么软件好)

手机学车模拟开车软件(考驾照下载什么软件好)

大家好,关于手机学车模拟开车软件很多朋友都还不太明白,不过没关系,因为今天小编就来为大家分享关于考驾照下载什么软件好的知识点,相信应该可以解决大家的一些困惑和问题,如果碰巧可以解决您的问题,还望关注下本站哦,希望对各位有所帮助!本文目录考驾

2024年5月19日 12:15

foxmail邮箱(@foxmail.com这种邮箱是什么怎么注册)

foxmail邮箱(@foxmail.com这种邮箱是什么怎么注册)

大家好,今天小编来为大家解答以下的问题,关于foxmail邮箱,@foxmail.com这种邮箱是什么怎么注册这个很多人还不知道,现在让我们一起来看看吧!本文目录@foxmail.com这种邮箱是什么怎么注册foxmail局域网内怎么设置邮

2024年5月27日 14:15

近期文章

本站热文

iphone vpn设置(ios设置vpn快捷开关)
2024-07-22 15:01:12 浏览:2334
windows12正式版下载(操作系统Windows Server 2012 R2,在哪能下载到,公司用的)
2024-07-20 17:26:53 浏览:1730
java安装教程(win10如何安装JAVA)
2024-07-19 19:55:49 浏览:1155
client mfc application未响应(每次进cf就提示client MFC Application未响应该怎么办啊!急急急)
2024-07-20 11:15:58 浏览:1152
标签列表

热门搜索