拓展阅读

Java Functional java 函数式编程

Java Lambda

当然可以,以下是你提供的Java 8新特性文档的中文翻译:

Java 8新特性

Lambda表达式

Lambda表达式是Java 8中引入的轻量级、高度简洁的匿名方法。你可以使用它们跳入Java编程的全新世界。

Lambda表达式简介

Lambda表达式是匿名函数。

传统方式

  [java]
1
2
3
4
5
6
7
8
public void old() { Runnable r = new Runnable() { public void run() { System.out.println("Hello World!"); } }; new Thread(r).start(); }

使用Lambda

  [java]
1
new Thread(() -> System.out.println("Lambda Hello World!")).start();

Lambda表达式语法

  [plaintext]
1
2
3
4
5
6
(Type1 param1, Type2 param2, ..., TypeN paramN) -> { statment1; statment2; //............. return statmentM; }

1、无参数。

  [plaintext]
1
() -> { //..... };

2、一个参数,可以省略参数类型,编译器可以从上下文中推断。

  [plaintext]
1
2
3
4
5
6
param1 -> { statment1; statment2; //............. return statmentM; }

3、只有一个语句,可以省略{}

  [plaintext]
1
param1 -> statment

4、省略参数类型。

  [plaintext]
1
2
3
4
5
6
(param1,param2, ..., paramN) -> { statment1; statment2; //... return statmentM; }

示例

Person接口

  [java]
1
2
3
public interface Person { void say(String string); }

使用Lambda调用

  [java]
1
2
3
4
public static void main(String[] args) { Person h = str -> System.out.println(str); h.say("Hello World"); }

结果

  [plaintext]
1
Hello World

接口默认方法

Person接口

  [java]
1
2
3
4
5
6
7
public interface Person { void say(); default void eat() { System.out.println("eat..."); } }

Student类

  [java]
1
2
3
4
5
6
public class Student implements Person { @Override public void say() { System.out.println("say..."); } }

测试

  [java]
1
2
3
4
5
public static void main(String[] args) { Student student = new Student(); student.say(); student.eat(); }

结果

  [plaintext]
1
2
3
4
say... eat... Process finished with exit code 0

方法引用

1、静态方法:ClassName::methodName

  [java]
1
2
3
4
public static void main(String[] args) { List<String> strs = Arrays.asList("aa","bb","cc"); strs.forEach(System.out::println); }

2、实例方法:instanceRefence::methodName

  [java]
1
2
3
4
5
6
7
8
9
10
11
12
public class HelloWorld { void print(){ System.out.println("instanceRefence::methodName"); } public void printInfo(){ System.out.println("printInfo"); //instance method reference new Thread(this::print).start(); } }

3、构造方法:Class::new

  [java]
1
2
3
4
5
6
7
8
9
10
11
public class User { String username; User(String username){ this.username = username; } public String getUsername(){ return username; } }
  [java]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
public class HelloWorld { @FunctionalInterface interface UserFactory<T extends User> { T create(String username); } private void test() { UserFactory<User> uf = User::new; List<User> users = new ArrayList<>(); for (int i = 0; i < 5; ++i) { users.add(uf.create("user"+i)); } users.stream().map(User::getUsername).forEach(System.out::println); } }

重复注解

  [java]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
public class RepeatingAnnotationsTest { @Target(ElementType.TYPE) @Retention(RetentionPolicy.RUNTIME) public @interface Filters { Filter[] value(); } @Target(ElementType.TYPE) @Retention(RetentionPolicy.RUNTIME) @Repeatable(Filters.class) public @interface Filter { String value(); } @Filter("filter1") @Filter("filter2") public interface Filterable { } public static void main(String[] args) { for (Filter filter : Filterable.class.getAnnotationsByType(Filter.class)) { System.out.println(filter.value()); } } }

结果

  [plaintext]
1
2
filter1 filter2

更好的类型推断

  [java]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
public class TypeInference<T> { public static <T> T defaultValue() { return null; } public T getOrDefault(T value, T defaultValue) { return (value != null) ? value : defaultValue; } @Test public void betterTest() { final TypeInference<String> value = new TypeInference<>(); String result = value.getOrDefault("22", TypeInference.defaultValue()); System.out.println(result); //22 } }

更多有用的注解

Java 8拓宽了注解的应用场景。现在,注解几乎可以使用在任何元素上:局部变量、接口类型、超类和接口实现类,甚至可以用在函数的异常定义上。

  [java]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
public class MoreUsefulAnnotationTest { @Retention(RetentionPolicy.RUNTIME) @Target({ElementType.TYPE_USE, ElementType.TYPE_PARAMETER}) public @interface NonEmpty { } public static class Holder<@NonEmpty T> extends @NonEmpty Object { public void method() throws @NonEmpty Exception { } } @SuppressWarnings("unused") public static void main(String[] args) { final Holder<String> holder = new @NonEmpty Holder<String>(); @NonEmpty Collection<@NonEmpty String> strings = new ArrayList<>(); } }

Optional

Optional是一个容器,可以存放T类型的值或者null。它提供了一些有用的接口来避免显式的null检查。

null测试

  [java]
1
2
3
4
5
6
7
@Test public void nullTest() { Optional< String > fullName = Optional.ofNullable( null ); System.out.println( "Full Name is set? " + fullName.isPresent() ); System.out.println( "Full Name: " + fullName.orElseGet( () -> "[none]" ) ); System.out.println( fullName.map( s -> "Hey " + s + "!" ).orElse( "Hey Stranger!" ) ); }

结果

  [plaintext]
1
2
3
Full Name is set? false Full Name: [none] Hey Stranger!

非null测试

  [java]
1
2
3
4
5
6
7
8
9
@Test public void notNullTest() { Optional< String > fullName = Optional.ofNullable( "ryo" ); System.out.println( "Full Name is set? " + fullName.isPresent() ); System.out.println( "Full Name: " + fullName.orElseGet( () -> "[none]" ) ); System.out.println( fullName.map( s -> "Hey " + s + "!" ).orElse( "Hey Stranger!" ) ); }

结果

  [plaintext]
1
2
3
Full Name is set? true Full Name: ryo Hey ryo!

Streams

Java 8新增的Stream API(java.util.stream)引入了函数式编程的概念到Java库中。

  [java]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
public class StreamsTest { private enum Status { OPEN, CLOSED } private static final class Task { private final Status status; private final Integer points; Task(final Status status, final Integer points) { this.status = status; this.points = points; } public Integer getPoints() { return points; } public Status getStatus() { return status; } @Override public String toString() { return String.format("[%s, %d]", status, points); } } }

测试

  [java]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
@Test public void totalTest() { final Collection<Task> tasks = Arrays.asList( new Task(Status.OPEN, 5), new Task(Status.OPEN, 13), new Task(Status.CLOSED, 8) ); final long totalPointsOfOpenTasks = tasks .stream() .filter(task -> task.getStatus() == Status.OPEN) .mapToInt(Task::getPoints) .sum(); System.out.println(totalPointsOfOpenTasks); //18 } @Test public void parallelTest() { final Collection<Task> tasks = Arrays.asList( new Task(Status.OPEN, 5), new Task(Status.OPEN, 13), new Task(Status.CLOSED, 8) ); final double totalPoints = tasks .stream() .parallel() .map(task -> task.getPoints()) .reduce(0, Integer::sum); System.out.println("Total points (all tasks): " + totalPoints); //Total points (all tasks): 26.0 } @Test public void groupByTest() { final Collection<Task> tasks = Arrays.asList( new Task(Status.OPEN, 5), new Task(Status.OPEN, 13), new Task(Status.CLOSED, 8) ); final Map< Status, List< Task >> map = tasks .stream() .collect( Collectors.groupingBy( Task::getStatus ) ); System.out.println( map ); //{OPEN=[[OPEN, 5], [OPEN, 13]], CLOSED=[[CLOSED, 8]]} } @Test public void percentTest() { final Collection<Task> tasks = Arrays.asList( new Task(Status.OPEN, 5), new Task(Status.OPEN, 13), new Task(Status.CLOSED, 8) ); final double totalPoints = tasks .stream() .parallel() .map( task -> task.getPoints() ) .reduce( 0, Integer::sum ); final Collection< String > result = tasks .stream() .mapToInt( Task::getPoints ) .asLongStream() .mapToDouble( points -> points / totalPoints ) .boxed() .mapToLong( weigth -> ( long )( weigth * 100 ) ) .mapToObj( percentage -> percentage + "%" ) .collect( Collectors.toList() ); System.out.println(result); //[19%, 50%, 30%] }

日期/时间API(JSR 310)

Java 8引入了新的Date-Time API(JSR 310)来改进时间、日期的处理。

Nashorn JavaScript

Java 8提供了新的Nashorn JavaScript引擎,使得我们可以在JVM上开发和运行js应用。

  [java]
1
2
3
4
5
6
7
8
9
10
11
12
public class NashornJSTest { @Test public void jsTest() throws ScriptException { ScriptEngineManager manager = new ScriptEngineManager(); ScriptEngine engine = manager.getEngineByName( "JavaScript" ); System.out.println( engine.getClass().getName() ); System.out.println( "Result:" + engine.eval( "function f() { return 1; }; f() + 1;" ) ); } }

结果

  [plaintext]
1
2
jdk.nashorn.api.scripting.NashornScriptEngine Result:2.0

Base64

Java 8加入了对Base64编码的支持

  [java]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
@Test public void base64Test() { final String text = "Base64 finally in Java 8!"; final String encoded = Base64 .getEncoder() .encodeToString(text.getBytes(StandardCharsets.UTF_8)); System.out.println(encoded); final String decoded = new String( Base64.getDecoder().decode(encoded), StandardCharsets.UTF_8); System.out.println(decoded); }

结果

  [plaintext]
1
2
QmFzZTY0IGZpbmFsbHkgaW4gSmF2YSA4IQ== Base64 finally in Java 8!

ParallelArrays

  [java]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
@Test public void parallelArraysTest() { long[] arrayOfLong = new long[20000]; Arrays.parallelSetAll(arrayOfLong, index -> ThreadLocalRandom.current().nextInt(1000000)); Arrays.stream(arrayOfLong).limit(10).forEach( i -> System.out.print(i + " ")); System.out.println(); Arrays.parallelSort(arrayOfLong); Arrays.stream(arrayOfLong).limit(10).forEach( i -> System.out.print(i + " ")); System.out.println(); }

结果

  [plaintext]
1
2
395659 19377 864569 289077 710936 742196 922967 850922 701156 551843 7 17 49 111 116 173 194 260 344 396