LeetCode-1. Two Sum

Two Sum

題目描述

Given an array of integers, return indices of the two numbers such that they add up to a specific target.
You may assume that each input would have exactly one solution, and you may not use the same element twice

Example

Given nums = [2, 7, 11, 15], target = 9,
Because nums[0] + nums[1] = 2 + 7 = 9,
return [0, 1].

中文描述

給定一個(gè)數(shù)組和一個(gè)目標(biāo)值蒲跨,返回和為目標(biāo)值的兩個(gè)數(shù)的索引值列粪。假設(shè)每一個(gè)輸入都有一個(gè)確定結(jié)果贷帮,同一個(gè)元素?zé)o法使用兩次

解題方案

  1. 首先想到的就是brute force算法,兩層的循環(huán)解決問(wèn)題屎飘,時(shí)間復(fù)雜度為O(n),空間復(fù)雜度為常數(shù)。代碼如下
public int[] twoSum(int[] nums, int target) {
    int[] answer= new int[2];
    for(int i=0;i<nums.length;i++){
        answer[0]=nums[i];
        int temp = target-answer[0];
        for(int j=i;j<nums.length;j++){
            if(nums[j]==temp){
                return answer;
            }
        }
    }
    return answer;
}
  1. 第二種方案就是考慮如何優(yōu)化上一種算法了,這里看到網(wǎng)上的采用HashMap的方式來(lái)優(yōu)化算法的時(shí)間復(fù)雜度章钾。

利用的是HashMap取數(shù)據(jù)時(shí)間是常數(shù)時(shí)間。

public int[] twoSum2(int[] nums, int target) {
   int[] res = new int[2];
   if(numbers==null||numbers.length<2) return null;
   Map<Integer, Integer> map = new HashMap<>();
   for(int i=0;i<numbers.length;i++){
       if(map.containsKey(target-numbers[i])){
           res[0]=map.get(target-numbers[i])+1;
           res[1]=i+1;
           return res;
       }
       map.put(numbers[i],i);
   }
   return null;
}

因?yàn)橹皩?duì)于Java集合類(lèi)的基礎(chǔ)了解不夠热芹,所以不太理解所謂的HashMap取值操作為常數(shù)時(shí)間贱傀。于是查閱了相關(guān)資料,現(xiàn)記錄一下。

Java HashMap 是基于哈希表的Map接口實(shí)現(xiàn)伊脓,以Key-Value的形式在HashMap中府寒,key-value總是會(huì)當(dāng)做一個(gè)整體來(lái)處理,系統(tǒng)會(huì)根據(jù)hash算法來(lái)來(lái)計(jì)算key-value的存儲(chǔ)位置报腔,我們總是可以通過(guò)key快速地存株搔、取value。下面就來(lái)分析HashMap的存取纯蛾。

HashMap的構(gòu)造函數(shù)

HashMap():構(gòu)造一個(gè)具有默認(rèn)初始容量 (16) 和默認(rèn)加載因子 (0.75) 的空 HashMap纤房。
HashMap(int initialCapacity):構(gòu)造一個(gè)帶指定初始容量和默認(rèn)加載因子 (0.75) 的空 HashMap。
HashMap(int initialCapacity, float loadFactor):構(gòu)造一個(gè)帶指定初始容量和加載因子的空 HashMap翻诉。

其中關(guān)于初始容量和加載因子和其他數(shù)據(jù)結(jié)構(gòu)的概念相同炮姨,初始容量是創(chuàng)建哈希表d額初始大小,加載因子表示哈希表的最大填充程度碰煌。也就是當(dāng)哈希表達(dá)到這個(gè)填充程度的時(shí)候舒岸,就需要對(duì)哈希表進(jìn)行擴(kuò)容。

HashMap存儲(chǔ)

下面簡(jiǎn)單看一下hashMap的put方法源碼

public V put(K key, V value) {
    //當(dāng)key為null芦圾,調(diào)用putForNullKey方法吁津,保存null與table第一個(gè)位置中,這是HashMap允許為null的原因
    if (key == null)
        return putForNullKey(value);
    //計(jì)算key的hash值
    int hash = hash(key.hashCode());                  ------(1)
    //計(jì)算key hash 值在 table 數(shù)組中的位置
    int i = indexFor(hash, table.length);             ------(2)
    //從i出開(kāi)始迭代 e,找到 key 保存的位置
    for (Entry<K, V> e = table[i]; e != null; e = e.next) {
        Object k;
        //判斷該條鏈上是否有hash值相同的(key相同)
        //若存在相同,則直接覆蓋value碍脏,返回舊value
        if (e.hash == hash && ((k = e.key) == key || key.equals(k))) {
            V oldValue = e.value;    //舊值 = 新值
            e.value = value;
            e.recordAccess(this);
            return oldValue;     //返回舊值
        }
    }
    //修改次數(shù)增加1
    modCount++;
    //將key梭依、value添加至i位置處
    addEntry(hash, key, value, i);
    return null;
}

通過(guò)源碼我們可以清晰看到HashMap保存數(shù)據(jù)的過(guò)程為:首先判斷key是否為null,若為null典尾,則直接調(diào)用putForNullKey方法役拴。若不為空則先計(jì)算key的hash值,然后根據(jù)hash值搜索在table數(shù)組中的索引位置钾埂,如果table數(shù)組在該位置處有元素河闰,則通過(guò)比較是否存在相同的key,若存在則覆蓋原來(lái)key的value褥紫,否則將該元素保存在鏈頭(最先保存的元素放在鏈尾)姜性。若table在該處沒(méi)有元素,則直接保存髓考。保存的過(guò)程看起來(lái)十分簡(jiǎn)單部念,但是其中有一部分比較重要的就是求Hash值函數(shù)


static int hash(int h) {
    h ^= (h >>> 20) ^ (h >>> 12);
    return h ^ (h >>> 7) ^ (h >>> 4);
}

對(duì)于HashMap的table而言,數(shù)據(jù)分布需要均勻(最好每項(xiàng)都只有一個(gè)元素氨菇,這樣就可以直接找到)儡炼,不能太緊也不能太松,太緊會(huì)導(dǎo)致查詢速度慢查蓉,太松則浪費(fèi)空間乌询。計(jì)算hash值后,怎么保證數(shù)據(jù)的分布呢豌研?HashMap調(diào)用indexFor方法妹田。

static int indexFor(int h, int length) {
   return h & (length-1);
}

這個(gè)函數(shù)我在一開(kāi)始沒(méi)有理解到底什么意思,通過(guò)看別人的講解后發(fā)現(xiàn)這個(gè)函數(shù)的作用和取模運(yùn)算是相同的鹃共。具體計(jì)算的結(jié)果可以看下面的表格秆麸。

h length-1 h&length-1 result
0 14 0000&1110=0000 0
1 14 0001&1110=0000 0
2 14 0010&1110=0010 2
3 14 0011&1110=0010 2
4 14 0100&1110=0100 4
5 14 0101&1110=0100 4
6 14 0110&1110=0110 6
7 14 0111&1110=0110 6
8 14 1000&1110=1000 8
9 14 1001&1110=1000 8
10 14 1010&1110=1010 10
11 14 1011&1110=1010 10
12 14 1100&1110=1100 12
13 14 1101&1110=1100 12
14 14 1110&1110=1110 14
15 14 1111&1110=1110 14

從上面的圖表中我們看到總共發(fā)生了8此碰撞,同時(shí)發(fā)現(xiàn)浪費(fèi)的空間非常大及汉,有1、3屯烦、5坷随、7、9驻龟、11温眉、13、15處沒(méi)有記錄翁狐,也就是沒(méi)有存放數(shù)據(jù)类溢。這是因?yàn)樗麄冊(cè)谂c14進(jìn)行&運(yùn)算時(shí),得到的結(jié)果最后一位永遠(yuǎn)都是0,即0001闯冷、0011砂心、0101、0111蛇耀、1001辩诞、1011、1101纺涤、1111位置處是不可能存儲(chǔ)數(shù)據(jù)的译暂,空間減少,進(jìn)一步增加碰撞幾率撩炊,這樣就會(huì)導(dǎo)致查詢速度慢外永。而當(dāng)length = 16時(shí),length – 1 = 15 即1111拧咳,那么進(jìn)行低位&運(yùn)算時(shí)伯顶,值總是與原來(lái)hash值相同,而進(jìn)行高位運(yùn)算時(shí)呛踊,其值等于其低位值砾淌。所以說(shuō)當(dāng)length = 2^n時(shí),不同的hash值發(fā)生碰撞的概率比較小谭网,這樣就會(huì)使得數(shù)據(jù)在table數(shù)組中分布較均勻汪厨,查詢速度也較快。

讀取的實(shí)現(xiàn)

相對(duì)于HashMap的存而言愉择,取就顯得比較簡(jiǎn)單了劫乱。通過(guò)key的hash值找到在table數(shù)組中的索引處的Entry,然后返回該key對(duì)應(yīng)的value即可锥涕。

public V get(Object key) {
    // 若為null衷戈,調(diào)用getForNullKey方法返回相對(duì)應(yīng)的value
    if (key == null)
        return getForNullKey();
    // 根據(jù)該 key 的 hashCode 值計(jì)算它的 hash 碼  
    int hash = hash(key.hashCode());
    // 取出 table 數(shù)組中指定索引處的值
    for (Entry<K, V> e = table[indexFor(hash, table.length)]; e != null; e = e.next) {
        Object k;
        //若搜索的key與查找的key相同,則返回相對(duì)應(yīng)的value
        if (e.hash == hash && ((k = e.key) == key || key.equals(k)))
            return e.value;
    }
    return null;
}

從上面的代碼當(dāng)中我們就可以得到為什么HashMap可以以常數(shù)的時(shí)間進(jìn)行存取數(shù)據(jù)了层坠。

再寫(xiě)這篇文章的時(shí)候殖妇,在CSDN的博客發(fā)現(xiàn)了另外的解法,

先對(duì)數(shù)組進(jìn)行排序破花,然后使用夾逼的方法找出滿足條件的pair谦趣,原理是因?yàn)閿?shù)組是有序的,那么假設(shè)當(dāng)前結(jié)果比target大座每,那么左端序號(hào)右移只會(huì)使兩個(gè)數(shù)的和更大前鹅,反之亦然。所以每次只會(huì)有一個(gè)選擇峭梳,從而實(shí)現(xiàn)線性就可以求出結(jié)果舰绘。該算法的時(shí)間復(fù)雜度是O(nlogn+n)=O(nlogn),空間復(fù)雜度取決于排序算法。

public int[] twoSum(int[] numbers, int target) {
    int[] res = new int[2];
    if(numbers==null || numbers.length<2)
        return null;
    Arrays.sort(numbers);
    int l = 0;
    int r = numbers.length-1;
    while(l<r)
    {
        if(numbers[l]+numbers[r]==target)
        {
            res[0] = number[l];
            res[1] = number[r];
            return res;
        }
        else if(numbers[l]+numbers[r]>target) r--;
        else l++;
    }
    return null;
}

參考鏈接

  1. http://blog.csdn.net/linhuanmars/article/details/19711387
  2. http://www.cnblogs.com/chenssy/p/3521565.html

項(xiàng)目GitHub鏈接
https://github.com/yanqinghe/leetcode/blob/master/leetcodeJava/src/Two_Sum_1/Solution.java

PS:寫(xiě)在后面捂寿,現(xiàn)在開(kāi)始補(bǔ)寫(xiě)文檔記錄刷leetcode的過(guò)程口四,也希望能把自己的想法記錄下來(lái)與別人分享,同時(shí)也希望能夠與更多的人交流者蠕。
GitHub主頁(yè)https://github.com/yanqinghe/leetcode

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末窃祝,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子踱侣,更是在濱河造成了極大的恐慌粪小,老刑警劉巖,帶你破解...
    沈念sama閱讀 206,968評(píng)論 6 482
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件抡句,死亡現(xiàn)場(chǎng)離奇詭異探膊,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)待榔,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,601評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門(mén)逞壁,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人锐锣,你說(shuō)我怎么就攤上這事腌闯。” “怎么了雕憔?”我有些...
    開(kāi)封第一講書(shū)人閱讀 153,220評(píng)論 0 344
  • 文/不壞的土叔 我叫張陵姿骏,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我斤彼,道長(zhǎng)分瘦,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 55,416評(píng)論 1 279
  • 正文 為了忘掉前任琉苇,我火速辦了婚禮嘲玫,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘并扇。我一直安慰自己去团,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,425評(píng)論 5 374
  • 文/花漫 我一把揭開(kāi)白布穷蛹。 她就那樣靜靜地躺著土陪,像睡著了一般。 火紅的嫁衣襯著肌膚如雪俩莽。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 49,144評(píng)論 1 285
  • 那天乔遮,我揣著相機(jī)與錄音扮超,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛出刷,可吹牛的內(nèi)容都是我干的璧疗。 我是一名探鬼主播,決...
    沈念sama閱讀 38,432評(píng)論 3 401
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼馁龟,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼崩侠!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起坷檩,我...
    開(kāi)封第一講書(shū)人閱讀 37,088評(píng)論 0 261
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤却音,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后矢炼,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體系瓢,經(jīng)...
    沈念sama閱讀 43,586評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,028評(píng)論 2 325
  • 正文 我和宋清朗相戀三年句灌,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了夷陋。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,137評(píng)論 1 334
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡胰锌,死狀恐怖骗绕,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情资昧,我是刑警寧澤酬土,帶...
    沈念sama閱讀 33,783評(píng)論 4 324
  • 正文 年R本政府宣布,位于F島的核電站榛搔,受9級(jí)特大地震影響诺凡,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜践惑,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,343評(píng)論 3 307
  • 文/蒙蒙 一腹泌、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧尔觉,春花似錦凉袱、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,333評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至钉稍,卻和暖如春涤躲,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背贡未。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,559評(píng)論 1 262
  • 我被黑心中介騙來(lái)泰國(guó)打工种樱, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留蒙袍,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 45,595評(píng)論 2 355
  • 正文 我出身青樓嫩挤,卻偏偏與公主長(zhǎng)得像害幅,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子岂昭,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,901評(píng)論 2 345

推薦閱讀更多精彩內(nèi)容

  • HashMap 是 Java 面試必考的知識(shí)點(diǎn)以现,面試官?gòu)倪@個(gè)小知識(shí)點(diǎn)就可以了解我們對(duì) Java 基礎(chǔ)的掌握程度。網(wǎng)...
    野狗子嗷嗷嗷閱讀 6,657評(píng)論 9 107
  • Related Topics:[Array][Hash Table]Similar Questions[3Sum]...
    lijia069閱讀 262評(píng)論 0 0
  • 實(shí)際上约啊,HashSet 和 HashMap 之間有很多相似之處邑遏,對(duì)于 HashSet 而言,系統(tǒng)采用 Hash 算...
    曹振華閱讀 2,508評(píng)論 1 37
  • 1棍苹,form提交 2无宿,js提交 return vallidate();這里的return可有可無(wú) type="bu...
    xiaolin_188閱讀 570評(píng)論 0 0
  • 事件:孩子學(xué)校成立家長(zhǎng)委員會(huì),孩子班主任想讓我擔(dān)任組長(zhǎng)枢里,孩子第一反應(yīng)是讓我別當(dāng)組長(zhǎng)孽鸡,當(dāng)個(gè)組員或其他 一、思維筆的空...
    碧霞閱讀 151評(píng)論 0 0