JDK1.8的几大新特性

一、接口的默认方法与静态方法,也就是接口中可以有实现方法

复制代码
 1 public class Test {
 2     public static void main(String[] args) {
 3         Formula a=new For();
 4         a.calculate(1);
 5         System.out.println(a.sqrt(8));
 6 
 7 
 8     }
 9     interface Formula {
10         double calculate(int a);
11         default double sqrt(int a) {//这个方法可以在实现类重写,或者直接使用
12             return Math.sqrt(a);
13         }
14     }
15     static class For implements Formula{
16         @Override
17         public double calculate(int a) {
18             return 0;
19         }
20 
21         @Override
22         public double sqrt(int a) {
23             return 0;
24         }
25     }
26 }
复制代码

 

复制代码
 1 package com.ming.jdk18;
 2 
 3 
 4 /**
 5  * 
 6  * @author mingge
 7  * jdk1.8接口支持静态方法与默认方法
 8  *
 9  */
10 public interface TestInterface {
11 
12     //这个是默认方法
13     default String get(String aa,String bb){
14         System.out.println("我是jdk1.8默认实现方法...");
15         return "";
16     }
17     
18     String aa="2222";
19     
20     //这个是静态方法
21     static void staticmethod(){System.out.println("我是静态方法"+aa);}
22 }
复制代码

 

复制代码
 1 package com.ming.jdk18;
 2 
 3 
 4 public class Main {
 5 
 6     
 7     public static void main(String[] args) {
 8         TestInterface.staticmethod();
 9     }
10     
11     
12 }
复制代码

 

 

 

 

以前的版本定义接口是不能有实现机制的,现在这样用了一个default关键字后,就可以实现,然后子类可以重写,也可以直接使用了。好处多多,感觉有点抽象类了...越来越灵活了。加入静态方法后,你以后的工具类就可以参考接口来设计啦.这是一个优点啦.

 

二、Lambda 表达式

复制代码
 1 public class Test {
 2     public static void main(String[] args) {
 3         List names = Arrays.asList("peter", "anna", "mike", "xenia");
 4         Collections.sort(names, new Comparator() {//java以前老版本的写法
 5             @Override
 6             public int compare(String a, String b) {
 7                 return b.compareTo(a);
 8             }
 9         });
10         for(String name:names){
11             System.out.println(name);
12         }
13     }
14 
15 }
复制代码
复制代码
 1 只需要给静态方法 Collections.sort 传入一个List对象以及一个比较器来按指定顺序排列。通常做法都是创建一个匿名的比较器对象然后将其传递给sort方法。
 2 在Java 8 中你就没必要使用这种传统的匿名对象的方式了,Java 8提供了更简洁的语法,lambda表达式:
 3 复制代码 代码如下:
 4 
 5 Collections.sort(names, (String a, String b) -> {
 6     return b.compareTo(a);
 7 });
 8 
 9 看到了吧,代码变得更段且更具有可读性,但是实际上还可以写得更短:
10 复制代码 代码如下:
11 
12 Collections.sort(names, (String a, String b) -> b.compareTo(a));
13 
14 对于函数体只有一行代码的,你可以去掉大括号{}以及return关键字,但是你还可以写得更短点:
15 复制代码 代码如下:
16 
17 Collections.sort(names, (a, b) -> b.compareTo(a));
18 
19 Java编译器可以自动推导出参数类型,所以你可以不用再写一次类型。接下来我们看看lambda表达式还能作出什么更方便的东西来:
复制代码

lambda表达式的使用简化了代码。

 

三、函数式接口与静态导入

 

Lambda表达式是如何在java的类型系统中表示的呢?每一个lambda表达式都对应一个类型,通常是接口类型。而“函数式接口”是指仅仅只包含一个抽象方法的接口,每一个该类型的lambda表达式都会被匹配到这个抽象方法。

 

因为默认方法不算抽象方法,所以你也可以给你的函数式接口添加默认方法。

 

我们可以将lambda表达式当作任意只包含一个抽象方法的接口类型,确保你的接口一定达到这个要求,你只需要给你的接口添加 @FunctionalInterface 注解,编译器如果发现你标注了这个注解的接口有多于一个抽象方法的时候会报错的。

 

复制代码
 1 public class Test {
 2     public static void main(String[] args) {
 3         Converter a = (from) -> Integer.valueOf(from);
 4         
 5         //Java 8 允许你使用 :: 关键字来传递方法或者构造函数引用
 6         Converter converter = Integer::valueOf;//这个是静态方式导入 
 7 
 8         Integer m=a.convert("123");
 9         System.out.println(m);
10         test test=mmm -> "aaaa";//mmm代表你要传入的参数
11         String testResult=test.aa("");//aa代表你要传入的方法
12         System.out.println(testResult);
13     }
14 
15     @FunctionalInterface
16     interface Converter {
17         T convert(F from);
18     }
19 
20     interface test{
21         String aa(String mmm);
22     }
23 
24     //使用函数式接口时,接口的定义只能有一个,@FunctionalInterface注解可有可无
25 }
复制代码

 

四、Lambda 作用域

在lambda表达式中访问外层作用域和老版本的匿名对象中的方式很相似。你可以直接访问标记了final的外层局部变量,或者实例的字段以及静态变量。

 

五、访问局部变量

复制代码
 1 public class Test {
 2     public static void main(String[] args) {
 3         final int num = 1;
 4         Converter stringConverter =
 5                 (from) -> String.valueOf(from + num);
 6         String str=stringConverter.convert(2);
 7         System.out.println(str);//输出3
 8         //final关键字可以去掉,但是默认是存在的,所以,对num变量不能再次赋值修改
 9     }
10 
11     @FunctionalInterface
12     interface Converter {
13         T convert(F from);
14     }
15 
16 }
复制代码

 

六、访问对象字段与静态变量

复制代码
 1 public class Test {
 2     public static void main(String[] args) {
 3         Lambda4 lambda4=new Lambda4();
 4         lambda4.testScopes();
 5     }
 6 
 7     @FunctionalInterface
 8     interface Converter {
 9         T convert(F from);
10     }
11 
12     static class Lambda4 {
13         static int outerStaticNum=0;
14         int outerNum=0;
15         void testScopes() {
16             Converter stringConverter1 = (from) -> {
17                 outerNum = 23;
18                 return String.valueOf(outerNum);
19             };
20             String a=stringConverter1.convert(outerNum);
21             System.out.println(a);//输出23
22             Converter stringConverter2 = (from) -> {
23                 outerStaticNum = 72;
24                 return String.valueOf(outerStaticNum);
25             };
26 
27         }
28     }
29     
30     //这里的意思是outerStaticNum与outerNum两个参数可以多次赋值
31     
32 }
复制代码

七、Date API

Java 8 在包java.time下包含了一组全新的时间日期API。新的日期API和开源的Joda-Time库差不多,但又不完全一样,下面的例子展示了这组新API里最重要的一些部分:

复制代码
  1 public class Test {
  2     public static void main(String[] args) {
  3         /**
  4          *(一) Clock 时钟
  5          * Clock类提供了访问当前日期和时间的方法,Clock是时区敏感的,可以用来取代 System.currentTimeMillis() 来获取当前的微秒数。
  6          * 某一个特定的时间点也可以使用Instant类来表示,Instant类也可以用来创建老的java.util.Date对象。
  7          */
  8         Clock clock = Clock.systemDefaultZone();
  9         long millis = clock.millis();
 10         System.out.println(millis+":"+System.currentTimeMillis());
 11         Instant instant = clock.instant();
 12         Date legacyDate = Date.from(instant);   // legacy java.util.Date
 13         System.out.println(legacyDate);
 14 
 15         /**
 16          *(二)Timezones 时区
 17          * 在新API中时区使用ZoneId来表示。时区可以很方便的使用静态方法of来获取到。
 18          * 时区定义了到UTS时间的时间差,在Instant时间点对象到本地日期对象之间转换的时候是极其重要的。
 19          */
 20         System.out.println(ZoneId.getAvailableZoneIds());//输出所有的时区
 21         ZoneId zone1 = ZoneId.of("Europe/Berlin");
 22         ZoneId zone2 = ZoneId.of("Brazil/East");
 23         System.out.println(zone1.getRules());//输出ZoneRules[currentStandardOffset=+01:00]
 24         System.out.println(zone2.getRules());//输出ZoneRules[currentStandardOffset=-03:00]
 25 
 26         /**
 27          *(三)LocalTime 本地时间
 28          * LocalTime 定义了一个没有时区信息的时间,例如 晚上10点,或者 17:30:15。
 29          * 下面的例子使用前面代码创建的时区创建了两个本地时间。之后比较时间并以小时和分钟为单位计算两个时间的时间差:
 30          */
 31         LocalTime now1 = LocalTime.now(zone1);
 32         LocalTime now2 = LocalTime.now(zone2);
 33         System.out.println(now1.isBefore(now2));  // false
 34         long hoursBetween = ChronoUnit.HOURS.between(now1, now2);
 35         long minutesBetween = ChronoUnit.MINUTES.between(now1, now2);
 36         System.out.println(hoursBetween);       // -2
 37         System.out.println(minutesBetween);     // -179
 38 
 39         /**LocalTime 提供了多种工厂方法来简化对象的创建,包括解析时间字符串。
 40          *
 41          */
 42         LocalTime late = LocalTime.of(23, 59, 59);
 43         System.out.println(late);       // 23:59:59
 44         DateTimeFormatter germanFormatter =
 45                 DateTimeFormatter
 46                         .ofLocalizedTime(FormatStyle.SHORT)
 47                         .withLocale(Locale.GERMAN);
 48         LocalTime leetTime = LocalTime.parse("13:37", germanFormatter);
 49         System.out.println(leetTime);   // 13:37
 50 
 51         /**
 52          * (四)LocalDate 本地日期
 53          * LocalDate 表示了一个确切的日期,比如 2014-03-11。该对象值是不可变的,用起来和LocalTime基本一致。
 54          * 下面的例子展示了如何给Date对象加减天/月/年。另外要注意的是这些对象是不可变的,操作返回的总是一个新实例。
 55          */
 56         LocalDate today = LocalDate.now();
 57         LocalDate tomorrow = today.plus(1, ChronoUnit.DAYS);
 58         LocalDate yesterday = tomorrow.minusDays(2);
 59         System.out.println(yesterday);
 60         LocalDate independenceDay = LocalDate.of(2014, Month.JULY, 4);
 61         DayOfWeek dayOfWeek = independenceDay.getDayOfWeek();
 62         System.out.println(dayOfWeek);    // FRIDAY
 63 
 64         DateTimeFormatter germanFormatter1 =
 65                 DateTimeFormatter
 66                         .ofLocalizedDate(FormatStyle.MEDIUM)
 67                         .withLocale(Locale.GERMAN);
 68         LocalDate xmas = LocalDate.parse("24.12.2014", germanFormatter1);
 69         System.out.println(xmas);   // 2014-12-24
 70         //用这种方式比以前使用日期的要方便多了
 71 
 72         /**
 73          * (五)LocalDateTime 本地日期时间
 74          * LocalDateTime 同时表示了时间和日期,相当于前两节内容合并到一个对象上了。
 75          * LocalDateTime和LocalTime还有LocalDate一样,都是不可变的。LocalDateTime提供了一些能访问具体字段的方法。
 76          */
 77         LocalDateTime sylvester = LocalDateTime.of(2014, Month.DECEMBER, 31, 23, 59, 59);
 78         DayOfWeek dayOfWeek1 = sylvester.getDayOfWeek();
 79         System.out.println(dayOfWeek1);      // WEDNESDAY
 80         Month month = sylvester.getMonth();
 81         System.out.println(month);          // DECEMBER
 82         long minuteOfDay = sylvester.getLong(ChronoField.MINUTE_OF_DAY);
 83         System.out.println(minuteOfDay);    // 1439
 84         /**
 85          * 只要附加上时区信息,就可以将其转换为一个时间点Instant对象,Instant时间点对象可以很容易的转换为老式的java.util.Date。
 86          */
 87         Instant instant1 = sylvester
 88                 .atZone(ZoneId.systemDefault())
 89                 .toInstant();
 90         Date legacyDate1 = Date.from(instant1);
 91         System.out.println(legacyDate1);     // Wed Dec 31 23:59:59 CET 2014
 92 
 93         /**
 94          * 格式化LocalDateTime和格式化时间和日期一样的,除了使用预定义好的格式外,我们也可以自己定义格式:
 95          */
 96         DateTimeFormatter formatter =
 97                 DateTimeFormatter
 98                         .ofPattern("MMM dd, yyyy - HH:mm");
 99         LocalDateTime parsed = LocalDateTime.parse("Nov 03, 2014 - 07:13", formatter);
100         String string = formatter.format(parsed);
101         System.out.println(string);     // Nov 03, 2014 - 07:13
102 
103         //和java.text.NumberFormat不一样的是新版的DateTimeFormatter是不可变的,所以它是线程安全的。
104         //关于时间日期格式的详细信息:http://download.java.net/jdk8/docs/api/java/time/format/DateTimeFormatter.html
105         //java jdk1.8新提供的这个time在某些时候还是有好处的
106 
107 
108     }
109 
110 
111 
112 }

你可能感兴趣的:(Java)