上篇我们介绍 Apache 的相关工具类的使用,这次我们介绍 Google 的工具类 Guava。
The Guava project contains several of Google's core libraries that we rely on in our Java-based projects: collections, caching, primitives support, concurrency libraries, common annotations, string processing, I/O, and so forth. Each of these tools really do get used every day by Googlers, in production services.
正如 Guava wiki 介绍,Guava 的是 Google 的工具类,提供字符串处理,I/O,集合类相关操作。如果你去翻看 JDK8 的代码,可以看到其很多新增类,Guava 之前其实已经实现,很多类名也都一样,可以看出 Guava 的影响。下面开始介绍 Guava 的使用。
Guava Collections
Guava 为集合类提供相关工具类以及一些集合的新类型,方便我们日常开发。
集合新工具类- Lists
Lists.newArrayList()
使用静态方法直接创建 List 集合。
// 在JDK 7之前,构造新的集合时我们需要重复声明范型
List<String> testList = new ArrayList<String>();
// 使用 Lists 之后我们可以使用静态方法自动推导
List<String> testList1 = Lists.newArrayList();
// 如果需要指定初始大小
List<String> capicatyList = Lists.newArrayListWithCapacity(100);
复制代码
Lists.partition()
按照指定大小拆分大集合
// 定义一个 10W 数量的集合
List<String> largeList = Lists.newArrayListWithCapacity(100 * 1000);
// 拆分成 100 个集合数量 1000 的大小的小集合
List<List<String>> smallLists = Lists.partition(largeList, 1000);
复制代码
Lists.transform
将集合数据类型转化成其他类型的集合
List<String> largeList = Lists.newArrayListWithCapacity(100 * 1000);
List<List<String>> smallLists = Lists.partition(largeList, 1000);
// function 方法可以执行转化动作,将类型 A 的对象 转化为 B 对象
List<Integer> a = Lists.transform(largeList, new Function<String, Integer>() {
@Override
public Integer apply(String input) {
return input.hashCode() + 111;
}
});
复制代码
但是使用这个类的是需要注意一点,转化之后的集合,不再支持 add 等修改集合的方法。如果使用遍历方法去修改集合里面对象,并不会产生效果。原因在于转化之后的 List 只是原 List 的视图而已,所以获取转化后集合里的对象,比如调用 get 时,每次调用会是会使用 function 转化方法。
Returns a list that applies function to each element of fromList. The returned list is a transformed view of fromList; changes to fromList will be reflected in the returned list and vice versa
Guava transform 问题
新集合类
Multiset 如果光看类名的话,可能会以为其只是实现 Set 接口的新类。其实不然,这个类实现是类似以下功能:
Map<String, Integer> counts = new HashMap<String, Integer>();
for (String word : words) {
Integer count = counts.get(word);
if (count == null) {
counts.put(word, 1);
} else {
counts.put(word, count + 1);
}
}
复制代码
如果存在上述需求,我们不再需要自己笨拙且容易出错的代码。现在我们可以看如果使用 Multiset 实现上述代码。
Multiset<String> multiset = HashMultiset.create();
// 统计每次单词出现的次数
for (String word : words) {
multiset.add(word);
}
// 输出单词的输出次数
for (String word : multiset.elementSet()) {
multiset.count(word);
}
复制代码
有时候我们需要实现一个键与多个值映射的功能,这时候我们可能会使用 Map<K, List<V>>
,但是这种方法可能在实现中不小心就导致各种名空指针异常等。下面我们使用 Multimap
实现这种需求。
Multimap<Integer, People> multimap = ArrayListMultimap.create();
// 统计同一年纪的 People 对象
for (People people : peopleList) {
multimap.put(people.getAge(), people);
}
// 输出统计结果
for (Integer key : multimap.keySet()) {
List<People> peoples = Lists.newArrayList(multimap.get(key));
System.out.println(peoples);
}
复制代码
字符串处理
Joiner
实现字符串连接功能。
// 实现的连接器将忽略 Null
Joiner joiner = Joiner.on(";").skipNulls();
// 字符串为Harry;Ron;Hermione
return joiner.join("Harry", null, "Ron", "Hermione");
复制代码
Splitter
字符串拆分器,实现字符串按照指定字符拆分功能
// 以下拆分将将结果去除首部与尾部空格,以及去除空字符串。所以拆分之后结果为 foo bar qux
Splitter.on(',')
.trimResults()
.omitEmptyStrings()
.split("foo,bar,, qux");
复制代码
CharMatcher
字符串匹配器,实现匹配筛选等动作。
String string = "12312ada[21`443#$$#@ asda";
System.out.println(CharMatcher.DIGIT.retainFrom(string));//只保留数字 1231221443
System.out.println(CharMatcher.JAVA_LETTER_OR_DIGIT.replaceFrom(string, "*")); // 数字与字母使用*号代替
System.out.println(CharMatcher.DIGIT.matchesAllOf(string));//字符串是否都为数字
复制代码
Stopwatch
有时候我们需要知道方法的调用时间,这个时候我们往往使用下列代码:
long startTime = System.currentTimeMillis();
// 业务方法调用
long endTime = System.currentTimeMillis();
long totalTime = endTime - startTime;
复制代码
这种写法如果需要转化时间单位我们还需要经过各种换算,如果需要计算多个调用时间,时间定义往往比较杂乱。我们可以使用 Stopwatch
代替。
Stopwatch stopwatch = Stopwatch.createUnstarted();
// 开始计量时间
stopwatch.start();
try {
TimeUnit.SECONDS.sleep(10);
} catch (InterruptedException e) {
e.printStackTrace();
}
// 停止计量时间
stopwatch.stop();
// 根据输入时间单位获取相应的时间
System.out.println(stopwatch.elapsed(TimeUnit.SECONDS));
复制代码