常用的 Java 工具类之 Guava 使用

上篇我们介绍 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));
复制代码

参考文档

  1. Guava 中文翻译
  2. Guava wiki
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值