guava 笔记

[toc]

guava: google 的开源工具包

帮助编写常用方法

hashCode,equals,toString,compareTo方法;前三个idea 提供了guava方式的代码生成

import com.google.common.base.MoreObjects;
import com.google.common.collect.ComparisonChain;

public class User implements Comparable<User>{
    private String name;
    private int age;

//omit setting and getting

    @Override
    public boolean equals(Object o) {
        if (this == o) return true;
        if (o == null || getClass() != o.getClass()) return false;
        User user = (User) o;
        return age == user.age &&
                com.google.common.base.Objects.equal(name, user.name);
    }

    @Override
    public int hashCode() {
        return com.google.common.base.Objects.hashCode(name, age);
    }

    @Override
    public String toString() {
        return MoreObjects.toStringHelper(this)
                .add("name", name)
                .add("age", age)
                .toString();
    }

    @Override
    public int compareTo(User other) {
        return  ComparisonChain.start().
                    compare(this.name,other.name).
                    compare(this.age,other.age).
                    result();
    }
}

检查方法调用的前置条件

一般尽快失败来拒绝不符合条件的调用

    private void addAge(User user,int age){
        //检查非空, throw NullPointerException
        Preconditions.checkNotNull(user,"%s must not be null","user");
        //检查参数,throw IllegalArgumentException
        Preconditions.checkArgument(age>0,"the %s should greater than 0; now: %s","age",age);
        //检查对象的状态,throw IllegalStateException
        Preconditions.checkState(user.getAge()>10,"the age of %s should larger than 10; now: %s",user.getName(),2);

        //检查index作为索引值对某个列表、字符串或数组是否有效。 throw IndexOutOfBoundsException
        List<String> list = new ArrayList<>();
        int size =  list.size();
        for (int i = 0; i < size; i++) {
            Preconditions.checkElementIndex(i,size);
            Preconditions.checkPositionIndex(i,size);
        }
    }

集合

支持空元素集合 不支持空元素集合
JDK 中的集合很多都支持空元素(ArrayList,LinkedList, HashSet, HashMap) EnumMap,ConcurrentHashMap 和 guava中的集合大多不支持空元素

Enum,Tree类型支持不支持看compare方法,TreeSet不支持空元素。TreeMap支持空元素。
最好严格控制不存null,防止出问题

转换成不可变集合

guava方式 比 Collections.unmodifiableXXX方法 更高效和安全;
最好使用 static final 显示声明,更明显

//使用of 方法直接构建不可变集合
public static final ImmutableSet<String> COLOR_NAMES = ImmutableSet.of(
        "red","orange","yellow","green","blue","purple");

//使用copyOf 转换成不可变集合
ImmutableList<String> immutableList = ImmutableList.copyOf(Collection);
ImmutableSet immutableSet = ImmutableSet.copyOf(Collection);
ImmutableMap<Integer,Integer> immutableMap = ImmutableMap.copyOf(Map);
jdk可变集合接口 guava中不可变版本
Collection ImmutableCollection
List ImmutableList
Set ImmutableSet
SortedSet/NavigableSet ImmutableSortedSet
Map ImmutableMap
SortedMap ImmutableSortedMap

新集合类型

Multiset

可以多次添加相等的元素 统计出现的次数;

    Multiset<String> set = HashMultiset.create();
    //Multiset<String> set = TreeMultiset.create();
    //Multiset<String> set = LinkedHashMultiset.create();
    //Multiset<String> set = ConcurrentHashMultiset.create();
    //Multiset<String> set = ImmutableMultiset.copyOf(set);
    set.add("a");
    set.add("b");
    set.setCount("c",101);
    System.out.println(set.count("c"));  //101
    System.out.println(set.size());      //103

Multimap

允许key重复

    //注意Multimap<K, V>不是Map<K,Collection<V>>,里面的key是有重复的,但entry(key & value)不能都一样
    Multimap multimap = HashMultimap.create();
    //Multimap multimap = TreeMultimap.create();
    //Multimap multimap = LinkedListMultimap.create();
    //ImmutableMultimap.copyOf(multimap);
    multimap.put("1","2");
    multimap.put("4","22");
    multimap.put("13",null);
    multimap.put("13","23");
    multimap.put("13","23");
    multimap.put("2","2");
    System.out.println(multimap.get("13").size()); //2   

BiMap

key和value都唯一,双向绑定,可以反转 key-value

    BiMap<String,Integer> bitMap = HashBiMap.create();
    //TreeMultimap.create();
    //BiMap<DemoEnum,OtherEnum> bitMap2 = EnumBiMap.create(DemoEnum.class,OtherEnum.class);
    bitMap.put("a",97);
    bitMap.put("b",98);
    bitMap.put("c",99);
    BiMap<Integer,String> inverseMap = bitMap.inverse();
    System.out.println(bitMap.get("a"));    //97
    System.out.println(inverseMap.get(97)); //'a'    

table

用"列"和"行" 组合做key

    //泛型参数分别是:列,行,值
    HashBasedTable<Integer,Integer,String> table = HashBasedTable.create();
    //TreeBasedTable.create();
    //ImmutableTable.copyOf(table);
    table.put(1,1,"1");
    table.put(1,2,"2");
    table.put(1,3,"3");
    table.put(2,1,"4");
    table.put(2,2,"5");
    table.put(2,3,"6");
    Map<Integer,String> map1 = table.row(2);    //检索第二列 {1=4, 2=5, 3=6}
    Map<Integer,String> map2 = table.column(1); //检索第一行 {1=1, 2=4}
    String value = table.get(1,3); //3    

ClassToInstanceMap

它的键是类型,而值是符合键所指类型的对象

    //泛型参数代表Map支持的类型的上界
    MutableClassToInstanceMap<Number> map =  MutableClassToInstanceMap.create();
    //ImmutableClassToInstanceMap.copyOf(map);
    map.put(Integer.class,1);
    map.put(Long.class,101L);
    map.put(BigDecimal.class,BigDecimal.ONE);
    map.put(short.class,Short.MAX_VALUE);
    System.out.println(map.getInstance(short.class));  //32767

RangeSet,TreeRangeMap

不相连的、非空的区间

    //使用Comparable 比较
    //RangeSet会合并区间,RangeMap不会合并区间
    RangeSet<Integer> rangeSet = TreeRangeSet.create();
    rangeSet.add(Range.closedOpen(0,10));
    rangeSet.add(Range.closed(10,20));
    rangeSet.add(Range.openClosed(21,30));
    System.out.println(rangeSet.toString()); //[[0..20], (21..30]
    
    TreeRangeMap<String,String>  treeRangeMap = TreeRangeMap.create();
    treeRangeMap.put(Range.closed("a","f"),"a");
    treeRangeMap.put(Range.closed("h","z"),"b");
    treeRangeMap.remove(Range.closed("c","d"));
    System.out.println(treeRangeMap.toString()); //[[a..c)=a, (d..f]=a, [h..o]=b]    

字符串处理

Joiner Splitter

连接

Joiner joiner = Joiner.on(":").skipNulls(); //skipNulls()方法是直接忽略null,否者null报错
//Joiner.on(":").useForNull("defalut value");  useForNull(String)方法可以给定某个字符串来替换null
String str = joiner.join("A", null, "B", "C");
System.out.print(str); //A:B:C

分割

//jdk
    String[] arr = ",a,,b ,".split(","); //结果:"", "a","", "b " 只有尾部的空字符串被忽略了。

//guava
    //支持字符串,正则,长度,CharMatcher分割方式
    //限制十个, 忽略空元素, 对分割后的每个元素进行trim
    Iterable<String> iterable = Splitter.on(",").limit(10).omitEmptyStrings().trimResults().split(",a,,b ,"); //结果:"a","b"
    Splitter.fixedLength(3).split("aaabbbcccdd"); //按固定长度拆分;最后一段可能比给定长度短,但不会为空。
    Splitter.onPattern("\\d").split("abc123de"); //正则分割

Joiner Splitter 类似BigDecimal都是不可变的

Joiner joiner = Joiner.on(',');
joiner.skipNulls(); // does nothing!

参考

https://github.com/google/guava
http://ifeve.com/google-guava/

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,491评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,856评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,745评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,196评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,073评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,112评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,531评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,215评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,485评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,578评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,356评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,215评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,583评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,898评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,174评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,497评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,697评论 2 335

推荐阅读更多精彩内容