LeetCode基礎(chǔ)算法-數(shù)組

LeetCode基礎(chǔ)算法-數(shù)組

算法 LeetCode 數(shù)組相關(guān)


1. 從排序數(shù)組中刪除重復(fù)項(xiàng)

描述:
給定一個(gè)排序數(shù)組,你需要在原地刪除重復(fù)出現(xiàn)的元素,使得每個(gè)元素只出現(xiàn)一次,返回移除后數(shù)組的新長度。

不要使用額外的數(shù)組空間,你必須在原地修改輸入數(shù)組并在使用 O(1) 額外空間的條件下完成。

解答思路:

  1. 雙指針法,index指針指向當(dāng)前已無重復(fù)數(shù)字?jǐn)?shù)組的最后一位;i指針遍歷nums數(shù)組。
  2. i與index處值相同時(shí),i指針繼續(xù)向后遍歷。
  3. i與index處值相同時(shí),將i處值復(fù)制給++index處。
  4. 無重復(fù)數(shù)組的長度為index+1。

代碼:

    public int removeDuplicates(int[] nums) {
        if (nums == null || nums.length == 0) {
            return 0;
        }
        // index從索引0開始
        int index = 0;
        // i從索引1開始遍歷
        for (int i = 1; i < nums.length; i++) {
            // 不相同則進(jìn)行擴(kuò)展,index永遠(yuǎn)指向符合條件數(shù)組的最后一個(gè)索引
            if (nums[i] != nums[index]) {
                nums[++index] = nums[i];
            }
        }
        return index + 1;
    }

2. 買賣股票的最佳時(shí)機(jī)

描述:
給定一個(gè)數(shù)組,它的第 i 個(gè)元素是一支給定股票第 i 天的價(jià)格。

設(shè)計(jì)一個(gè)算法來計(jì)算你所能獲取的最大利潤。你可以盡可能地完成更多的交易(多次買賣一支股票)。

注意:你不能同時(shí)參與多筆交易(你必須在再次購買前出售掉之前的股票)。

示例 1:
輸入: [7,1,5,3,6,4]
輸出: 7
解釋: 在第 2 天(股票價(jià)格 = 1)的時(shí)候買入,在第 3 天(股票價(jià)格 = 5)的時(shí)候賣出, 這筆交易所能獲得利潤 = 5-1 = 4 。
隨后,在第 4 天(股票價(jià)格 = 3)的時(shí)候買入,在第 5 天(股票價(jià)格 = 6)的時(shí)候賣出, 這筆交易所能獲得利潤 = 6-3 = 3 。

示例 2:
輸入: [1,2,3,4,5]
輸出: 4
解釋: 在第 1 天(股票價(jià)格 = 1)的時(shí)候買入,在第 5 天 (股票價(jià)格 = 5)的時(shí)候賣出, 這筆交易所能獲得利潤 = 5-1 = 4 。
注意你不能在第 1 天和第 2 天接連購買股票,之后再將它們賣出。
因?yàn)檫@樣屬于同時(shí)參與了多筆交易,你必須在再次購買前出售掉之前的股票。

示例 3:
輸入: [7,6,4,3,1]
輸出: 0
解釋: 在這種情況下, 沒有交易完成, 所以最大利潤為 0。

解答思路(貪心算法):

  1. minPrice指向當(dāng)前的最低價(jià)(假設(shè)為買入),sum記錄當(dāng)前盈利的最大值。
  2. minPrice初始化為數(shù)組第一個(gè)值。
  3. 當(dāng)前值<minPrice,此時(shí)賣出會(huì)賠錢,因此不能賣出,將當(dāng)前值賦值給minPrice。
  4. 當(dāng)前值>minPrice,此時(shí)賣出可以賺錢,因此賣出,將當(dāng)前值賦值給minPrice。

代碼實(shí)現(xiàn):

    public int maxProfit(int[] prices) {
        if(prices==null || prices.length==0){
            return 0;
        }
        
        int sum = 0;
        int minPrice = prices[0];
        for(int i = 1;i<prices.length;i++){
            if(prices[i]>minPrice){
                // 此時(shí)賣出。
                sum += prices[i] - minPrice;
                minPrice = prices[i];
            }else{
                // 以當(dāng)前價(jià)格買入
                minPrice = prices[i];
            }
        }

        return sum;
    }


3. 旋轉(zhuǎn)數(shù)組

描述:
給定一個(gè)數(shù)組,將數(shù)組中的元素向右移動(dòng) k 個(gè)位置,其中 k 是非負(fù)數(shù)。

示例 1:
輸入: [1,2,3,4,5,6,7] 和 k = 3
輸出: [5,6,7,1,2,3,4]
解釋:
向右旋轉(zhuǎn) 1 步: [7,1,2,3,4,5,6]
向右旋轉(zhuǎn) 2 步: [6,7,1,2,3,4,5]
向右旋轉(zhuǎn) 3 步: [5,6,7,1,2,3,4]

解題思路:

  1. 如果K為數(shù)組長度的整倍數(shù),那么無需做任何操作。
  2. k = k%length
  3. 循環(huán)右移數(shù)組,將數(shù)組的最后一個(gè)值賦值給數(shù)組第一個(gè)值。
    public void rotate(int[] nums, int k) {
        if (null == nums || nums.length == 0) {
            return;
        }
        if (k % nums.length == 0) {
            return;
        }
        k = k % nums.length;
        int last;
        for (int i = 0; i < k; i++) {
            last = nums[nums.length - 1];
            for (int j = nums.length - 1; j > 0; j--) {
                nums[j] = nums[j - 1];
            }
            nums[0] = last;
        }
    }

4. 只出現(xiàn)一次的數(shù)字

給定一個(gè)非空整數(shù)數(shù)組,除了某個(gè)元素只出現(xiàn)一次以外,其余每個(gè)元素均出現(xiàn)兩次。找出那個(gè)只出現(xiàn)了一次的元素。

說明:
你的算法應(yīng)該具有線性時(shí)間復(fù)雜度。 你可以不使用額外空間來實(shí)現(xiàn)嗎?

解題思路:

  1. 使用數(shù)學(xué)上的異或的特點(diǎn),兩個(gè)相同的數(shù)字異或等于0;
  2. 11=0,11^2=2

代碼:

    public int singleNumber(int[] nums) {
        if (nums == null || nums.length == 0) {
            return -1;
        }
        int sigleNum = nums[0];
        for (int i = 1; i < nums.length; i++) {
            sigleNum ^= nums[i];
        }
        return sigleNum;
    }


5. 兩個(gè)數(shù)組的交集

描述:
給定兩個(gè)數(shù)組,編寫一個(gè)函數(shù)來計(jì)算它們的交集。

解題思路:

  1. 先排序,然后使用三個(gè)指針:i,j,z來遍歷保存數(shù)據(jù)。
    // 兩個(gè)數(shù)組的交集
    public int[] intersect(int[] nums1, int[] nums2) {
        if (nums1 == null || nums2 == null) {
            return new int[0];
        }
        Arrays.sort(nums1);
        Arrays.sort(nums2);
        int[] result;
        if (nums1.length > nums2.length) {
            result = new int[nums2.length];
        } else {
            result = new int[nums1.length];
        }

        int i = 0, j = 0, z = 0;
        while (i < nums1.length && j < nums2.length) {
            if (nums1[i] == nums2[j]) {
                result[z++] = nums1[i];
                i++;
                j++;
            } else if (nums1[i] > nums2[j]) {
                j++;
            } else {
                i++;
            }
        }
        int realResult[] = new int[z];
        for (int index = 0; index < z; index++) {
            realResult[index] = result[index];

        }
        return realResult;

    }


6. 加1

給定一個(gè)由整數(shù)組成的非空數(shù)組所表示的非負(fù)整數(shù),在該數(shù)的基礎(chǔ)上加一。

最高位數(shù)字存放在數(shù)組的首位, 數(shù)組中每個(gè)元素只存儲(chǔ)一個(gè)數(shù)字。

你可以假設(shè)除了整數(shù) 0 之外,這個(gè)整數(shù)不會(huì)以零開頭。

解題思路:

  1. 從數(shù)組尾部遍歷,職要當(dāng)前位不為9,那么當(dāng)前位+1,循環(huán)結(jié)束。
  2. 如果當(dāng)前位為9,則當(dāng)前位設(shè)置為0,繼續(xù)向前遍歷。
    public int[] plusOne(int[] digits) {
        if (digits == null || digits.length == 0) {
            return new int[0];
        }
        for (int i = digits.length - 1; i >= 0; i--) {
            if (digits[i] == 9) {
                digits[i] = 0;
            } else {
                digits[i] += 1;
                return digits;
            }
        }
        int[] newDigits = new int[digits.length + 1];
        for (int i = 0; i < digits.length; i++) {
            newDigits[i] = digits[i];
        }
        newDigits[0] = 1;
        return newDigits;
    }


7. 移動(dòng)零

給定一個(gè)數(shù)組 nums,編寫一個(gè)函數(shù)將所有 0 移動(dòng)到數(shù)組的末尾,同時(shí)保持非零元素的相對順序。

示例:

輸入: [0,1,0,3,12]
輸出: [1,3,12,0,0]
說明:

必須在原數(shù)組上操作,不能拷貝額外的數(shù)組。
盡量減少操作次數(shù)。

解題思路:

  1. 使用雙指針循環(huán)遍歷。
    public void moveZeroes(int[] nums) {
        if (nums == null || nums.length == 0) {
            return;
        }

        int index = 0, start = 0;
        while (start < nums.length) {
            if (nums[start] != 0) {
                nums[index++] = nums[start];
            }
            start++;
        }

        while (index < nums.length) {
            nums[index] = 0;
            index++;
        }

    }

8. 兩數(shù)之和

給定一個(gè)整數(shù)數(shù)組和一個(gè)目標(biāo)值,找出數(shù)組中和為目標(biāo)值的兩個(gè)數(shù)。
你可以假設(shè)每個(gè)輸入只對應(yīng)一種答案,且同樣的元素不能被重復(fù)利用。

解題思路:

  1. 使用Map,遍歷放入時(shí)檢查map中是否已經(jīng)存在余數(shù)值,存在立即返回;不存在繼續(xù)遍歷。
    public int[] twoSum(int[] nums, int target) {
        Map<Integer, Integer> map = new HashMap<>();
        for (int i = 0; i < nums.length; i++) {
            if (map.containsKey(target - nums[i])) {
                if (map.get(target - nums[i]) != i) {
                    return new int[]{map.get(target - nums[i]), i};
                }
            }
            map.put(nums[i], i);
        }
        return new int[]{-1, -1};
    }


9. 有效的數(shù)獨(dú)

判斷一個(gè) 9x9 的數(shù)獨(dú)是否有效。只需要根據(jù)以下規(guī)則,驗(yàn)證已經(jīng)填入的數(shù)字是否有效即可。

數(shù)字 1-9 在每一行只能出現(xiàn)一次。
數(shù)字 1-9 在每一列只能出現(xiàn)一次。
數(shù)字 1-9 在每一個(gè)以粗實(shí)線分隔的 3x3 宮內(nèi)只能出現(xiàn)一次。

解題思路:
使用三個(gè)數(shù)組來記錄是否已經(jīng)存在該值的信息。
rowFlag[9][9]:rowFlag[i][c],第i行是否出現(xiàn)了值c。
colFlag[9][9]:rowFlag[c][j],c出現(xiàn)在第j列。
cellFalg[9][9]:cell[3(i/3)+j/3],第3(i/3)+j/3個(gè)單元中出現(xiàn)了c。

    public boolean isValidSudoku(char[][] board) {
        boolean rowFlag[][] = new boolean[9][9];
        boolean colFlag[][] = new boolean[9][9];
        boolean cellFlag[][] = new boolean[9][9];

        for (int i = 0; i < board.length; i++) {
            for (int j = 0; j < board[i].length; i++) {
                if (board[i][j] >= '1' && board[i][j] <= '9'){
                    int c = board[i][j] - '1';
                    if (rowFlag[i][c] || colFlag[j][c] || cellFlag[3 * (i / 3) + j / 3][c]) {
                        return false;
                    }
                    rowFlag[i][c] = true;
                    colFlag[j][c] = true;
                    cellFlag[3 * (i / 3) + j / 3][c] = true;
                }

            }
        }

        return true;

    }


10. 旋轉(zhuǎn)數(shù)組

給定一個(gè) n × n 的二維矩陣表示一個(gè)圖像。

將圖像順時(shí)針旋轉(zhuǎn) 90 度。

說明:

你必須在原地旋轉(zhuǎn)圖像,這意味著你需要直接修改輸入的二維矩陣。請不要使用另一個(gè)矩陣來旋轉(zhuǎn)圖像。

給定 matrix =
[
[1,2,3],
[4,5,6],
[7,8,9]
],

原地旋轉(zhuǎn)輸入矩陣,使其變?yōu)?
[
[7,4,1],
[8,5,2],
[9,6,3]
]

    public void rotate(int[][] matrix) {

        // 首先沿著對角線旋轉(zhuǎn)
        for (int i = 0; i < matrix.length; i++) {
            for (int j = i + 1; j < matrix.length; j++) {
                int temp = matrix[i][j];
                matrix[i][j] = matrix[j][i];
                matrix[j][i] = temp;
            }
        }

        // 再沿著數(shù)組垂直中線進(jìn)行旋轉(zhuǎn)
        for (int i = 0; i < matrix.length; i++) {
            for (int j = 0; j < matrix.length / 2; j++) {
                int temp = matrix[i][j];
                matrix[i][j] = matrix[i][matrix.length - 1 - j];
                matrix[i][matrix.length - 1 - j] = temp;
            }
        }

    }


11. 三數(shù)之和

給定一個(gè)包含 n 個(gè)整數(shù)的數(shù)組 nums,判斷 nums 中是否存在三個(gè)元素 a,b,c ,使得 a + b + c = 0 ?找出所有滿足條件且不重復(fù)的三元組。

注意:答案中不可以包含重復(fù)的三元組。

例如, 給定數(shù)組 nums = [-1, 0, 1, 2, -1, -4],
滿足要求的三元組集合為:
[
  [-1, 0, 1],
  [-1, -1, 2]
]

解題思路:

  1. 首先對數(shù)組進(jìn)行排序,逐個(gè)遍歷數(shù)組,以index上的數(shù)據(jù)nums[index]為例分析:
  2. 如果nums[index]>0,結(jié)束循環(huán),因?yàn)榇撕蟮臄?shù)組值必大于0。
  3. 如果nums[index]<=0,定義target = 0-nums[index],兩數(shù)之和等于target的過程。
  4. 我們使用雙指針來解決兩數(shù)之和為target的求解過程。
public List<List<Integer>> threeSum(int[] nums) {
        List<List<Integer>> re = new ArrayList<>();
        if (nums == null || nums.length < 3) {
            return re;
        }

        // 1. 首先排序
        Arrays.sort(nums);
        for (int i = 0; i < nums.length; i++) {
            // 選取第一個(gè)值
            if (nums[i] > 0) break;
            // 去除重復(fù)
            if (i >= 1) if (nums[i] == nums[i - 1]) continue;
            int target = 0 - nums[i];
            int low = i + 1;
            int high = nums.length - 1;
            while (low < high) {
                if (nums[low] + nums[high] == target) {
                    List<Integer> list = new ArrayList();
                    list.add(nums[i]);
                    list.add(nums[low]);
                    list.add(nums[high]);
                    re.add(list);
                    // 為了去重復(fù)。
                    while (low + 1 < high && nums[low] == nums[low + 1]) {
                        low++;
                    }
                    // 為了去重復(fù)。
                    while (high - 1 > low && nums[high] == nums[high - 1]) {
                        high--;
                    }
                    low++;
                    high--;
                } else if (nums[low] + nums[high] < target) {
                    low++;
                } else {
                    high--;
                }
            }

        }
        return re;
    }

12. 矩陣置零

給定一個(gè) m x n 的矩陣,如果一個(gè)元素為 0,則將其所在行和列的所有元素都設(shè)為 0。請使用原地算法。

示例 1:

輸入:
[
[1,1,1],
[1,0,1],
[1,1,1]
]
輸出:
[
[1,0,1],
[0,0,0],
[1,0,1]
]

示例 2:

輸入:
[
[0,1,2,0],
[3,4,5,2],
[1,3,1,5]
]
輸出:
[
[0,0,0,0],
[0,4,5,0],
[0,3,1,0]
]
進(jìn)階:

一個(gè)直接的解決方案是使用 O(mn) 的額外空間,但這并不是一個(gè)好的解決方案。
一個(gè)簡單的改進(jìn)方案是使用 O(m + n) 的額外空間,但這仍然不是最好的解決方案。
你能想出一個(gè)常數(shù)空間的解決方案嗎?

解題思路:

  1. 我們直接使用常數(shù)空間來解決問題。
  2. 首先我們借助s0行和0列來存儲(chǔ)哪些行和類稍后需要置為0。
  3. 其次我們定義兩個(gè)遍歷來分別記錄0行0列原來是否出現(xiàn)過0。
public void setZeroes(int[][] matrix) {
        int rows = matrix.length;
        int cols = matrix[0].length;
        boolean firstRowIsZero = false;
        boolean firstColIsZero = false;
        for (int i = 0; i < rows; i++) {
            for (int j = 0; j < cols; j++) {
                if (matrix[i][j] == 0) {
                    if (i != 0 && j != 0) {
                        matrix[i][0] = 0;
                        matrix[0][j] = 0;
                    } else {
                        firstColIsZero = j == 0 ? true : firstColIsZero;
                        firstRowIsZero = i == 0 ? true : firstRowIsZero;
                    }
                }
            }
        }
        for (int i = 1; i < rows; i++) {
            for (int j = 1; j < cols; j++) {
                if (matrix[0][j] == 0 || matrix[i][0] == 0) {
                    matrix[i][j] = 0;
                }
            }
        }

        //第一列含0
        if(firstColIsZero){
            for(int i=0;i<matrix.length;i++){
                matrix[i][0] = 0;
            }
        }
        //第一行含0
        if(firstRowIsZero){
            for(int j=0;j<matrix[0].length;j++){
                matrix[0][j] = 0;
            }
        }
    }

13. 字謎分組

給定一個(gè)字符串?dāng)?shù)組,將字母異位詞組合在一起。字母異位詞指字母相同,但排列不同的字符串。

示例:

輸入: ["eat", "tea", "tan", "ate", "nat", "bat"],
輸出:
[
["ate","eat","tea"],
["nat","tan"],
["bat"]
]
說明:

所有輸入均為小寫字母。
不考慮答案輸出的順序。

解題思路:

  1. 怎樣判斷異位詞呢?我們之前做過判斷兩個(gè)詞是否為異位詞的題目,如果按照暴力搜索比對的話,肯定是要超時(shí)的。
  2. 怎樣判斷兩個(gè)詞為異位詞呢?我們采用對字符串的字節(jié)數(shù)組進(jìn)行排序的方式來判斷兩個(gè)字符串是否為異位詞。
  3. 使用Mapl來完成分組存儲(chǔ)異位詞。
    public List<List<String>> groupAnagrams(String[] strs) {
        if (strs == null || strs.length == 0) {
            return new ArrayList<List<String>>();
        }
        Map<String, List<String>> map = new HashMap<>();
        for (int i = 0; i < strs.length; i++) {
            char[] chars = strs[i].toCharArray();
            Arrays.sort(chars);
            String key = String.valueOf(chars);
            if (map.containsKey(key) == false) {
                map.put(key, new ArrayList<String>());
            }
            map.get(key).add(strs[i]);
        }
        return new ArrayList<>(map.values());
    }

14. 無重復(fù)字符的最長子串

給定一個(gè)字符串,找出不含有重復(fù)字符的最長子串的長度。

示例 1:

輸入: "abcabcbb"
輸出: 3
解釋: 無重復(fù)字符的最長子串是 "abc",其長度為 3。
示例 2:

輸入: "bbbbb"
輸出: 1
解釋: 無重復(fù)字符的最長子串是 "b",其長度為 1。

解題思路:

  1. 核心問題是如果出現(xiàn)了重復(fù),我們的start位置從何處計(jì)算。
  2. 因?yàn)樽址拈L度不固定,因此我們使用數(shù)組來存儲(chǔ)字符最近一次出現(xiàn)的下一個(gè)位置。
    public int lengthOfLongestSubstring(String s) {
        int n = s.length(), ans = 0;
        // current index of character
        int[] index = new int[128]; 
        // try to extend the range [i, j]
        for (int j = 0, i = 0; j < n; j++) {
            i = Math.max(index[s.charAt(j)], i);
            ans = Math.max(ans, j - i + 1);
            index[s.charAt(j)] = j + 1;
        }
        return ans;
    }

14. 遞增的三元子序列

給定一個(gè)未排序的數(shù)組,判斷這個(gè)數(shù)組中是否存在長度為 3 的遞增子序列。

數(shù)學(xué)表達(dá)式如下:

如果存在這樣的 i, j, k, 且滿足 0 ≤ i < j < k ≤ n-1,
使得 arr[i] < arr[j] < arr[k] ,返回 true ; 否則返回 false 。
說明: 要求算法的時(shí)間復(fù)雜度為 O(n),空間復(fù)雜度為 O(1) 。

解題思路:

  1. 使用兩個(gè)指針來解決問題,first代表三元子序列的第一個(gè)值,second代表三元子序列的第二個(gè)值。
  2. first<second<第三個(gè)值。
  3. 只要出現(xiàn)比first小的值,我們就更新first的值。
  4. 出現(xiàn)比first大比second小的值,我們更新second.
  5. 出現(xiàn)比second大的值時(shí),三元子序列就找到了。
    public boolean increasingTriplet(int[] nums) {
        if (nums == null || nums.length < 3) {
            return false;
        }
        if (nums == null || nums.length < 3) {
            return false;
        }
        int first = Integer.MAX_VALUE, second = Integer.MAX_VALUE;
        for (int num : nums) {
            if (first > num) {
                first = num;
            } else if (first < num && second > num) {
                second = num;
            } else if (num > second) {
                return true;
            }
        }
        return false;      
    }
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請結(jié)合常識與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • 前言 2. 實(shí)現(xiàn) Singleton 3. 數(shù)組中重復(fù)的數(shù)字 4. 二維數(shù)組中的查找 5. 替換空格 6. 從尾到...
    Observer_____閱讀 3,152評論 0 1
  • 1、什么是 CSS hack2、談一談瀏覽器兼容的思路3、列舉5種以上瀏覽器兼容的寫法4、以下工具/名詞是做什么的...
  • Android使用OpenGL ES2.0繪制3D圖像或者加載3D模型時(shí),為了達(dá)到立體效果往往需要設(shè)置視見轉(zhuǎn)換矩陣...
    android_opengl閱讀 2,370評論 0 7
  • 睡到凌晨三點(diǎn)卻怎么也睡不著了,輾轉(zhuǎn)反側(cè)也不能寐,索性站到陽臺(tái)上。天是南方特有的燥熱,空氣都是悶悶的,沒有一絲風(fēng)的訊...
    十七又一閱讀 272評論 0 0

友情鏈接更多精彩內(nèi)容