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

資訊專欄INFORMATION COLUMN

兩數(shù)之和問題各變種多解法小結(jié)

lentoo / 528人閱讀

摘要:兩數(shù)之和問題各變種多解法小結(jié)聲明文章均為本人技術(shù)筆記,轉(zhuǎn)載請注明出處兩數(shù)之和等于題目大意給出未排序數(shù)組和指定目標(biāo),返回數(shù)組中兩數(shù)之和的組合元素下標(biāo)要求下標(biāo)從開始,而且,保證題目中有且只有個可行解解法暴力時間復(fù)雜度求解解題思路暴力二重循環(huán)求解

兩數(shù)之和問題各變種多解法小結(jié) 聲明

文章均為本人技術(shù)筆記,轉(zhuǎn)載請注明出處:
[1] https://segmentfault.com/u/yzwall
[2] blog.csdn.net/j_dark/

LintCode_56:兩數(shù)之和等于target

題目大意:給出未排序數(shù)組nums和指定目標(biāo)target,返回數(shù)組中兩數(shù)之和$= target$的組合元素下標(biāo)[index1, index2], 要求下標(biāo)從1開始,而且$index1 < index2$,保證題目中有且只有1個可行解;

解法1:暴力$O(n^2)$時間復(fù)雜度求解

解題思路:暴力二重循環(huán)求解;
復(fù)雜度分析:時間復(fù)雜度$O(n^2)$,空間復(fù)雜度$O(1)$

/**
 * 解法1:時間復(fù)雜度O(n^2),空間復(fù)雜度O(1)
 * 遍歷求兩數(shù)之和等于target,返回兩數(shù)下標(biāo)(從1開始)
 * http://www.lintcode.com/zh-cn/problem/two-sum/
 * @author yzwall
 */
class Solution {
    public int[] twoSum(int[] nums, int target) {
        int[] results = new int[2];
        for (int i = 0; i < nums.length; i++) {
            for (int j = i + 1; j < nums.length; j++) {
                if (nums[i] + nums[j] == target) {
                    results[0] = i + 1;
                    results[1] = j + 1;
                    return results;
                }
            }
        }
        return results;
    }
}
解法2:HashMap $O(n)$時間復(fù)雜度求解

解題思路:耗費$O(n)$空間構(gòu)造哈希表,遍歷數(shù)組每個元素nums[i],哈希表對應(yīng)存儲,存儲nums[i]期望的“另一半”,一旦哈希表中包含nums[i],代表“另一半”早已存儲在哈希表中,直接返回即可;
復(fù)雜度分析:時間復(fù)雜度$O(n)$,空間復(fù)雜度$O(n)$

/**
 * 解法2:HashMap求解,時間復(fù)雜度O(n),空間復(fù)雜度O(n)
 * 遍歷求兩數(shù)之和等于target,返回兩數(shù)下標(biāo)(從1開始)
 * http://www.lintcode.com/zh-cn/problem/two-sum/
 * @author yzwall
 */
class Solution {
    public int[] twoSum(int[] nums, int target) {
        HashMap map = new HashMap<>();
        int[] results = new int[2];
        for (int i = 0; i < nums.length; i++) {
            if (map.containsKey(nums[i])) {
                results[0] = map.get(nums[i]) + 1;
                results[1] = i + 1;
                break;
            }
            map.put(target - nums[i], i);
        }
        return results;
    }
}
解法3:雙指針$O(nlog(n))$時間復(fù)雜度求解

解題思路:首先將數(shù)組排序(時間復(fù)雜度$O(nlog(n))$),然后通過雙指針ij分別從數(shù)組兩頭同時遍歷,保存數(shù)組排序前的元素位置可使用HashMap保存(空間復(fù)雜度$O(n)$),也可用輔助類保存(空間復(fù)雜度$O(1)$);
復(fù)雜度分析:時間復(fù)雜度$O(nlog(n))$,空間復(fù)雜度$O(n)$ or $O(1)$;

/**
 * 解法3:HashMap + 雙指針求解,時間復(fù)雜度O(nlogn),空間復(fù)雜度O(n)
 * 遍歷求兩數(shù)之和等于target,返回兩數(shù)下標(biāo)(從1開始)
 * http://www.lintcode.com/zh-cn/problem/two-sum/
 * @author yzwall
 */
class Solution {
    public int[] twoSum(int[] nums, int target) {
        HashMap> map = new HashMap<>();
        int[] results = new int[2];
        // HashMap用于記錄排序前數(shù)組元素對應(yīng)下標(biāo)
        for (int i = 0; i < nums.length; i++) {
            if (map.containsKey(nums[i])) {
                map.get(nums[i]).add(i);
                continue;
            }
            map.put(nums[i], new ArrayList());
            map.get(nums[i]).add(i);
        }
        
        int i = 0, j = nums.length - 1;
        // 排序后雙指針求解
        Arrays.sort(nums);
        while (i < j) {
            if (nums[i] + nums[j] == target) {
                int index1 = map.get(nums[i]).get(0);
                // 重復(fù)元素已經(jīng)訪問過一次,從對應(yīng)位置列表中剔除
                map.get(nums[i]).remove(0);
                int index2 = map.get(nums[j]).get(0);
                // 保證results[0] < result[1]
                results[0] = Math.min(index1, index2) + 1;
                results[1] = Math.max(index1, index2) + 1;
                return results;
            }
            if (nums[i] + nums[j] > target) {
                j--;
            } else {
                i++;
            }
        }
        return results;
    }
}
LintCode_587:兩數(shù)之和等于target的不重復(fù)組合數(shù)目

題目大意:給出未排序數(shù)組nums和指定目標(biāo)target,返回數(shù)組中兩數(shù)之和$= target$的所有不重復(fù)組合數(shù);

解法:雙指針法$O(n)$時間復(fù)雜度求解

解題思路:數(shù)組排序后使用雙指針分別從起點和終點遍歷,如果存在$a + b = target$,則如果找到$a$所有組合方案,則$b$無需再找;
復(fù)雜度分析:時間復(fù)雜度$O(nlog(n))$,空間復(fù)雜度通過HashSet去重,耗費額外空間$O(n)$(可優(yōu)化到$O(1)$)

/**
 * 雙指針找兩數(shù)和等于target的不重復(fù)組合數(shù)目,時間復(fù)雜度O(n),空間復(fù)雜度O(n)
 * 求兩數(shù)之和等于target的所有不重復(fù)組合數(shù)目
 * http://www.lintcode.com/zh-cn/problem/two-sum-unique-pairs/
 * @author yzwall
 */
class Solution {
    public int twoSum6(int[] nums, int target) {
        int pairs = 0;
        if (nums == null || nums.length < 2) {
            return pairs;
        }
        Arrays.sort(nums);
        int i = 0;
        int j = nums.length - 1;
        HashSet set = new HashSet<>();
        while (i < j) {
            // 如果a + b = target, a找到后,b無需再找
            while (i < j && set.contains(nums[i])) {
                i++;
            }
            while (i < j && set.contains(nums[j])) {
                j--;
            }
            if (i < j) {
                if (nums[i] + nums[j] == target) {
                    set.add(nums[i]);
                    set.add(nums[j]);
                    pairs++;
                } else if (nums[i] + nums[j] > target) {
                    j--;
                } else {
                    i++;
                }
            }
        }
        return pairs;
    }
}
LintCode_608:兩數(shù)之和等于target(數(shù)組已排序)

題目大意:題目是LintCode_56的簡化版,解法1和解法2可直接使用;與解法1,2相比,解法3雙指針法充分利用數(shù)組已排序條件,時間復(fù)雜度降低到$O(n)$,空間復(fù)雜度降低到$O(1)$;

LintCode_443:兩數(shù)之和大于target

題目大意:求出數(shù)組nums中兩數(shù)之和$> target$的組合數(shù)目;

解法1:暴力$O(n^2)$時間復(fù)雜度求解

二重循環(huán)暴力求解;

解法2:雙指針法$O(nlog(n))$時間復(fù)雜度求解

解題思路:首先將數(shù)組nums升序排序,雙指針$i$從起點開始,指針$j$從終點開始,一旦有:$$nums[i] + nums[j] > target, $$則由于數(shù)組嚴(yán)格不遞減,$$forall numin[nums[i], nums[j]], num + nums[j] > target$$,因此執(zhí)行pairs += (j - i),此時$nums[j]$所有方案搜索完畢,執(zhí)行j--

復(fù)雜度分析:時間復(fù)雜度$O(nlog(n))$,空間復(fù)雜度$O(1)$

/**
 * 解法2:雙指針法求解,時間復(fù)雜度O(logn),空間復(fù)雜度O(1) 
 * 求兩數(shù)之和大于target的組合數(shù)目
 * http://www.lintcode.com/en/problem/two-sum-greater-than-target/
 * @author yzwall
 */
class Solution {
     public int twoSum2(int[] nums, int target) {
            int pairs = 0;
            if (nums == null || nums.length < 2) {
                return pairs;
            }
            Arrays.sort(nums);
            int i = 0;
            int j = nums.length - 1;
            while (i < j) {
                if (nums[i] + nums[j] > target) {
                    pairs += j - i;
                    // nums[j]所有方案求解完畢,j--
                    j--;
                } else {
                    i++;
                }
            }
            return pairs;
     }
}
LintCode_609:兩數(shù)之和不超過target

題目大意:求出數(shù)組nums中兩數(shù)之和$leqslant target$的組合數(shù)目;

LintCode_610:兩數(shù)之差等于target 解法1:暴力$O(n^2)$時間復(fù)雜度求解

二重循環(huán)暴力求解;

解法2:雙指針法$O(nlog(n))$時間復(fù)雜度求解

解題思路:首先將數(shù)組nums升序排序,雙指針$i$從起點開始,指針$j$從終點開始,一旦有:$$nums[i] + nums[j] leq target, $$則由于數(shù)組嚴(yán)格不遞減,$$forall numin[nums[i], nums[j]], num + nums[j] leq target$$,因此執(zhí)行pairs += (j - i),此時$nums[i]$所有方案搜索完畢,執(zhí)行i++;
復(fù)雜度分析:時間復(fù)雜度$O(nlog(n))$,空間復(fù)雜度$O(1)$

/**
 * 雙指針法求解,時間復(fù)雜度O(nlogn),空間復(fù)雜度O(1)
 * 求兩數(shù)之和小于等于target的所有組合數(shù)目
 * http://www.lintcode.com/en/problem/two-sum-less-than-or-equal-to-target/
 * @author yzwall
 */
class Solution {
     public int twoSum5(int[] nums, int target) {
        int pairs = 0;
        if (nums == null || nums.length < 2) {
            return pairs;
        }
        Arrays.sort(nums);
        int i = 0;
        int j = nums.length - 1;
        while (i < j) {
            // nums[i]的所有組合 = j - i
            if (nums[i] + nums[j] <= target) {
                pairs += j - i;
                i++;
            } else {
                j--;
            }
        }
        return pairs;
     }
}
LintCode_533:兩數(shù)之和最接近target

題目大意:求出數(shù)組nums中兩數(shù)之和與target的最近距離(非負(fù));

解法1:暴力$O(n^2)$時間復(fù)雜度求解

解題思路:二重循環(huán)不斷迭代最小距離;
復(fù)雜度分析:時間復(fù)雜度$O(n^2)$,空間復(fù)雜度$O(1)$;

/**
 * 解法1:暴力時間復(fù)雜度O(n^2)
 * 求兩數(shù)之和最接近target,求最近距離
 * http://www.lintcode.com/en/problem/two-sum-closest-to-target/
 * @author yzwall
 */
class Solution {
    public int twoSumClosest(int[] nums, int target) {
        if (nums == null || nums.length < 2) {
            return target;
        }
        
        int min = Integer.MAX_VALUE;
        int temp;
        for (int i = 0; i < nums.length; i++) {
            for (int j = i + 1; j < nums.length; j++) {
                temp = target - (nums[i] + nums[j]);
                min = Math.min(min, Math.abs(temp));
            }
        }
        return min;
    }
}
解法2:雙指針法$O(nlog(n))$時間復(fù)雜度求解

解題思路:首先將數(shù)組排序,雙指針分別從起點和終點遍歷,臨時距離
$$
temp = left | target - (nums[i] + nums[j]) right |
$$
不停與最短距離$min$比較迭代,$temp = 0$時直接返回;
復(fù)雜度分析:時間復(fù)雜度$O(nlog(n))$,空間復(fù)雜度$O(1)$;

/**
 * 解法2:雙指針法求解,時間復(fù)雜度O(nlogn)
 * 求兩數(shù)之和最接近target,求最近距離
 * http://www.lintcode.com/en/problem/two-sum-closest-to-target/
 * @author yzwall
 */
class Solution {
    public int twoSumClosest(int[] nums, int target) {
        if (nums == null || nums.length < 2) {
            return target;
        }
        
        Arrays.sort(nums);
        int i = 0;
        int j = nums.length - 1;
        int min = Integer.MAX_VALUE;
        int temp;
        while (i < j) {
            temp = Math.abs(target - (nums[i] + nums[j]));
            if (temp == 0) {
                return 0;
            }
            min = Math.min(min, temp);
            if (nums[i] + nums[j] > target) {
                // 距離過大, j--
                j--;
            } else {
                // 距離過小, i++
                i++;
            }
        }
        return min;
   }
}
LintCode_610:兩數(shù)之差等于target 解法1:暴力$O(n^2)$時間復(fù)雜度求解

二重循環(huán)暴力求解;

解法2:雙指針法$O(nlog n)$時間復(fù)雜度求解

解題思路:首先將數(shù)組nums升序排序,雙指針$i$從起點開始,指針$j$從終點開始,一旦有:$$nums[i] + nums[j] leq target$$則由于數(shù)組嚴(yán)格不遞減,$$forall numin[nums[i], nums[j]], num + nums[j] leq target$$,因此執(zhí)行pairs += (j - i),此時$nums[i]$所有方案搜索完畢,執(zhí)行i++
復(fù)雜度分析:時間復(fù)雜度$O(nlog(n))$,空間復(fù)雜度$O(1)$

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

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

相關(guān)文章

  • 【前端來刷LeetCode】兩數(shù)之和兩數(shù)相加

    摘要:給定表,存在函數(shù),對任意給定的關(guān)鍵字值,代入函數(shù)后若能得到包含該關(guān)鍵字的記錄在表中的地址,則稱表為哈希表,函數(shù)為哈希函數(shù)。而中的對象就是基于哈希表結(jié)構(gòu),所以我們構(gòu)造一個對象即可,是當(dāng)前遍歷到的值,是其與目標(biāo)值的差。 大部分玩前端的小伙伴,在算法上都相對要薄弱些,畢竟調(diào)樣式、調(diào)兼容就夠掉頭發(fā)的了,哪還有多余的頭發(fā)再去折騰。 確實在前端中需要使用到算法的地方是比較少,但若要往高級方向發(fā)展,...

    BLUE 評論0 收藏0
  • LeetCode - 001 - 兩數(shù)之和(two-sum)

    摘要:解法返回目錄解題代碼執(zhí)行測試解題思路使用雙重循環(huán)破解。解法返回目錄解題代碼執(zhí)行測試知識點遍歷數(shù)組,返回遍歷項,返回當(dāng)前索引。 Create by jsliang on 2019-05-16 22:19:13 Recently revised in 2019-05-17 14:22:40 Hello 小伙伴們,如果覺得本文還不錯,記得給個 star , 小伙伴們的 star 是我持續(xù)更新的動...

    habren 評論0 收藏0
  • LeetCode 167:兩數(shù)之和 II - 輸入有序數(shù)組 Two Sum II - Input a

    摘要:公眾號愛寫給定一個已按照升序排列的有序數(shù)組,找到兩個數(shù)使得它們相加之和等于目標(biāo)數(shù)。函數(shù)應(yīng)該返回這兩個下標(biāo)值和,其中必須小于。示例輸入輸出解釋與之和等于目標(biāo)數(shù)。 公眾號: 愛寫bug(ID:icodebugs) 給定一個已按照升序排列 的有序數(shù)組,找到兩個數(shù)使得它們相加之和等于目標(biāo)數(shù)。 函數(shù)應(yīng)該返回這兩個下標(biāo)值 index1 和 index2,其中 index1 必須小于 index2。...

    張春雷 評論0 收藏0
  • LeetCode 167:兩數(shù)之和 II - 輸入有序數(shù)組 Two Sum II - Input a

    摘要:公眾號愛寫給定一個已按照升序排列的有序數(shù)組,找到兩個數(shù)使得它們相加之和等于目標(biāo)數(shù)。函數(shù)應(yīng)該返回這兩個下標(biāo)值和,其中必須小于。示例輸入輸出解釋與之和等于目標(biāo)數(shù)。 公眾號: 愛寫bug(ID:icodebugs) 給定一個已按照升序排列 的有序數(shù)組,找到兩個數(shù)使得它們相加之和等于目標(biāo)數(shù)。 函數(shù)應(yīng)該返回這兩個下標(biāo)值 index1 和 index2,其中 index1 必須小于 index2。...

    Me_Kun 評論0 收藏0
  • 【leetcode系列】001-兩數(shù)之和

    摘要:題意給定一個整數(shù)數(shù)組和一個目標(biāo)值,請你在該數(shù)組中找出和為目標(biāo)值的那兩個整數(shù),并返回他們的數(shù)組下標(biāo)。也就是說,字典里記錄的是每個數(shù)據(jù)希望找到的另一半的值的大小。返回這兩個下標(biāo)就行,如果沒有存在于字典里,那么繼續(xù)存入字典。 showImg(https://segmentfault.com/img/bVbvgPA); 題意: 給定一個整數(shù)數(shù)組 nums 和一個目標(biāo)值 target,請你在該數(shù)...

    EddieChan 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<