leetcode-443-Median of Two Sorted Arrays

443. String Compression

题目:

Given an array of characters, compress it in-place.

The length after compression must always be smaller than or equal to the original array.

Every element of the array should be a character (not int) of length 1.

After you are done modifying the input array in-place, return the new length of the array.

Follow up:
Could you solve it using only O(1) extra space?

Example1:

Input:
["a","a","b","b","c","c","c"]

Output:
Return 6, and the first 6 characters of the input array should be: ["a","2","b","2","c","3"]

Explanation:
"aa" is replaced by "a2". "bb" is replaced by "b2". "ccc" is replaced by "c3".

Example2:

Input:
["a"]

Output:
Return 1, and the first 1 characters of the input array should be: ["a"]

Explanation:
Nothing is replaced.

Example3:

Input:
["a","b","b","b","b","b","b","b","b","b","b","b","b"]

Output:
Return 4, and the first 4 characters of the input array should be: ["a","b","1","2"].

Explanation:
Since the character "a" does not repeat, it is not compressed. "bbbbbbbbbbbb" is replaced by "b12".
Notice each digit has it's own entry in the array.

解法一

解析

        还是用一个新的数组,先把答案存起来,最后交换chars。

代码(C++)

class Solution {
public:
    int compress(vector& chars) {
        if(chars.size() == 0) return 0;
        vector v;
        v.push_back(chars[0]);
        int sum = 1;
        for(int i = 1;i < chars.size(); i ++){
            if(chars[i-1] == chars[i]&&i != chars.size()-1) sum++;
            else {
                if(chars[i-1] == chars[i]&&i == chars.size()-1) {
                    f(v, sum+1);
                    break;
                }
                if(sum != 1) f(v, sum);
                v.push_back(chars[i]);
                sum = 1;
            }
        }
        swap(v, chars);
        return chars.size();
    }
    void f(vector& v, int n){
        string s = to_string(n);
        for(int i = 0; i < s.length(); i ++)
            v.push_back(s[i]);
    }
};

解法二

解析

实现O(1)的空间复杂度,需要建一个索引cur,直接修改原串,记录压缩后的字符串。遍历字符串,逐个找字符连续出现的范围。先将该字符拷贝到原串的cur上,如果该字符出现一次,不再操作,如果多次,则将该字符的次数追加到cur之后,修改cur。

例如:["a","b","b","c","c","c"]

开始cur = 0,“a”出现一次,无需压缩。chars[cur++] = chars[0]即可。接着"b"出现2次,则有chars[cur++] = chars[1], chars[cur++] = chars[2]。原串中chars[2]变为"2",cur = 3,类似地,有chars[cur++] = chars[3], chars[cur++] = chars[4]。chars[4]变为"3",此时cur = 5,即为压缩后的字符串长度。原串压缩为["a","b","2","c","3"]。

这种压缩方式,字符连续出现频率高,才能有较好的压缩效果。

代码(C++)

class Solution {
public:
    int compress(vector& chars) {
        int n = chars.size();
        int cur = 0; // 记录当前字符的索引,最后为压缩字符串的长度
        for(int i = 0; i < n; ) {
            int j = i;
            while( j < n - 1 && chars[j] == chars[j+1]) {// 查找字符连续相同的个数
                j++;
            }
            chars[cur++] = chars[i];// 将当前字符写入原字符串中
            if(i != j) {
                string times = to_string(j - i + 1);// 字符连续相同的个数
                int tLen = times.length();
                for(int k = 0; k < tLen; k++) {//把字符连续相同个数写入字符串,用来压缩
                    chars[cur++] = times[k];
                }
            }
            i = j + 1;//接着处理下一个字符
        }
        return cur;
    }
};  

代码(python)

class Solution(object):
    def compress(self, chars):
        """
        :type chars: List[str]
        :rtype: int
        """
        n = len(chars) 
        cur = 0 # 当前字符的索引,用以压缩原字符串
        i = 0
        while i < n:
            j = i
            while j < n - 1 and chars[j] == chars[j+1]:# 找字符连续出现的次数
                j += 1
            chars[cur] = chars[i] # 记录当前处理的字符
            cur += 1
            if i != j:
                times = str(j-i+1) # 将字符的次数写入原串中
                tLen = len(times)
                for k in range(tLen):
                    chars[cur+k] = times[k]
                cur += tLen
            i = j + 1 # 处理下一个字符
        return cur

你可能感兴趣的:(leetcode-443-Median of Two Sorted Arrays)