2017年你不能錯(cuò)過的Java類庫(kù)
各位讀者好,這篇文章是在我看過 Andres Almiray 的一篇介紹文后,整理出來的。因?yàn)閮?nèi)容非常好,我便將它整理成參考列表分享給大家, 同時(shí)附上各個(gè)庫(kù)的特性簡(jiǎn)介和示例。請(qǐng)欣賞!
Guice
Guice (發(fā)音同 ‘juice’) ,是一個(gè) Google 開發(fā)的輕量級(jí)依賴性注入框架,適合 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ò)連接的途徑,也是我們進(jìn)行數(shù)據(jù)和媒體交換的工具。高效使用HTTP能使你的東西加載更快,并節(jié)省帶寬。
OkHttp是一個(gè)非常高效的HTTP客戶端,默認(rèn)情況下:
- 支持HTTP/2,允許對(duì)同一主機(jī)的請(qǐng)求共用一個(gè)套接字。
- 如果HTTP/2 不可用,連接池會(huì)減少請(qǐng)求延遲。
- 透明的GZIP可以減少下載流量。
- 響應(yīng)的緩存避免了重復(fù)的網(wǎng)絡(luò)請(qǐng)求。
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 類實(shí)現(xiàn) GitHubService 接口:
Retrofit retrofit = new Retrofit.Builder()
.baseUrl("https://api.github.com/")
.build();
GitHubService service = retrofit.create(GitHubService.class);
來自 GitHubService 的每個(gè) Call 都能產(chǎn)生為遠(yuǎn)程 Web 服務(wù)產(chǎn)生一個(gè)異步或同步 HTTP 請(qǐng)求:
Call<List<Repo>> repos = service.listRepos("octocat");
JDeferred
與JQuery類似的Java Deferred/Promise類庫(kù)
- Deferred 對(duì)象和 Promise
- Promise 回調(diào):
.then(…),.done(…),.fail(…),.progress(…),.always(…) - 支持多個(gè)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)式編程擴(kuò)展 – 是一個(gè)為Java虛擬機(jī)編寫的使用可觀察序列的構(gòu)建異步的基于事件的程序的類庫(kù)。
它基于觀察者模式實(shí)現(xiàn)對(duì)數(shù)據(jù)/事件的序列的支持,并添加了一些操作符,允許你以聲明式構(gòu)建序列, 使得開發(fā)者無需關(guān)心底層的線程、同步、線程安全和并發(fā)數(shù)據(jù)結(jié)構(gòu)。
RxJava最常見的一個(gè)用法就是在后臺(tái)線程運(yùn)行一些計(jì)算和網(wǎng)絡(luò)請(qǐng)求,而在UI線程顯示結(jié)果(或者錯(cuò)誤):
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是一個(gè)實(shí)現(xiàn)了發(fā)布-訂閱模式的輕量級(jí)的,高性能的事件總線。它易于使用,并力求功能豐富,易于擴(kuò)展,而同時(shí)又保證資源的高效利用和高性能。
MBassador的高性能的核心是一個(gè)專業(yè)的數(shù)據(jù)結(jié)構(gòu),它提供了非阻塞的讀取器,并最小化寫入器的鎖爭(zhēng)用,因此并發(fā)讀寫訪問的性能衰減會(huì)是最小的。
- 注解驅(qū)動(dòng)的
- 提供任何東西,慎重對(duì)待類型層次結(jié)構(gòu)
- 同步和異步的消息傳遞
- 可配置的引用類型
- 消息過濾
- 封裝的消息
- 處理器的優(yōu)先級(jí)
- 自定義錯(cuò)誤處理
- 可擴(kuò)展性
// 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項(xiàng)目
使用注解來減少Java中的重復(fù)代碼,比如getter,setters,非空檢查,生成的Builder等。
- val - 總算有了!無憂的final本地變量。
- @NonNull - 或:我如何學(xué)會(huì)不再擔(dān)心并愛上了非空異常(NullPointerException)。
- @Cleanup - 自動(dòng)的資源管理:安全調(diào)用你的close() 方法,無需任何麻煩。
- @Getter / @Setter - 再也不用寫
public int getFoo() {return foo;}了。 - @ToString - 無需啟動(dòng)調(diào)試器來檢查你的字段:就讓Lombok來為你生成一個(gè)toString方法吧!
- @EqualsAndHashCode - 實(shí)現(xiàn)相等的判斷變得容易了:它會(huì)從你的對(duì)象的字段里為你生成hashCode和equals方法的實(shí)現(xiàn)。
- @NoArgsConstructor, @RequiredArgsConstructor and @AllArgsConstructor - 定做構(gòu)造函數(shù):為你生成各種各樣的構(gòu)造函數(shù),包括無參的,每一個(gè)final或非空的字段作為一個(gè)參數(shù)的,或者每一個(gè)字段都作為參數(shù)的。
- @Data - 所有的都同時(shí)生成:這是一個(gè)快捷方式,可以為所有字段生成
@ToString,@EqualsAndHashCode,@Getter注解,以及為所有非final的字段生成@Setter注解,以及生成@RequiredArgsConstructor! - @Value - 聲明一個(gè)不可變類變得非常容易。
- @Builder - … 而且鮑伯是你叔叔:創(chuàng)建對(duì)象的無爭(zhēng)議且奢華的接口!
- @SneakyThrows - 在以前沒有人拋出檢查型異常的地方大膽的拋出吧!
- @Synchronized - 正確的實(shí)現(xiàn)同步:不要暴露你的鎖。
- @Getter(lazy=true) 懶惰是一種美德!
- @Log - 船長(zhǎng)日志,星歷24435.7: “那一行又是什么呢?”
Java簡(jiǎn)單日志門面(SLF4J)
Java簡(jiǎn)單日志門面 (SLF4J) 為不同的日志框架(比如java.util.logging, logback, log4j)提供了簡(jiǎn)單的門面或者抽象的實(shí)現(xiàn),允許最終用戶在部署時(shí)能夠接入自己想要使用的日志框架。
簡(jiǎn)言之,類庫(kù)和其他嵌入式的組件都應(yīng)該考慮采用SLF4J作為他們的日志需求,因?yàn)轭悗?kù)無法將它們對(duì)日志框架的選擇強(qiáng)加給最終用戶。另一方面,對(duì)于獨(dú)立的應(yīng)用來說,就不一定需要使用SLF4J。獨(dú)立應(yīng)用可以直接調(diào)用他們自己選擇的日志框架。而對(duì)于logback來說,這個(gè)問題是沒有意義的,因?yàn)?code>logback是通過SLF4J來暴露其日志接口的。
JUnitParams
對(duì)測(cè)試進(jìn)行參數(shù)化,還不錯(cuò)
@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ù)化運(yùn)行器的區(qū)別如下:
- 更明確 – 參數(shù)實(shí)在測(cè)試方法的參數(shù)中,而不是在類的字段中
- 更少的代碼 – 你不需要用構(gòu)造函數(shù)來設(shè)置參數(shù)
- 你可以在同一個(gè)類混合使用參數(shù)化和非參數(shù)化的方法。
- 參數(shù)可以通過一個(gè)CSV字符串或者一個(gè)參數(shù)提供類傳入。
- 參數(shù)提供類可以擁有盡可能多的參數(shù)提供方法,這樣你可以給不同的用例進(jìn)行分類。
- 你可以擁有可以提供參數(shù)的測(cè)試方法 (再也不需要外部類或者靜態(tài)類了)
- 你可以在你的集成開發(fā)工具中看到實(shí)際的參數(shù)值(而在JUnit的Parametrised里,只有連續(xù)數(shù)目的參數(shù))
Mockito
Java里單元測(cè)試的非常棒(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));
//以下代碼拋出運(yùn)行時(shí)異
System.out.println(mockedList.get(1));
//以下代碼打印出"null",因?yàn)間et(999)沒有被打樁
System.out.println(mockedList.get(999));
//盡管是可以驗(yàn)證一個(gè)打過樁的調(diào)用,但通常是多余的
//如果你的代碼關(guān)心get(0)返回值的內(nèi)容,那么其他東西就會(huì)中斷(往往在verify()執(zhí)行之前就發(fā)生了)。
//如果你的代碼不關(guān)心get(0)返回值的內(nèi)容,那么它就不應(yīng)該被打樁。不相信嗎?看看這里。
verify(mockedList).get(0);
Jukito
它結(jié)合了JUnit、Guice和Mockito的能力。 而且它還聽起來像一門很酷的武術(shù)。
- 極大的減少了諸如自動(dòng)mock的樣板,從而使測(cè)試更加易讀。
- 可以使得測(cè)試能夠根據(jù)被測(cè)試的對(duì)象上的API的改變而彈性變化。
標(biāo)有@Inject注解的字段會(huì)被自動(dòng)注入,不需要擔(dān)心會(huì)遺忘掉它們- 使得將對(duì)象連接在一起變得容易,因此你可以將一個(gè)單元測(cè)試變成集成測(cè)試的一部分
@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是一個(gè)小型的Java領(lǐng)域?qū)S谜Z言(DSL),用于對(duì)異步的操作進(jìn)行同步。
測(cè)試異步的系統(tǒng)是比較困難的。不僅需要處理線程、超時(shí)和并發(fā)問題,而且測(cè)試代碼的本來意圖也有可能被這些細(xì)節(jié)所蒙蔽。Awaitility是一個(gè)領(lǐng)域?qū)S谜Z言,可以允許你以一種簡(jiǎn)潔且易讀的方式來表達(dá)異步系統(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è)級(jí)的測(cè)試和規(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ù)的工具
- 對(duì)HTTP響應(yīng)進(jìn)行打樁,可以匹配URL、header頭信息和body內(nèi)容的模式
- 請(qǐng)求驗(yàn)證
- 在單元測(cè)試?yán)镞\(yùn)行,但是是作為一個(gè)對(duì)立的進(jìn)程或者一個(gè)WAR應(yīng)用的形式
- 可通過流暢的Java API、JSON文件和基于HTTP的JSON進(jìn)行配置
- 對(duì)stub的錄制/回放
- 故障注入
- 針對(duì)每個(gè)請(qǐng)求的根據(jù)條件進(jìn)行代理
- 針對(duì)請(qǐng)求的檢查和替換進(jìn)行瀏覽器的代理
- 有狀態(tài)的行為模擬
- 可配置的響應(yīng)延遲
{
"request": {
"method": "GET",
"url": "/some/thing"
},
"response": {
"status": 200,
"statusMessage": "Everything was just fine!"
}
}
感謝
非常感謝閱讀!

































