2017年你不能錯過的Java類庫
各位讀者好,這篇文章是在我看過 Andres Almiray 的一篇介紹文后,整理出來的。因為內(nèi)容非常好,我便將它整理成參考列表分享給大家, 同時附上各個庫的特性簡介和示例。請欣賞!
Guice
Guice (發(fā)音同 ‘juice’) ,是一個 Google 開發(fā)的輕量級依賴性注入框架,適合 Java 6 以上的版本。
# Typical dependency injection
public class DatabaseTransactionLogProvider implements Provider<TransactionLog> {
@Inject Connection connection;
public TransactionLog get() {
return new DatabaseTransactionLog(connection);
}
}
# FactoryModuleBuilder generates factory using your interface
public interface PaymentFactory {
Payment create(Date startDate, Money amount);
}
GitHub, JavaDoc, 使用指南, FactoryModuleBuilder
OKHttp
HTTP是現(xiàn)代應(yīng)用程序?qū)崿F(xiàn)網(wǎng)絡(luò)連接的途徑,也是我們進行數(shù)據(jù)和媒體交換的工具。高效使用HTTP能使你的東西加載更快,并節(jié)省帶寬。
OkHttp是一個非常高效的HTTP客戶端,默認情況下:
- 支持HTTP/2,允許對同一主機的請求共用一個套接字。
- 如果HTTP/2 不可用,連接池會減少請求延遲。
- 透明的GZIP可以減少下載流量。
- 響應(yīng)的緩存避免了重復(fù)的網(wǎng)絡(luò)請求。
OkHttpClient client = new OkHttpClient();
String run(String url) throws IOException {
Request request = new Request.Builder()
.url(url)
.build();
Response response = client.newCall(request).execute();
return response.body().string();
}
Retrofit
Retrofit 是 Square 下的類型安全的 HTTP 客戶端,支持 Android 和 Java 等,它能將你的 HTTP API 轉(zhuǎn)換為 Java 接口。
Retrofit 將 HTTP API 轉(zhuǎn)換為 Java 接口:
public interface GitHubService {
@GET("users/{user}/repos")
Call<List<Repo>listRepos(@Path("user") String user);
}
Retrofit 類實現(xiàn) GitHubService 接口:
Retrofit retrofit = new Retrofit.Builder()
.baseUrl("https://api.github.com/")
.build();
GitHubService service = retrofit.create(GitHubService.class);
來自 GitHubService 的每個 Call 都能產(chǎn)生為遠程 Web 服務(wù)產(chǎn)生一個異步或同步 HTTP 請求:
Call<List<Repo>> repos = service.listRepos("octocat");
JDeferred
與JQuery類似的Java Deferred/Promise類庫
- Deferred 對象和 Promise
- Promise 回調(diào):
.then(…)
,.done(…)
,.fail(…)
,.progress(…)
,.always(…)
- 支持多個promises -
.when(p1, p2, p3, …).then(…)
- Callable 和 Runnable -
wrappers.when(new Runnable() {…})
- 使用 Executor 服務(wù)
- 支持Java 泛型:
Deferred<Integer, Exception, Doubledeferred;
,deferred.resolve(10);
,deferred.reject(new Exception());
,deferred.notify(0.80);
, - 支持Android
- Java 8 Lambda的友好支持
RxJava
RxJava – JVM的響應(yīng)式編程擴展 – 是一個為Java虛擬機編寫的使用可觀察序列的構(gòu)建異步的基于事件的程序的類庫。
它基于觀察者模式實現(xiàn)對數(shù)據(jù)/事件的序列的支持,并添加了一些操作符,允許你以聲明式構(gòu)建序列, 使得開發(fā)者無需關(guān)心底層的線程、同步、線程安全和并發(fā)數(shù)據(jù)結(jié)構(gòu)。
RxJava最常見的一個用法就是在后臺線程運行一些計算和網(wǎng)絡(luò)請求,而在UI線程顯示結(jié)果(或者錯誤):
Flowable.fromCallable(() -{
Thread.sleep(1000); // imitate expensive computation
return "Done";
})
.subscribeOn(Schedulers.io())
.observeOn(Schedulers.single())
.subscribe(System.out::println, Throwable::printStackTrace);
Thread.sleep(2000); // <--- wait for the flow to finish
MBassador
MBassador是一個實現(xiàn)了發(fā)布-訂閱模式的輕量級的,高性能的事件總線。它易于使用,并力求功能豐富,易于擴展,而同時又保證資源的高效利用和高性能。
MBassador的高性能的核心是一個專業(yè)的數(shù)據(jù)結(jié)構(gòu),它提供了非阻塞的讀取器,并最小化寫入器的鎖爭用,因此并發(fā)讀寫訪問的性能衰減會是最小的。
- 注解驅(qū)動的
- 提供任何東西,慎重對待類型層次結(jié)構(gòu)
- 同步和異步的消息傳遞
- 可配置的引用類型
- 消息過濾
- 封裝的消息
- 處理器的優(yōu)先級
- 自定義錯誤處理
- 可擴展性
// Define your listener
class SimpleFileListener{
@Handler
public void handle(File msg){
// do something with the file
}
}
// somewhere else in your code
MBassador bus = new MBassador();
Object listener = new SimpleFileListener();
bus.subscribe (listener);
bus.post(new File("/tmp/smallfile.csv")).now();
bus.post(new File("/tmp/bigfile.csv")).asynchronously();
Lombok項目
使用注解來減少Java中的重復(fù)代碼,比如getter,setters,非空檢查,生成的Builder等。
- val - 總算有了!無憂的final本地變量。
- @NonNull - 或:我如何學(xué)會不再擔(dān)心并愛上了非空異常(NullPointerException)。
- @Cleanup - 自動的資源管理:安全調(diào)用你的close() 方法,無需任何麻煩。
- @Getter / @Setter - 再也不用寫
public int getFoo() {return foo;}了
。 - @ToString - 無需啟動調(diào)試器來檢查你的字段:就讓Lombok來為你生成一個toString方法吧!
- @EqualsAndHashCode - 實現(xiàn)相等的判斷變得容易了:它會從你的對象的字段里為你生成hashCode和equals方法的實現(xiàn)。
- @NoArgsConstructor, @RequiredArgsConstructor and @AllArgsConstructor - 定做構(gòu)造函數(shù):為你生成各種各樣的構(gòu)造函數(shù),包括無參的,每一個final或非空的字段作為一個參數(shù)的,或者每一個字段都作為參數(shù)的。
- @Data - 所有的都同時生成:這是一個快捷方式,可以為所有字段生成
@ToString
,@EqualsAndHashCode
,@Getter注解,以及為所有非final的字段生成
@Setter
注解,以及生成@RequiredArgsConstructor!
- @Value - 聲明一個不可變類變得非常容易。
- @Builder - … 而且鮑伯是你叔叔:創(chuàng)建對象的無爭議且奢華的接口!
- @SneakyThrows - 在以前沒有人拋出檢查型異常的地方大膽的拋出吧!
- @Synchronized - 正確的實現(xiàn)同步:不要暴露你的鎖。
- @Getter(lazy=true) 懶惰是一種美德!
- @Log - 船長日志,星歷24435.7: “那一行又是什么呢?”
Java簡單日志門面(SLF4J)
Java簡單日志門面 (SLF4J) 為不同的日志框架(比如java.util.logging
, logback
, log4j)提供了簡單的門面或者抽象的實現(xiàn),允許最終用戶在部署時能夠接入自己想要使用的日志框架。
簡言之,類庫和其他嵌入式的組件都應(yīng)該考慮采用SLF4J作為他們的日志需求,因為類庫無法將它們對日志框架的選擇強加給最終用戶。另一方面,對于獨立的應(yīng)用來說,就不一定需要使用SLF4J。獨立應(yīng)用可以直接調(diào)用他們自己選擇的日志框架。而對于logback來說,這個
問題是沒有意義的,因為logback是通過
SLF4J來暴露其日志接口的。
JUnitParams
對測試進行參數(shù)化,還不錯
@Test
@Parameters({"17, false",
"22, true" })
public void personIsAdult(int age, boolean valid) throws Exception {
assertThat(new Person(age).isAdult(), is(valid));
}
與標(biāo)準(zhǔn)的JUnit 參數(shù)化運行器的區(qū)別如下:
- 更明確 – 參數(shù)實在測試方法的參數(shù)中,而不是在類的字段中
- 更少的代碼 – 你不需要用構(gòu)造函數(shù)來設(shè)置參數(shù)
- 你可以在同一個類混合使用參數(shù)化和非參數(shù)化的方法。
- 參數(shù)可以通過一個CSV字符串或者一個參數(shù)提供類傳入。
- 參數(shù)提供類可以擁有盡可能多的參數(shù)提供方法,這樣你可以給不同的用例進行分類。
- 你可以擁有可以提供參數(shù)的測試方法 (再也不需要外部類或者靜態(tài)類了)
- 你可以在你的集成開發(fā)工具中看到實際的參數(shù)值(而在JUnit的Parametrised里,只有連續(xù)數(shù)目的參數(shù))
Mockito
Java里單元測試的非常棒(tasty)的模擬框架:
//你可以模擬具體的類,而不只是接口
LinkedList mockedList = mock(LinkedList.class);
//打樁
when(mockedList.get(0)).thenReturn("first");
when(mockedList.get(1)).thenThrow(new RuntimeException());
//以下代碼打印出"first"字符串
System.out.println(mockedList.get(0));
//以下代碼拋出運行時異
System.out.println(mockedList.get(1));
//以下代碼打印出"null",因為get(999)沒有被打樁
System.out.println(mockedList.get(999));
//盡管是可以驗證一個打過樁的調(diào)用,但通常是多余的
//如果你的代碼關(guān)心get(0)返回值的內(nèi)容,那么其他東西就會中斷(往往在verify()執(zhí)行之前就發(fā)生了)。
//如果你的代碼不關(guān)心get(0)返回值的內(nèi)容,那么它就不應(yīng)該被打樁。不相信嗎?看看這里。
verify(mockedList).get(0);
Jukito
它結(jié)合了JUnit、Guice和Mockito的能力。 而且它還聽起來像一門很酷的武術(shù)。
- 極大的減少了諸如自動mock的樣板,從而使測試更加易讀。
- 可以使得測試能夠根據(jù)被測試的對象上的API的改變而彈性變化。
標(biāo)有@Inject注解的字段會被自動注入,
不需要擔(dān)心會遺忘掉它們- 使得將對象連接在一起變得容易,因此你可以將一個單元測試變成集成測試的一部分
@RunWith(JukitoRunner.class)
public class EmailSystemTest {
@Inject EmailSystemImpl emailSystem;
Email dummyEmail;
@Before
public void setupMocks( IncomingEmails incomingEmails, EmailFactory factory) {
dummyEmail = factory.createDummy();
when(incomingEmails.count()).thenReturn(1);
when(incomingEmails.get(0)).thenReturn(dummyEmail);
}
@Test
public void shouldFetchEmailWhenStarting( EmailView emailView) {
// WHEN
emailSystem.start();
// THEN
verify(emailView).addEmail(dummyEmail);
}
}
Awaitility
Awaitility是一個小型的Java領(lǐng)域?qū)S谜Z言(DSL),用于對異步的操作進行同步。
測試異步的系統(tǒng)是比較困難的。不僅需要處理線程、超時和并發(fā)問題,而且測試代碼的本來意圖也有可能被這些細節(jié)所蒙蔽。Awaitility是一個領(lǐng)域?qū)S谜Z言,可以允許你以一種簡潔且易讀的方式來表達異步系統(tǒng)的各種期望結(jié)果。
@Test
public void updatesCustomerStatus() throws Exception {
// Publish an asynchronous event:
publishEvent(updateCustomerStatusEvent);
// Awaitility lets you wait until the asynchronous operation completes:
await().atMost(5, SECONDS).until(customerStatusIsUpdated());
...
}
Spock
企業(yè)級的測試和規(guī)范框架。
class HelloSpockSpec extends spock.lang.Specification {
def "length of Spock's and his friends' names"() {
expect:
name.size() == length
where:
name | length
"Spock" | 5
"Kirk" | 4
"Scotty" | 6
}
}
WireMock
用于模擬HTTP服務(wù)的工具
- 對HTTP響應(yīng)進行打樁,可以匹配URL、header頭信息和body內(nèi)容的模式
- 請求驗證
- 在單元測試里運行,但是是作為一個對立的進程或者一個WAR應(yīng)用的形式
- 可通過流暢的Java API、JSON文件和基于HTTP的JSON進行配置
- 對stub的錄制/回放
- 故障注入
- 針對每個請求的根據(jù)條件進行代理
- 針對請求的檢查和替換進行瀏覽器的代理
- 有狀態(tài)的行為模擬
- 可配置的響應(yīng)延遲
{
"request": {
"method": "GET",
"url": "/some/thing"
},
"response": {
"status": 200,
"statusMessage": "Everything was just fine!"
}
}
感謝
非常感謝閱讀!