前言
你們有木有喜歡看程式碼的領導啊,我的領導就喜歡看我寫的程式碼,有事沒事就喜歡跟我探討怎麼寫才最好,哈哈哈...挺好。
今天我們就一起來看看可以節省 90% 的加班時間的第三方開源庫吧,第一個介紹的必須是 Apache 下的 Commons 庫。第二個是 google 開源的 Guava 庫。
Apache Commons
Apache Commons 是一個功能非常強大、經常被使用到的庫。它有 40 個左右的類庫,包含了對字串、日期、陣列等的操作。
Lang3
Lang3 是一個處理 Java 中基本物件的包,比如用 StringUtils 類操作字串、ArrayUtils 類運算元組、DateUtils 類可以處理日期、MutablePair 類可以返回多個欄位等等。
包結構:
maven 依賴
<dependency>
<groupId>org.apache.commons</groupId>
<artifactId>commons-lang3</artifactId>
<version>3.11</version>
</dependency>
字串操作
對字串快速操作,在 if else 的少寫判空條件。
public static void main(String[] args) {
boolean blank = StringUtils.isBlank(" ");//注意此處是null哦 這和isEmpty不一樣的
System.out.println(blank);
boolean empty = StringUtils.isEmpty(" ");//注意這裡是false
System.out.println(empty);
boolean anyBlank = StringUtils.isAnyBlank("a", " ", "c");// 其中一個是不是空字串
System.out.println(anyBlank);
boolean numeric = StringUtils.isNumeric("1");//字串是不是全是數字組成,"." 不算數字
System.out.println(numeric);
String remove = StringUtils.remove("abcdefgh", "a");//移除字串
System.out.println(remove);
}
輸出結果:
true
false
true
true
bcdefgh
Process finished with exit code 0
日期操作
終於可以不用 SimpleDateFormat 格式化日期了,DateUtils.iterator 可以獲取一段時間。
public static void main(String[] args) throws ParseException {
Date date = DateUtils.parseDate("2021-07-15", "yyyy-MM-dd");
Date date1 = DateUtils.addDays(date, 1);//加一天
System.out.println(date1);
boolean sameDay = DateUtils.isSameDay(date, new Date());//比較
System.out.println(sameDay);
/*
獲取一段日期
RANGE_WEEK_SUNDAY 從週日開始獲取一週日期
RANGE_WEEK_MONDAY 從週一開始獲取一週日期
RANGE_WEEK_RELATIVE 從當前時間開始獲取一週日期
RANGE_WEEK_CENTER 以當前日期為中心獲取一週日期
RANGE_MONTH_SUNDAY 從週日開始獲取一個月日期
RANGE_MONTH_MONDAY 從週一開始獲取一個月日期
*/
Iterator<Calendar> iterator = DateUtils.iterator(date, DateUtils.RANGE_WEEK_CENTER);
while (iterator.hasNext()) {
Calendar next = iterator.next();
System.out.println(DateFormatUtils.format(next, "yyyy-MM-dd"));
}
}
輸出結果:
Fri Jul 16 00:00:00 CST 2021
false
2021-07-12
2021-07-13
2021-07-14
2021-07-15
2021-07-16
2021-07-17
2021-07-18
Process finished with exit code 0
返回多個欄位
有時候在一個方法中需要返回多個值的時候,經常會使用 HashMap 返回或者是 JSON 返回。Lang3 下已經幫我們提供了這樣的工具類,不需要再多寫 HashMap 和 JSON 了。
public static void main(String[] args) {
MutablePair<Integer, String> mutablePair = MutablePair.of(2, "這是兩個值");
System.out.println(mutablePair.getLeft() + " " + mutablePair.getRight());
MutableTriple<Integer, String, Date> mutableTriple = MutableTriple.of(2, "這是三個值", new Date());
System.out.println(mutableTriple.getLeft() + " " + mutableTriple.getMiddle() + " " + mutableTriple.getRight());
}
輸出結果:
2 這是兩個值
2 這是三個值 Fri Jul 16 15:24:40 CST 2021
Process finished with exit code 0
ArrayUtils 陣列操作
ArrayUtils 是專門處理陣列的類,可以讓方便的處理陣列而不是需要各種迴圈操作。
public static void main(String[] args) {
//合併陣列
String[] array1 = new String[]{"value1", "value2"};
String[] array2 = new String[]{"value3", "value4"};
String[] array3 = ArrayUtils.addAll(array1, array2);
System.out.println("array3:"+ArrayUtils.toString(array3));
//clone 陣列
String[] array4 = ArrayUtils.clone(array3);
System.out.println("array4:"+ArrayUtils.toString(array4));
//陣列是否相同
boolean b = EqualsBuilder.reflectionEquals(array3, array4);
System.out.println(b);
//反轉陣列
ArrayUtils.reverse(array4);
System.out.println("array4反轉後:"+ArrayUtils.toString(array4));
//二維陣列轉 map
Map<String, String> arrayMap = (HashMap) ArrayUtils.toMap(new String[][]{
{"key1", "value1"}, {"key2", "value2"}
});
for (String s : arrayMap.keySet()) {
System.out.println(arrayMap.get(s));
}
}
輸出結果:
array3:{value1,value2,value3,value4}
array4:{value1,value2,value3,value4}
true
array4反轉後:{value4,value3,value2,value1}
value1
value2
Process finished with exit code 0
EnumUtils 列舉操作
- getEnum(Class enumClass, String enumName) 通過類返回一個列舉,可能返回空;
- getEnumList(Class enumClass) 通過類返回一個列舉集合;
- getEnumMap(Class enumClass) 通過類返回一個列舉map;
- isValidEnum(Class enumClass, String enumName) 驗證enumName是否在列舉中,返回true或false。
public enum ImagesTypeEnum {
JPG,JPEG,PNG,GIF;
}
public static void main(String[] args) {
ImagesTypeEnum imagesTypeEnum = EnumUtils.getEnum(ImagesTypeEnum.class, "JPG");
System.out.println("imagesTypeEnum = " + imagesTypeEnum);
System.out.println("--------------");
List<ImagesTypeEnum> imagesTypeEnumList = EnumUtils.getEnumList(ImagesTypeEnum.class);
imagesTypeEnumList.stream().forEach(
imagesTypeEnum1 -> System.out.println("imagesTypeEnum1 = " + imagesTypeEnum1)
);
System.out.println("--------------");
Map<String, ImagesTypeEnum> imagesTypeEnumMap = EnumUtils.getEnumMap(ImagesTypeEnum.class);
imagesTypeEnumMap.forEach((k, v) -> System.out.println("key:" + k + ",value:" + v));
System.out.println("-------------");
boolean result = EnumUtils.isValidEnum(ImagesTypeEnum.class, "JPG");
System.out.println("result = " + result);
boolean result1 = EnumUtils.isValidEnum(ImagesTypeEnum.class, null);
System.out.println("result1 = " + result1);
}
輸出結果:
imagesTypeEnum = JPG
--------------
imagesTypeEnum1 = JPG
imagesTypeEnum1 = JPEG
imagesTypeEnum1 = PNG
imagesTypeEnum1 = GIF
--------------
key:JPG,value:JPG
key:JPEG,value:JPEG
key:PNG,value:PNG
key:GIF,value:GIF
-------------
result = true
result1 = false
Process finished with exit code 0
collections4 集合操作
commons-collections4 增強了 Java 集合框架,提供了一系列簡單的 API 方便操作集合。
maven 依賴
<dependency>
<groupId>org.apache.commons</groupId>
<artifactId>commons-collections4</artifactId>
<version>4.4</version>
</dependency>
CollectionUtils 工具類
這是一個工具類,可以檢查 null 元素不被加入集合,合併列表,過濾列表,兩個列表的並集、差集、合集。有部分功能在 Java 8 中可以被 Stream API 替換。
public static void main(String[] args) {
//null 元素不能加進去
List<String> arrayList1 = new ArrayList<>();
arrayList1.add("a");
CollectionUtils.addIgnoreNull(arrayList1, null);
System.out.println(arrayList1.size());
//排好序的集合,合併後還是排序的
List<String> arrayList2 = new ArrayList<>();
arrayList2.add("a");
arrayList2.add("b");
List<String> arrayList3 = new ArrayList<>();
arrayList3.add("c");
arrayList3.add("d");
System.out.println("arrayList3:" + arrayList3);
List<String> arrayList4 = CollectionUtils.collate(arrayList2, arrayList3);
System.out.println("arrayList4:" + arrayList4);
//交集
Collection<String> strings = CollectionUtils.retainAll(arrayList4, arrayList3);
System.out.println("arrayList3和arrayList4的交集:" + strings);
//並集
Collection<String> union = CollectionUtils.union(arrayList4, arrayList3);
System.out.println("arrayList3和arrayList4的並集:" + union);
//差集
Collection<String> subtract = CollectionUtils.subtract(arrayList4, arrayList3);
System.out.println("arrayList3和arrayList4的差集:" + subtract);
// 過濾,只保留 b
CollectionUtils.filter(arrayList4, s -> s.equals("b"));
System.out.println(arrayList4);
}
輸出結果:
1
arrayList3:[c, d]
arrayList4:[a, b, c, d]
arrayList3和arrayList4的交集:[c, d]
arrayList3和arrayList4的並集:[a, b, c, d]
arrayList3和arrayList4的差集:[a, b]
[b]
Process finished with exit code 0
Bag 統計次數
用於統計值在集合中出現的次數。
public static void main(String[] args) {
Bag bag = new HashBag<String>();
bag.add("a");
bag.add("b");
bag.add("a");
bag.add("c", 3);
System.out.println(bag);
System.out.println(bag.getCount("c"));
}
輸出結果:
[2:a,1:b,3:c]
3
Process finished with exit code 0
beanutils Bean 操作
beanutils 是通過反射機制對 JavaBean 進行操作的。比如對 Bean 進行復制、map 轉物件、物件轉 Map。
maven 依賴
<dependency>
<groupId>commons-beanutils</groupId>
<artifactId>commons-beanutils</artifactId>
<version>1.9.4</version>
</dependency>
public class User {
private String name;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
public static void main(String[] args) throws Exception {
User user1 = new User();
user1.setName("李四");
User user2 = (User) BeanUtils.cloneBean(user1);
System.out.println(user2.getName());
//User 轉 map
Map<String, String> describe = BeanUtils.describe(user1);
System.out.println(describe);
//Map 轉 User
Map<String, String> beanMap = new HashMap();
beanMap.put("name", "張三");
User user3 = new User();
BeanUtils.populate(user3, beanMap);
System.out.println(user3.getName());
}
輸出結果:
李四
{name=李四}
張三
Process finished with exit code 0
Guava
Google 開源的一個基於 Java 擴充套件專案,包含了一些基本工具、集合擴充套件、快取、併發工具包、字串處理等。
maven 依賴
<dependency>
<groupId>com.google.guava</groupId>
<artifactId>guava</artifactId>
<version>30.1.1-jre</version>
</dependency>
Map<String, List> 型別
在java 程式碼中經常會遇到需要寫 Map<String, List> map 的區域性變數的時候。有時候業務情況還會更復雜一點。
public static void main(String[] args) {
//以前
Map<String, List<String>> map = new HashMap<>();
List<String> list = new ArrayList<>();
list.add("張三");
list.add("李四");
map.put("名稱", list);
System.out.println(map.get("名稱"));
//現在
Multimap<String, String> multimap = ArrayListMultimap.create();
multimap.put("名稱", "張三");
multimap.put("名稱", "李四");
System.out.println(multimap.get("名稱"));
}
輸出結果:
[張三, 李四]
[張三, 李四]
Process finished with exit code 0
value 不能重複的 Map
在 Map 中 value 的值時可以重複的,Guava 可以建立一個 value 不可重複的 Map,並且 Map 和 value 可以對調。
public static void main(String[] args) {
//會報異常
BiMap<String ,String> biMap = HashBiMap.create();
biMap.put("key1", "value");
biMap.put("key2", "value");
System.out.println(biMap.get("key1"));
}
輸出結果:
Exception in thread "main" java.lang.IllegalArgumentException: value already present: value
at com.google.common.collect.HashBiMap.put(HashBiMap.java:287)
at com.google.common.collect.HashBiMap.put(HashBiMap.java:262)
at org.example.clone.Test.main(Test.java:17)
Process finished with exit code 1
public static void main(String[] args) {
BiMap<String ,String> biMap = HashBiMap.create();
biMap.put("key1", "value1");
biMap.put("key2", "value2");
System.out.println(biMap.get("key1"));
//key-value 對調
biMap = biMap.inverse();
System.out.println(biMap.get("value1"));
}
輸出結果:
value1
key1
Process finished with exit code 0
Guava cache
寫業務的時候肯定會使用快取,當不想用第三方作為快取的時候,Map 又不夠強大,就可以使用 Guava 的快取。
快取的併發級別
Guava提供了設定併發級別的API
,使得快取支援併發的寫入和讀取。與ConcurrentHashMap
類似,Guava cache的併發也是通過分離鎖實現。在通常情況下,推薦將併發級別設定為伺服器cpu核心數。
CacheBuilder.newBuilder()
// 設定併發級別為cpu核心數,預設為4
.concurrencyLevel(Runtime.getRuntime().availableProcessors())
.build();
快取的初始容量設定
我們在構建快取時可以為快取設定一個合理大小初始容量,由於Guava的快取使用了分離鎖的機制,擴容的代價非常昂貴。所以合理的初始容量能夠減少快取容器的擴容次數。
CacheBuilder.newBuilder()
// 設定初始容量為100
.initialCapacity(100)
.build();
設定最大儲存
Guava Cache可以在構建快取物件時指定快取所能夠儲存的最大記錄數量。當Cache中的記錄數量達到最大值後再呼叫put方法向其中新增物件,Guava會先從當前快取的物件記錄中選擇一條刪除掉,騰出空間後再將新的物件儲存到Cache中。
public static void main(String[] args) {
Cache<String, String> cache = CacheBuilder.newBuilder().maximumSize(2).build();
cache.put("key1", "value1");
cache.put("key2", "value2");
cache.put("key3", "value3");
System.out.println(cache.getIfPresent("key1")); //key1 = null
}
輸出結果:
null
Process finished with exit code 0
過期時間
expireAfterAccess() 可以設定快取的過期時間。
public static void main(String[] args) throws InterruptedException {
//設定過期時間為2秒
Cache<String, String> cache1 = CacheBuilder.newBuilder().maximumSize(2).expireAfterAccess(2, TimeUnit.SECONDS).build();
cache1.put("key1", "value1");
Thread.sleep(1000);
System.out.println(cache1.getIfPresent("key1"));
Thread.sleep(2000);
System.out.println(cache1.getIfPresent("key1"));
}
輸出結果:
value1
null
Process finished with exit code 0
LoadingCache
使用自定義ClassLoader
載入資料,置入記憶體中。從LoadingCache
中獲取資料時,若資料存在則直接返回;若資料不存在,則根據ClassLoader
的load
方法載入資料至記憶體,然後返回該資料。
public class Test {
public static void main(String[] args) throws Exception {
System.out.println(numCache.get(1));
Thread.sleep(1000);
System.out.println(numCache.get(1));
Thread.sleep(1000);
numCache.put(1, 6);
System.out.println(numCache.get(1));
}
private static LoadingCache<Integer, Integer> numCache = CacheBuilder.newBuilder().
expireAfterWrite(5L, TimeUnit.MINUTES).
maximumSize(5000L).
build(new CacheLoader<Integer, Integer>() {
@Override
public Integer load(Integer key) throws Exception {
System.out.println("no cache");
return key * 5;
}
});
}
輸出結果:
no cache
5
5
6
Process finished with exit code 0
總結
通過 Apache Commons 和 Guava 兩個第三方的開源工具庫,可以減少迴圈、ifelse 的程式碼。寫出的程式碼更有健壯性並且可以在新人面前裝一波。Apache Commons 和 Guava 有許許多多的工具類,這裡只列出了小小的部分,可以在官網例子中檢視到各種用法。
最後
我是一個正在被打擊還在努力前進的碼農。如果文章對你有幫助,記得點贊、關注喲,謝謝!