Guava——Immutable Collections

不可变集合:ImmutableSet

public static final ImmutableSet<String> COLOR_NAMES = ImmutableSet.of(

  "red",

  "orange",

  "yellow",

  "green",

  "blue",

  "purple");


class Foo {

  final ImmutableSet<Bar> bars;

  Foo(Set<Bar> bars) {

    this.bars = ImmutableSet.copyOf(bars); // defensive copy!

  }

}

对不可靠的客户代码库来说,它使用安全,可以在未受信任的类库中安全的使用这些对象

线程安全的:immutable对象在多线程下安全,没有竞态条件

不需要支持可变性, 可以尽量节省空间和时间的开销. 所有的不可变集合实现都比可变集合更加有效的利用内存 (analysis)

可以被使用为一个常量,并且期望在未来也是保持不变的

构造方法: 都是采用静态方法

ImmutableSet.of()  来创建,参数可以有多个。当参数多余1个时会调用construct方法。

public static <E> ImmutableSet<E> of() {

    return EmptyImmutableSet.INSTANCE;

}


public static <E> ImmutableSet<E> of(E element) {

    return new SingletonImmutableSet(element);

}


public static <E> ImmutableSet<E> of(E e1, E e2) {

    return construct(2, e1, e2);

}


public static <E> ImmutableSet<E> of(E e1, E e2, E e3) {

    return construct(3, e1, e2, e3);

}


public static <E> ImmutableSet<E> of(E e1, E e2, E e3, E e4) {

    return construct(4, e1, e2, e3, e4);

}


public static <E> ImmutableSet<E> of(E e1, E e2, E e3, E e4, E e5) {

    return construct(5, e1, e2, e3, e4, e5);

}


public static <E> ImmutableSet<E> of(E e1, E e2, E e3, E e4, E e5, E e6, E... others) {

    int paramCount = true;

    Object[] elements = new Object[6 + others.length];

    elements[0] = e1;

    elements[1] = e2;

    elements[2] = e3;

    elements[3] = e4;

    elements[4] = e5;

    elements[5] = e6;

    System.arraycopy(others, 0, elements, 6, others.length);

    return construct(elements.length, elements);

}

construct

当元素超过1个时调用默认方法。

private static <E> ImmutableSet<E> construct(int n, Object... elements) {

    switch(n) {

    case 0:

        return of();

    case 1:

        E elem = elements[0];

        return of(elem);

    default:

        int tableSize = chooseTableSize(n);

        Object[] table = new Object[tableSize]; //创建一个Object的数组

        int mask = tableSize - 1;

        int hashCode = 0;

        int uniques = 0;

        int i = 0;


        for(; i < n; ++i) {

            Object element = ObjectArrays.checkElementNotNull(elements[i], i);

            int hash = element.hashCode();

            int j = Hashing.smear(hash);


            while(true) {

                int index = j & mask;

                Object value = table[index];

                if (value == null) {

                    elements[uniques++] = element;

                    table[index] = element;

                    hashCode += hash;

                    break;

                }


                if (value.equals(element)) {

                    break;

                }


                ++j;

            }

        }


        Arrays.fill(elements, uniques, n, (Object)null);

        if (uniques == 1) {

            E element = elements[0];

            return new SingletonImmutableSet(element, hashCode);

        } else if (tableSize != chooseTableSize(uniques)) {

            return construct(uniques, elements);

        } else {

            Object[] uniqueElements = uniques < elements.length ? ObjectArrays.arraysCopyOf(elements, uniques) : elements;

            return new RegularImmutableSet(uniqueElements, hashCode, table, mask);

        }

    }

}

会根据n确定参数setSzie的大小,当超出1073741824 会报错。

static int chooseTableSize(int setSize) {

    if (setSize >= 751619276) {

        Preconditions.checkArgument(setSize < 1073741824, "collection too large");

        return 1073741824;

    } else {

        int tableSize;

        for(tableSize = Integer.highestOneBit(setSize - 1) << 1; (double)tableSize * 0.7D < (double)setSize; tableSize <<= 1) {

            ;

        }

        return tableSize;

    }

}

复制

public static <E> ImmutableSet<E> copyOf(E[] elements) {

    switch(elements.length) {

    case 0:

        return of();

    case 1:

        return of(elements[0]);

    default:

        return construct(elements.length, (Object[])elements.clone());//并不是对elements的更改 而是克隆

    }

}

//还是会调用of()函数创建


public static  ImmutableSet copyOf(Iterable

    return elements instanceof Collection ? copyOf((Collection)elements) : copyOf(elements.iterator());

}


public static  ImmutableSet copyOf(Iterator

    if (!elements.hasNext()) {

        return of();

    } else {

        E first = elements.next();

        return !elements.hasNext() ? of(first) : (new ImmutableSet.Builder()).add(first).addAll(elements).build();

    }

}


public static  ImmutableSet copyOf(Collection

    if (elements instanceof ImmutableSet && !(elements instanceof ImmutableSortedSet)) {

        ImmutableSet<E> set = (ImmutableSet)elements;

        if (!set.isPartialView()) {

            return set;

        }

    } else if (elements instanceof EnumSet) {

        return copyOfEnumSet((EnumSet)elements);

    }


    Object[] array = elements.toArray();

    return construct(array.length, array);

}


private static 

    return ImmutableEnumSet.asImmutable(EnumSet.copyOf(enumSet));

}

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

推荐阅读更多精彩内容