摘要:首先我們定義一個有兩個不同控制器的然后,我們創(chuàng)建一個特定的工廠接口來創(chuàng)建新的對象不需要手動的去繼承實(shí)現(xiàn)該工廠接口,我們只需要將控制器的引用傳遞給該接口對象就好了的控制器會自動選擇合適的構(gòu)造器方法。這種指向時間軸的對象即是類。
本文為翻譯文章,原文地址 這里
歡迎來到本人對于Java 8的系列介紹教程,本教程會引導(dǎo)你一步步領(lǐng)略最新的語法特性。通過一些簡單的代碼示例你即可以學(xué)到默認(rèn)的接口方法、Lambda表達(dá)式、方法引用以及重復(fù)注解等等。本文的最后還提供了譬如Stream API之類的詳細(xì)的介紹。
Default Methods for InterfacesJava 8 允許我們利用default關(guān)鍵字來向接口中添加非抽象的方法作為默認(rèn)方法。下面是一個小例子:
interface Formula { double calculate(int a); default double sqrt(int a) { return Math.sqrt(a); } }
在接口Formula中,除了抽象方法calculate之外,還定義了一個默認(rèn)的方法sqrt。實(shí)現(xiàn)類只需要實(shí)現(xiàn)抽象方法calculate,而sqrt方法可以跟普通方法一樣調(diào)用。
Formula formula = new Formula() { @Override public double calculate(int a) { return sqrt(a * 100); } }; formula.calculate(100); // 100.0 formula.sqrt(16); // 4.0Lambda表達(dá)式
首先以簡單的字符串排序?yàn)槔齺碚故荆?/p>
Listnames = Arrays.asList("peter", "anna", "mike", "xenia"); Collections.sort(names, new Comparator () { @Override public int compare(String a, String b) { return b.compareTo(a); } });
靜態(tài)的工具類方法Collections.sort接受一個列表參數(shù)和一個比較器對象來對于指定列表中的元素進(jìn)行排序。我們常常需要創(chuàng)建匿名比較器并且將他們傳遞給排序方法。而Java 8中提供的一種更短的Lambda表達(dá)式的方法來完成該工作:
Collections.sort(names, (String a, String b) -> { return b.compareTo(a); });
可以發(fā)現(xiàn)用如上的方法寫會更短并且更加的可讀,并且可以更短:
Collections.sort(names, (String a, String b) -> b.compareTo(a));
這種寫法就完全不需要{}以及return關(guān)鍵字,再進(jìn)一步簡化的話,就變成了:
names.sort((a, b) -> b.compareTo(a));Functional Interfaces
Lambda表達(dá)式是如何適配進(jìn)Java現(xiàn)存的類型系統(tǒng)的呢?每個Lambda表達(dá)式都會關(guān)聯(lián)到一個由接口確定的給定的類型。這種所謂的函數(shù)式接口必須只能包含一個抽象方法,而每個該類型的Lambda表達(dá)式都會關(guān)聯(lián)到這個抽象方法。不過由于默認(rèn)方法不是抽象的,因此可以隨便添加幾個默認(rèn)的方法到函數(shù)式接口中。我們可以將任意的接口作為Lambda表達(dá)式使用,只要該接口僅含有一個抽象方法即可。為了保證你的接口滿足這個需求,應(yīng)該添加@FunctionalInterface這個注解。編譯器會在你打算向某個函數(shù)式接口中添加第二個抽象方法時候報錯。
//聲明 @FunctionalInterface interface ConverterMethod and Constructor References{ T convert(F from); } //使用 Converter converter = (from) -> Integer.valueOf(from); Integer converted = converter.convert("123"); System.out.println(converted); // 123
上述的代碼可以使用靜態(tài)方法引用而更加的簡化:
Converterconverter = Integer::valueOf; Integer converted = converter.convert("123"); System.out.println(converted); // 123
Java 8允許將方法或則構(gòu)造器的引用通過::關(guān)鍵字進(jìn)行傳遞,上述的例子是演示了如何關(guān)聯(lián)一個靜態(tài)方法,不過我們也可以關(guān)聯(lián)一個對象方法:
class Something { String startsWith(String s) { return String.valueOf(s.charAt(0)); } }
Something something = new Something(); Converterconverter = something::startsWith; String converted = converter.convert("Java"); System.out.println(converted); // "J"
接下來看 :: 關(guān)鍵字怎么在構(gòu)造器中起作用。首先我們定義一個有兩個不同控制器的Java Bean:
class Person { String firstName; String lastName; Person() {} Person(String firstName, String lastName) { this.firstName = firstName; this.lastName = lastName; } }
然后,我們創(chuàng)建一個特定的Person工廠接口來創(chuàng)建新的Person對象:
interface PersonFactory{ P create(String firstName, String lastName); }
不需要手動的去繼承實(shí)現(xiàn)該工廠接口,我們只需要將控制器的引用傳遞給該接口對象就好了:
PersonFactorypersonFactory = Person::new; Person person = personFactory.create("Peter", "Parker");
Java的控制器會自動選擇合適的構(gòu)造器方法。
Lambda Scopes從Lambda表達(dá)式中訪問外部作用域中變量非常類似于匿名對象,可以訪問本地的final變量、實(shí)例域以及靜態(tài)變量。
Accessing local variables在匿名對象中,我們可以從Lambda表達(dá)式的域中訪問外部的final變量。
final int num = 1; ConverterstringConverter = (from) -> String.valueOf(from + num); stringConverter.convert(2); // 3
但是不同于匿名對象只能訪問final變量,Lambda表達(dá)式中可以訪問final變量:
int num = 1; ConverterstringConverter = (from) -> String.valueOf(from + num); stringConverter.convert(2); // 3
不過需要注意的是,盡管變量不需要聲明為final,但是也是隱式的不可變:
int num = 1; ConverterstringConverter = (from) -> String.valueOf(from + num); num = 3;
譬如如上的寫法就會被報錯。
Accessing fields and static variables不同于本地變量,我們可以在Lambda表達(dá)式中任意的讀寫:
class Lambda4 { static int outerStaticNum; int outerNum; void testScopes() { ConverterAccessing Default Interface MethodsstringConverter1 = (from) -> { outerNum = 23; return String.valueOf(from); }; Converter stringConverter2 = (from) -> { outerStaticNum = 72; return String.valueOf(from); }; } }
注意,Lambda表達(dá)式中是不可以訪問默認(rèn)方法的:
Formula formula = (a) -> sqrt( a * 100);
上述代碼是編譯通不過的。
Built-in Functional InterfacesJDK 1.8 的API中包含了許多的內(nèi)建的函數(shù)式接口,其中部分的譬如Comparator、Runnable被改寫成了可以由Lambda表達(dá)式支持的方式。除此之外,Java 8還添加了許多來自于Guava中的依賴庫,并將其改造為了Lambda接口。
PredicatesPredicates是包含一個參數(shù)的返回為布爾值的接口,接口包含了許多的默認(rèn)方法來進(jìn)行不同的復(fù)雜的邏輯組合:
PredicateFunctionspredicate = (s) -> s.length() > 0; predicate.test("foo"); // true predicate.negate().test("foo"); // false Predicate nonNull = Objects::nonNull; Predicate isNull = Objects::isNull; Predicate isEmpty = String::isEmpty; Predicate isNotEmpty = isEmpty.negate();
Functions接口接受一個參數(shù)并且產(chǎn)生一個結(jié)果,同樣提供了部分默認(rèn)的方法來鏈?zhǔn)浇M合不同的函數(shù)(compose,andThen)。
FunctionSupplierstoInteger = Integer::valueOf; Function backToString = toInteger.andThen(String::valueOf); backToString.apply("123"); // "123"
SupplierConsumerspersonSupplier = Person::new; personSupplier.get(); // new Person
ConsumerComparatorsgreeter = (p) -> System.out.println("Hello, " + p.firstName); greeter.accept(new Person("Luke", "Skywalker"));
Comparators類似于舊版本中的用法,Java 8是添加了一些默認(rèn)的用法。
ComparatorOptionalscomparator = (p1, p2) -> p1.firstName.compareTo(p2.firstName); Person p1 = new Person("John", "Doe"); Person p2 = new Person("Alice", "Wonderland"); comparator.compare(p1, p2); // > 0 comparator.reversed().compare(p1, p2); // < 0
Optionals并不是一個函數(shù)式接口,但是非常有用的工具類來防止NullPointerException。Optional是一個簡單的容器用于存放那些可能為空的值。對于那種可能返回為null的方法可以考慮返回Optional而不是null:
OptionalStreamsoptional = Optional.of("bam"); optional.isPresent(); // true optional.get(); // "bam" optional.orElse("fallback"); // "bam" optional.ifPresent((s) -> System.out.println(s.charAt(0))); // "b"
一個Java的Stream對象代表了一系列可以被附加多個操作的元素的序列集合。常用的Stream API分為媒介操作與終止操作。其中終止操作會返回某個特定類型的值,而媒介操作則會返回流本身以方便下一步的鏈?zhǔn)讲僮?。Streams可以從java.util.Collection的數(shù)據(jù)類型譬如lists或者sets(不支持maps)中創(chuàng)建。而Streams的操作可以順序執(zhí)行也可以并發(fā)地執(zhí)行。
Stream.js是一個利用JavaScript實(shí)現(xiàn)的Java 8的流接口。
首先我們創(chuàng)建待處理的數(shù)據(jù):
ListstringCollection = new ArrayList<>(); stringCollection.add("ffffd2"); stringCollection.add("aaa2"); stringCollection.add("bbb1"); stringCollection.add("aaa1"); stringCollection.add("bbb3"); stringCollection.add("ccc"); stringCollection.add("bbb2"); stringCollection.add("ffffd1");
接下來可以利用Collection.stream() or Collection.parallelStream()來轉(zhuǎn)化為一個流對象。
FilterFilter會接受一個Predicate對象來過濾流中的元素,這個操作屬于媒介操作,譬如可以在該操作之后調(diào)用另一個流操作(forEach)。ForEach操作屬于終止操作,接受一個Consumer對象來對于過濾之后的流中的每一個元素進(jìn)行操作。
stringCollection .stream() .filter((s) -> s.startsWith("a")) .forEach(System.out::println); // "aaa2", "aaa1"Sorted
Sorted操作屬于一個媒介操作,會將流對象作為返回值返回。元素會默認(rèn)按照自然的順序返回,除非你傳入了一個自定義的Comparator對象。
stringCollection .stream() .sorted() .filter((s) -> s.startsWith("a")) .forEach(System.out::println); // "aaa1", "aaa2"
需要記住的是,Sorted操作并不會改變流中的元素的順序,只會創(chuàng)建一個經(jīng)過排序的視圖,譬如:
System.out.println(stringCollection); // ffffd2, aaa2, bbb1, aaa1, bbb3, ccc, bbb2, ffffd1Map
map操作也是媒介操作的一種,可以通過給定的函數(shù)將每個元素映射到其他對象。下面的代碼示例就是將所有的字符串轉(zhuǎn)化為大寫字符串。不過map操作是可以將任意對象轉(zhuǎn)化為任意類型,流返回的泛型類型取決于傳遞給map的函數(shù)的返回值。
stringCollection .stream() .map(String::toUpperCase) .sorted((a, b) -> b.compareTo(a)) .forEach(System.out::println); // "DDD2", "DDD1", "CCC", "BBB3", "BBB2", "AAA2", "AAA1"Match
Java 8提供了一些列的匹配的終止操作符來幫助開發(fā)者判斷流當(dāng)中的元素是否符合某些判斷規(guī)則。所有的匹配類型的操作都會返回布爾類型。
boolean anyStartsWithA = stringCollection .stream() .anyMatch((s) -> s.startsWith("a")); System.out.println(anyStartsWithA); // true boolean allStartsWithA = stringCollection .stream() .allMatch((s) -> s.startsWith("a")); System.out.println(allStartsWithA); // false boolean noneStartsWithZ = stringCollection .stream() .noneMatch((s) -> s.startsWith("z")); System.out.println(noneStartsWithZ); // trueCount
Count 也是終止操作的一種,它會返回流中的元素的數(shù)目。
long startsWithB = stringCollection .stream() .filter((s) -> s.startsWith("b")) .count(); System.out.println(startsWithB); // 3Reduce
該操作根據(jù)指定的方程對于流中的元素進(jìn)行了指定的減少的操作。結(jié)果是Optional類型。
OptionalParallel Streamsreduced = stringCollection .stream() .sorted() .reduce((s1, s2) -> s1 + "#" + s2); reduced.ifPresent(System.out::println); // "aaa1#aaa2#bbb1#bbb2#bbb3#ccc#ffffd1#ffffd2"
正如上文中提及的,流可以是順序的也可以是并行的。所有在順序流上執(zhí)行的流操作都是在單線程中運(yùn)行的,而在并行流中進(jìn)行的操作都是在多線程中運(yùn)行的。如下的代碼演示了如何利用并行流來提供性能,首先我們創(chuàng)建一個待比較的序列:
int max = 1000000; ListSequential Sortvalues = new ArrayList<>(max); for (int i = 0; i < max; i++) { UUID uuid = UUID.randomUUID(); values.add(uuid.toString()); }
long t0 = System.nanoTime(); long count = values.stream().sorted().count(); System.out.println(count); long t1 = System.nanoTime(); long millis = TimeUnit.NANOSECONDS.toMillis(t1 - t0); System.out.println(String.format("sequential sort took: %d ms", millis)); // sequential sort took: 899 msParallel Sort
long t0 = System.nanoTime(); long count = values.parallelStream().sorted().count(); System.out.println(count); long t1 = System.nanoTime(); long millis = TimeUnit.NANOSECONDS.toMillis(t1 - t0); System.out.println(String.format("parallel sort took: %d ms", millis)); // parallel sort took: 472 msMaps
正如上文所說,Map并不支持流操作,但是也提供了很多有用的方法來進(jìn)行通用的操作。
Mapmap = new HashMap<>(); for (int i = 0; i < 10; i++) { map.putIfAbsent(i, "val" + i); } map.forEach((id, val) -> System.out.println(val));
上述的代碼中,putIfAbsent避免了寫太多額外的空檢查。forEach會接受一個Consumer參數(shù)來遍歷Map中的元素。下面的代碼演示如何進(jìn)行計算:
map.computeIfPresent(3, (num, val) -> val + num); map.get(3); // val33 map.computeIfPresent(9, (num, val) -> null); map.containsKey(9); // false map.computeIfAbsent(23, num -> "val" + num); map.containsKey(23); // true map.computeIfAbsent(3, num -> "bam"); map.get(3); // val33
還有,Map提供了如何根據(jù)給定的key,vaue來刪除Map中給定的元素:
map.remove(3, "val3"); map.get(3); // val33 map.remove(3, "val33"); map.get(3); // null
還有一個比較有用的方法:
map.getOrDefault(42, "not found"); // not found
同時,Map還提供了merge方法來幫助有效地對于值進(jìn)行修正:
map.merge(9, "val9", (value, newValue) -> value.concat(newValue)); map.get(9); // val9 map.merge(9, "concat", (value, newValue) -> value.concat(newValue)); map.get(9); // val9concatDate API
Java 8包含了一個全新的日期與時間的API,在java.time包下,新的時間API集成了Joda-Time的庫。
ClockClock方便我們?nèi)プx取當(dāng)前的日期與時間。Clocks可以根據(jù)不同的時區(qū)來進(jìn)行創(chuàng)建,并且可以作為System.currentTimeMillis()的替代。這種指向時間軸的對象即是Instant類。Instants可以被用于創(chuàng)建java.util.Date對象。
Clock clock = Clock.systemDefaultZone(); long millis = clock.millis(); Instant instant = clock.instant(); Date legacyDate = Date.from(instant); // legacy java.util.DateTimezones
Timezones以ZoneId來區(qū)分??梢酝ㄟ^靜態(tài)構(gòu)造方法很容易的創(chuàng)建,Timezones定義了Instants與Local Dates之間的轉(zhuǎn)化關(guān)系:
System.out.println(ZoneId.getAvailableZoneIds()); // prints all available timezone ids ZoneId zone1 = ZoneId.of("Europe/Berlin"); ZoneId zone2 = ZoneId.of("Brazil/East"); System.out.println(zone1.getRules()); System.out.println(zone2.getRules()); // ZoneRules[currentStandardOffset=+01:00] // ZoneRules[currentStandardOffset=-03:00]LocalTime
LocalTime代表了一個與時間無關(guān)的本地時間,譬如 10pm 或者 17:30:15。下述的代碼展示了根據(jù)不同的時間軸創(chuàng)建的不同的本地時間:
LocalTime now1 = LocalTime.now(zone1); LocalTime now2 = LocalTime.now(zone2); System.out.println(now1.isBefore(now2)); // false long hoursBetween = ChronoUnit.HOURS.between(now1, now2); long minutesBetween = ChronoUnit.MINUTES.between(now1, now2); System.out.println(hoursBetween); // -3 System.out.println(minutesBetween); // -239
LocalTime提供了很多的工廠方法來簡化創(chuàng)建實(shí)例的步驟,以及對于時間字符串的解析:
LocalTime late = LocalTime.of(23, 59, 59); System.out.println(late); // 23:59:59 DateTimeFormatter germanFormatter = DateTimeFormatter .ofLocalizedTime(FormatStyle.SHORT) .withLocale(Locale.GERMAN); LocalTime leetTime = LocalTime.parse("13:37", germanFormatter); System.out.println(leetTime); // 13:37LocalDate
LocalDate代表了一個獨(dú)立的時間類型,譬如2014-03-11。它是一個不可變的對象并且很類似于LocalTime。下列代碼展示了如何通過增減時間年月來計算日期:
LocalDate today = LocalDate.now(); LocalDate tomorrow = today.plus(1, ChronoUnit.DAYS); LocalDate yesterday = tomorrow.minusDays(2); LocalDate independenceDay = LocalDate.of(2014, Month.JULY, 4); DayOfWeek dayOfWeek = independenceDay.getDayOfWeek(); System.out.println(dayOfWeek); // FRIDAY
從字符串解析得到LocalDate對象也像LocalTime一樣簡單:
DateTimeFormatter germanFormatter = DateTimeFormatter .ofLocalizedDate(FormatStyle.MEDIUM) .withLocale(Locale.GERMAN); LocalDate xmas = LocalDate.parse("24.12.2014", germanFormatter); System.out.println(xmas); // 2014-12-24LocalDateTime
LocalDateTime代表了時間日期類型,它組合了上文提到的Date類型以及Time類型。LocalDateTime同樣也是一種不可變類型,很類似于LocalTime以及LocalDate。
LocalDateTime sylvester = LocalDateTime.of(2014, Month.DECEMBER, 31, 23, 59, 59); DayOfWeek dayOfWeek = sylvester.getDayOfWeek(); System.out.println(dayOfWeek); // WEDNESDAY Month month = sylvester.getMonth(); System.out.println(month); // DECEMBER long minuteOfDay = sylvester.getLong(ChronoField.MINUTE_OF_DAY); System.out.println(minuteOfDay); // 1439
上文中提及的Instant也可以用來將時間根據(jù)時區(qū)轉(zhuǎn)化:
Instant instant = sylvester .atZone(ZoneId.systemDefault()) .toInstant(); Date legacyDate = Date.from(instant); System.out.println(legacyDate); // Wed Dec 31 23:59:59 CET 2014
從格式化字符串中解析獲取到數(shù)據(jù)對象,也是非常簡單:
DateTimeFormatter formatter = DateTimeFormatter .ofPattern("MMM dd, yyyy - HH:mm"); LocalDateTime parsed = LocalDateTime.parse("Nov 03, 2014 - 07:13", formatter); String string = formatter.format(parsed); System.out.println(string); // Nov 03, 2014 - 07:13Annotations
Java 8中的注解現(xiàn)在是可以重復(fù)的,下面我們用例子直接說明。首先,創(chuàng)建一個容器注解可以用來存儲一系列真實(shí)的注解:
@interface Hints { Hint[] value(); } @Repeatable(Hints.class) @interface Hint { String value(); }
通過添加 @Repeatable注解,就可以在同一個類型中使用多個注解。
Variant 1: Using the container annotation (old school)@Hints({@Hint("hint1"), @Hint("hint2")}) class Person {}Variant 2: Using repeatable annotations (new school)
@Hint("hint1") @Hint("hint2") class Person {}
Hint hint = Person.class.getAnnotation(Hint.class); System.out.println(hint); // null Hints hints1 = Person.class.getAnnotation(Hints.class); System.out.println(hints1.value().length); // 2 Hint[] hints2 = Person.class.getAnnotationsByType(Hint.class); System.out.println(hints2.length); // 2Further Reading
Java 8 Stream Tutorial
Java 8 Nashorn Tutorial
Java 8 Concurrency Tutorial: Threads and Executors
Java 8 Concurrency Tutorial: Synchronization and Locks
Java 8 Concurrency Tutorial: Atomic Variables and ConcurrentMap
Java 8 API by Example: Strings, Numbers, Math and Files
Avoid Null Checks in Java 8
Fixing Java 8 Stream Gotchas with IntelliJ IDEA
Using Backbone.js with Java 8 Nashorn
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/64730.html
摘要:大家好,我是冰河有句話叫做投資啥都不如投資自己的回報率高。馬上就十一國慶假期了,給小伙伴們分享下,從小白程序員到大廠高級技術(shù)專家我看過哪些技術(shù)類書籍。 大家好,我是...
摘要:大家好,我是樂字節(jié)的小樂,上一次我們說到了核心特性之函數(shù)式接口,接下來我們繼續(xù)了解又一核心特性方法引用。方法引用是一種更簡潔易懂的表達(dá)式。感謝光臨閱讀小樂的,敬請關(guān)注樂字節(jié)后續(xù)將繼續(xù)講述等前沿知識技術(shù)。 大家好,我是樂字節(jié)的小樂,上一次我們說到了Java8核心特性之函數(shù)式接口,接下來我們繼續(xù)了解Java8又一核心特性——方法引用。 showImg(https://segmentfaul...
摘要:大家好,上一篇小樂給大家講述了樂字節(jié)核心特性表達(dá)式,點(diǎn)擊回顧。接下來繼續(xù)核心特性之函數(shù)式接口。感謝大家欣賞小樂帶來的核心特性之函數(shù)式接口,接下來還會更多核心技術(shù)講解,請關(guān)注樂字節(jié)如需要視頻課程,請搜索樂字節(jié)騰訊課堂 大家好,上一篇小樂給大家講述了《樂字節(jié)-Java8核心特性-Lambda表達(dá)式》,點(diǎn)擊回顧。接下來繼續(xù):Java8核心特性之函數(shù)式接口。 什么時候可以使用Lambda?通常...
摘要:使用表達(dá)式,使得應(yīng)用變得簡潔而緊湊。很多語言等從設(shè)計之初就支持表達(dá)式。表達(dá)式的參數(shù)與函數(shù)式接口內(nèi)方法的參數(shù),返回值類型相互對應(yīng)。更多教程和資料請上騰訊課堂樂字節(jié) showImg(https://segmentfault.com/img/bVbtotg?w=935&h=345); Java8 引入Lambda表達(dá)式,允許開發(fā)者將函數(shù)當(dāng)成參數(shù)傳遞給某個方法,或者把代碼本身當(dāng)作數(shù)據(jù)進(jìn)行處理。...
摘要:實(shí)戰(zhàn)高并發(fā)程序設(shè)計推薦豆瓣評分書的質(zhì)量沒的說,推薦大家好好看一下。推薦,豆瓣評分,人評價本書介紹了在編程中條極具實(shí)用價值的經(jīng)驗(yàn)規(guī)則,這些經(jīng)驗(yàn)規(guī)則涵蓋了大多數(shù)開發(fā)人員每天所面臨的問題的解決方案。 很早就想把JavaGuide的書單更新一下了,昨晚加今天早上花了幾個時間對之前的書單進(jìn)行了分類和補(bǔ)充完善。雖是終極版,但一定還有很多不錯的 Java 書籍我沒有添加進(jìn)去,會繼續(xù)完善下去。希望這篇...
閱讀 3575·2023-04-25 14:20
閱讀 1196·2021-09-10 10:51
閱讀 1154·2019-08-30 15:53
閱讀 463·2019-08-30 15:43
閱讀 2316·2019-08-30 14:13
閱讀 2797·2019-08-30 12:45
閱讀 1207·2019-08-29 16:18
閱讀 1166·2019-08-29 16:12