成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

【J2SE】java編程思想之?dāng)?shù)組與集合學(xué)習(xí)總結(jié)

zhangxiangliang / 3562人閱讀

摘要:將散列碼作為數(shù)組的下標(biāo),該數(shù)組的一個(gè)單元指向一個(gè)鏈表,鏈表上的一個(gè)節(jié)點(diǎn)就是一個(gè)對(duì)象。散列碼不必獨(dú)一無二,應(yīng)關(guān)注生成速度。好的應(yīng)該產(chǎn)生分布均勻的散列碼。

數(shù)組 簡(jiǎn)述

數(shù)組是一種效率最高的存儲(chǔ)和隨機(jī)訪問對(duì)象引用的一個(gè)簡(jiǎn)單的線性序列,雖然訪問快速,但為之付出的代價(jià)是數(shù)組的大小固定,并且在其生命周期中不可改變。數(shù)組與其他容器之間的區(qū)別在于:效率、類型和保存基本類型的能力。但隨著自動(dòng)包裝機(jī)制的出現(xiàn),容器已經(jīng)可以與數(shù)組幾乎一樣方便,而數(shù)組僅存的優(yōu)點(diǎn)就是效率。

應(yīng)該 “優(yōu)先選擇容器而不是數(shù)組”,只有在已經(jīng)證明性能稱為問題(并且切換到數(shù)組對(duì)性能提高有所幫助)時(shí),你才應(yīng)該將程序重構(gòu)為使用數(shù)組

數(shù)組標(biāo)識(shí)符就是一個(gè)引用,用以保存指向其他對(duì)象的引用,數(shù)組的一個(gè)單元指向在堆中創(chuàng)建的一個(gè)真實(shí)對(duì)象。對(duì)像數(shù)組與基本類型數(shù)組的區(qū)別在于,對(duì)象保存的是引用,基本類型數(shù)組保存的是基本類型的值。

數(shù)組的 length屬性,表示數(shù)組的大小,而不是實(shí)際保存的元素個(gè)數(shù)。新生成的對(duì)象數(shù)組,會(huì)被默認(rèn)的初始化為 null,基本類型數(shù)組則會(huì)初始化為對(duì)應(yīng)基本類型的默認(rèn)值,對(duì)于對(duì)象數(shù)組,可以檢測(cè)其中的引用是否為 null,進(jìn)而確定某個(gè)位置是否存在對(duì)象。當(dāng)數(shù)組不被需要時(shí),垃圾回收器會(huì)負(fù)責(zé)清理數(shù)組,否則將一直存在。

實(shí)用數(shù)組方法

System.arraycopy():復(fù)制一個(gè)數(shù)組比用 for 循環(huán)復(fù)制要快得多,且該方法對(duì)所有類型做了重載。當(dāng)復(fù)制對(duì)象數(shù)組時(shí),只是復(fù)制對(duì)象的引用(屬淺拷貝)。

Arrays.asList(T... a):將多個(gè)元素轉(zhuǎn)換成 List 列表,底層表示的還是數(shù)組,當(dāng)嘗試進(jìn)行 add()、delete()操作時(shí)將在運(yùn)行時(shí)報(bào)錯(cuò)。

Arrays.sort(T[] a, Comparator c):按照給定的比較器對(duì)指定的數(shù)組進(jìn)行排序。

Arrays.binarySearch(T[] a,T key, Comparator c):對(duì)一個(gè)有序的數(shù)組采用二分查找獲取對(duì)象下標(biāo),如果數(shù)組存在重復(fù)元素,可能返回的不精確。注:如果使用了Comparator排序了一個(gè)對(duì)象數(shù)組,則調(diào)用此方法時(shí)傳入的Comparator必須是同一個(gè)。

Comparator比較器

comparator用于指定元素的排列順序,在常見的數(shù)據(jù)類型中已經(jīng)被默認(rèn)實(shí)現(xiàn),對(duì)于需要按照某種規(guī)則進(jìn)行排序的時(shí)候,提供Comparator或者實(shí)現(xiàn) java.lang.Comparable接口。

class DemoComparator implements Comparator {

    @Override
    public int compare(Demo o1, Demo o2) {
        if (o1.value == o2.value) {
            return 0;
        }
        return o1.value < o2.value ? 1 : -1;
    }
    
}
容器 Collection

一個(gè)獨(dú)立元素的序列,這些元素都服從一條或多條規(guī)則,提供了存放一組對(duì)象的方式。List必須按照插入的順序保存元素,而Set不能有重復(fù)元素。Queue按照隊(duì)列排隊(duì)規(guī)則來確定對(duì)象產(chǎn)生的順序。

PriorityQueue:優(yōu)先級(jí)隊(duì)列,聲明下一個(gè)彈出最需要的元素(具有最高的優(yōu)先級(jí)),通過默認(rèn)排序或提供 Comparator。當(dāng)調(diào)用 peek()poll()、remove()方法時(shí),獲取的元素是隊(duì)列中優(yōu)先級(jí)最高的。

List
描述
ArrayList 常用于隨機(jī)訪問元素,但是在 List的中間插入和刪除元素時(shí)較慢。
LinkedList 對(duì)中間插入和刪除元素的操作中提供了優(yōu)化的順序列表,在隨機(jī)訪問方面相對(duì)比較慢,但是它的特性集較 ArrayList更大。

ListIterator:Iterator的子類,只用于對(duì)各種 List 類的訪問,可以實(shí)現(xiàn)雙向移動(dòng)。hasNext() / next()、hasPrevious() / previous()

對(duì)于隨機(jī)訪問的 get/set操作,背后有數(shù)組支持的 List 比 ArrayList 稍快一點(diǎn),但是對(duì)于 LinkedList,將會(huì)變得很慢,因?yàn)樗皇轻槍?duì)隨機(jī)訪問操作而設(shè)計(jì)的。

最佳的做法是將 ArrayList 作為默認(rèn)首選,當(dāng)因?yàn)榻?jīng)常插入和刪除操作而性能下降時(shí),才去選擇 LinkedList。如果使用的是固定數(shù)量的元素,既可以選擇List,也可以選擇數(shù)組。

Set
描述
Set (interface) 存入Set的每一個(gè)元素都必須要唯一,因?yàn)镾et不允許重復(fù)元素。加入Set的元素必須定義 equals()方法以確保對(duì)象的唯一性。Set 和 Collection 有完全一樣的接口。 Set 接口不保證維護(hù)元素的次序。
HashSet 為快速查找而設(shè)計(jì)的Set。存入 HashSet的元素必須定義 hashCode()。
TreeSet 保持次序的 Set,底層為數(shù)據(jù)結(jié)構(gòu)。使用它可以從 Set中提取有序的序列。元素必須實(shí)現(xiàn) Comparator接口。
LinkedHashSet 具有 HashSet的查詢速度,且內(nèi)部使用鏈表維護(hù)元素的順序(插入的次序)。于是在使用迭代遍歷 Set時(shí),結(jié)果會(huì)按元素插入的次序顯示。元素必須定義 hashCode()方法。
HashSet以某種順序保存元素,LinkedHashSet按照元素插入的順序保存元素,而 TressSet按照排序維護(hù)元素。

HashSet的性能基本上總是比 TreeSet好,特別在添加和查詢?cè)貢r(shí)。 TreeSet存在的唯一原因是它維持元素的排序順序,因此迭代的時(shí)候,TreeSet通常比用HashSet要快。

Map

Map的各個(gè)實(shí)現(xiàn)類的行為特性的不同在于,效率、鍵值對(duì)的保存及呈現(xiàn)次序、對(duì)象的保存周期、映射表如何在多線程程序中工作和判定 ”鍵“等價(jià)的策略等方面。

描述
HashMap Map基于散列表的實(shí)現(xiàn)(取代HashTable)。插入和查詢“鍵值對(duì)”的開銷是固定的??梢酝ㄟ^構(gòu)造器設(shè)置容量和負(fù)載因子,以調(diào)整容器的性能。
LinkedHashMap 類似于HashMap,但是迭代遍歷它時(shí),取得 “鍵值對(duì)” 的順序就是其插入次序,或者是最近最少使用(LRU)的次序。只比HashMap慢一點(diǎn);而在迭代訪問時(shí)反而快,因?yàn)樗褂面湵砭S護(hù)內(nèi)部次序。
TreeMap 基于紅黑樹的實(shí)現(xiàn)。查看 “鍵” 或“鍵值對(duì)”時(shí),它們會(huì)被排序。TreeMap的特點(diǎn)在于,所得到的的結(jié)果是經(jīng)過排序的。TreeSet是唯一一個(gè)帶有 subMap()方法的Map,它可以返回一個(gè)子樹。
WeekHashMap 弱鍵映射,允許釋放映射所指向的對(duì)象,這是為解決某類特殊問題而設(shè)計(jì)的。如果映射 之外沒有引用指向某個(gè)“鍵”,則此“鍵”可以被垃圾收集器回收。
ConcurrentHashMap 一種線程安全的Map,它不涉及同步加鎖。
IdentityHashMap 使用 ==代替equals()對(duì)“鍵”進(jìn)行比較的散列映射。專為解決特殊問題而設(shè)計(jì)的
使用Map時(shí),首選HashMap,TreeMap維護(hù)著散列數(shù)據(jù)結(jié)構(gòu)的同時(shí)還要維護(hù)鏈表,在迭代的時(shí)候速度快,其余方面比HashMap要慢。

插入操作隨著Map尺寸的變大,速度會(huì)明顯變慢(除了 IdentityHashMap),但是查找所耗費(fèi)的代價(jià)比插入小很多。對(duì)于插入操作,由于維護(hù)鏈表所帶來的的開銷,導(dǎo)致LinkedHashSet 比 HashSet的代價(jià)高很多。

HashMap的性能因子:

容量:表中的桶位數(shù)。

初始容量:表在創(chuàng)建時(shí)所擁有的桶位數(shù)。

尺寸:表中當(dāng)前存儲(chǔ)的項(xiàng)數(shù)。

負(fù)載因子:尺寸/容量。 空表的負(fù)載因子為 0,半滿表為 0.5。負(fù)載越輕的表產(chǎn)生的沖突性越小,HashMap默認(rèn)負(fù)載因子為 0.75,達(dá)到默認(rèn)值時(shí),將進(jìn)行散列。

散列與散列碼

散列碼是一個(gè)無符號(hào)的十六進(jìn)制數(shù),散列的目的在于使用一個(gè)對(duì)象來查找另一個(gè)對(duì)象;散列的價(jià)值在于能夠快速進(jìn)行查詢。

在Map上的實(shí)現(xiàn):通過鍵對(duì)象生成一個(gè)數(shù)字,這個(gè)數(shù)字就是散列碼。將散列碼作為數(shù)組的下標(biāo),該數(shù)組的一個(gè)單元指向一個(gè)鏈表,鏈表上的一個(gè)節(jié)點(diǎn)就是一個(gè) Entry 對(duì)象。數(shù)組的容量是固定的,不同的鍵可以產(chǎn)生相同的下標(biāo),這將導(dǎo)致 “沖突”。

Example:參考Java編程思想 P493

static final int SIZE = 1024;
LinkedList>[] buckets = new LinkedList[SIZE]; 

public V put(K key, V value) {
    V oldValue = null;
    // 對(duì)散列值取余獲取數(shù)組下標(biāo)
    int index = Math.abs(key.hashCode()) % SIZE;
    if (buckets[index] == null) {
        buckets[index] = new LinkedList>();
    }

    // 獲取下標(biāo)所指向的鏈表
    LinkedList> bucket = buckets[index];
    Entry pair = new Entry<>();
    boolean found = false;

    // 獲取迭代器進(jìn)行迭代,判斷是否存在,存在則覆蓋
    ListIterator> it = bucket.listIterator();
    while (it.hasNext()) {
        Entry iPair = it.next();
        if (iPair.getKey().equals(key)) {
            oldValue = iPair.getValue();
            it.set(pair);
            found = true;
            break;
        }
    }
    // 沒有找到就添加
    if (!found) {
        buckets[index].add(pair);
    }
    return oldValue;
}

equals()

自反性。

對(duì)稱性。

傳遞性。

一致性。

對(duì)任何不是 null的 x, x.equals(null) 一定返回 false。

hashcode()

對(duì)同一個(gè)對(duì)象調(diào)用hashcode() 都應(yīng)該生成同樣的值。

基于對(duì)象的內(nèi)容生成散列碼,讓其富有意義。

散列碼不必獨(dú)一無二,應(yīng)關(guān)注生成速度。

通過 hashCode()equals(),必須能夠完全確定對(duì)象的身份。

好的 hashCode()應(yīng)該產(chǎn)生分布均勻的散列碼。賦予一個(gè)非零常量值。

public int hashCode() {
    int result = 19;
    return result * field.hashCode();
}

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/73767.html

相關(guān)文章

  • 從“數(shù)學(xué)歸納法”到理解“遞歸算法”!

    摘要:前言相信大家在面試或者工作中偶爾會(huì)遇到遞歸算法的提問或者編程,我們今天來聊一聊從數(shù)學(xué)歸納法到理解遞歸算法。這種廣義的數(shù)學(xué)歸納法應(yīng)用于數(shù)學(xué)邏輯和計(jì)算機(jī)科學(xué)領(lǐng)域,稱作結(jié)構(gòu)歸納法。 showImg(https://img-blog.csdnimg.cn/20190426221838971.gif);showImg(https://img-blog.csdnimg.cn/20190429222...

    oogh 評(píng)論0 收藏0
  • Java新手的一些建議——Java知識(shí)點(diǎn)歸納(Java基礎(chǔ)部分)

    摘要:中很多特性或者說知識(shí)點(diǎn)都是和面向?qū)ο缶幊谈拍钕嚓P(guān)的。在多線程中內(nèi)容有很多,只是簡(jiǎn)單說明一下中初步使用多線程需要掌握的知識(shí)點(diǎn),以后有機(jī)會(huì)單獨(dú)再詳細(xì)介紹一些高級(jí)特性的使用場(chǎng)景。   寫這篇文章的目的是想總結(jié)一下自己這么多年來使用java的一些心得體會(huì),主要是和一些java基礎(chǔ)知識(shí)點(diǎn)相關(guān)的,所以也希望能分享給剛剛?cè)腴T的Java程序員和打算入Java開發(fā)這個(gè)行當(dāng)?shù)臏?zhǔn)新手們,希望可以給大家一些經(jīng)...

    lykops 評(píng)論0 收藏0
  • 一份送給Java初學(xué)者的指南

    摘要:編程思想第版這本書要常讀,初學(xué)者可以快速概覽,中等程序員可以深入看看,老鳥還可以用之回顧的體系。以下視頻整理自慕課網(wǎng)工程師路徑相關(guān)免費(fèi)課程。 我自己總結(jié)的Java學(xué)習(xí)的系統(tǒng)知識(shí)點(diǎn)以及面試問題,目前已經(jīng)開源,會(huì)一直完善下去,歡迎建議和指導(dǎo)歡迎Star: https://github.com/Snailclimb/Java-Guide 筆者建議初學(xué)者學(xué)習(xí)Java的方式:看書+視頻+實(shí)踐(初...

    banana_pi 評(píng)論0 收藏0
  • 我的阿里路+Java面經(jīng)考點(diǎn)

    摘要:我的是忙碌的一年,從年初備戰(zhàn)實(shí)習(xí)春招,年三十都在死磕源碼,三月份經(jīng)歷了阿里五次面試,四月順利收到實(shí)習(xí)。因?yàn)槲倚睦砗芮宄业哪繕?biāo)是阿里。所以在收到阿里之后的那晚,我重新規(guī)劃了接下來的學(xué)習(xí)計(jì)劃,將我的短期目標(biāo)更新成拿下阿里轉(zhuǎn)正。 我的2017是忙碌的一年,從年初備戰(zhàn)實(shí)習(xí)春招,年三十都在死磕JDK源碼,三月份經(jīng)歷了阿里五次面試,四月順利收到實(shí)習(xí)offer。然后五月懷著忐忑的心情開始了螞蟻金...

    姘擱『 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<