leetcode *394. 字符串解码(2020.5.28)

这篇博客介绍了LeetCode 394题目的解法,主要探讨了两种不同的解题思路:单栈和双栈策略。通过详细解释编码规则和示例,展示了如何将编码字符串正确解码回原始字符串。在单栈策略中,利用栈处理数字和字符,遇到右括号时将相应内容转化为字符串并压入栈中。而在双栈策略中,分别使用数字栈和字符串栈来处理编码信息。

【题目】*394. 字符串解码

给定一个经过编码的字符串,返回它解码后的字符串。
编码规则为: k[encoded_string],表示其中方括号内部的 encoded_string 正好重复 k 次。注意 k 保证为正整数。
你可以认为输入字符串总是有效的;输入字符串中没有额外的空格,且输入的方括号总是符合格式要求的。
此外,你可以认为原始数据不包含数字,所有的数字只表示重复的次数 k ,例如不会出现像 3a 或 2[4] 的输入。

示例:
s = “3[a]2[bc]”, 返回 “aaabcbc”.
s = “3[a2[c]]”, 返回 “accaccacc”.
s = “2[abc]3[cd]ef”, 返回 “abcabccdcdcdef”.

【解题思路1】单栈

用栈,碰到数字、字母、左中括号都压入栈中,碰到右括号开始弹出,先弹出拿到字符串,再拿到数字(重复次数),把这一段转成字符串压入栈中,继续执行,最后把栈内存的reverse一下

class Solution {
    int ptr;

    public String decodeString(String s) {
        LinkedList<String> stk = new LinkedList<String>();
        ptr = 0;

        while (ptr < s.length()) {
            char cur = s.charAt(ptr);
            if (Character.isDigit(cur)) {
                // 遇到数字,进栈
                String digits = getDigits(s);
                stk.addLast(digits); //addLast()将指定元素追加在此列表的末尾
            } else if (Character.isLetter(cur) || cur == '[') {
                // 遇到字母或者左括号,进栈
                stk.addLast(String.valueOf(s.charAt(ptr++))); 
            } else {
                ++ptr;
                LinkedList<String> sub = new LinkedList<String>();
                //遇到右括号开始出栈,一直出栈知道遇到左括号
                while (!"[".equals(stk.peekLast())) {
                    sub.addLast(stk.removeLast());
                }
                //将出栈的字母反转
                Collections.reverse(sub);
                // 左括号出栈
                stk.removeLast();
                // 此时栈顶为当前 sub 对应的字符串应该出现的次数
                int repTime = Integer.parseInt(stk.removeLast());
                StringBuffer t = new StringBuffer();
                String o = getString(sub);
                // 构造重复n次的字符串
                while (repTime-- > 0) {
                    t.append(o);
                }
                // 将构造好的字符串入栈
                stk.addLast(t.toString());
            }
        }

        return getString(stk);
    }

    public String getDigits(String s) {
        StringBuffer ret = new StringBuffer();
        while (Character.isDigit(s.charAt(ptr))) {
            ret.append(s.charAt(ptr++));
        }
        return ret.toString();
    }

    public String getString(LinkedList<String> v) {
        StringBuffer ret = new StringBuffer();
        for (String s : v) {
            ret.append(s);
        }
        return ret.toString();
    }
}

最后在反转

class Solution {
    public static String decodeString(String s) {
        char[] chs = s.toCharArray();

        //用 Object
        Stack<Object> stack = new Stack<>();

        // 计算中括号前的数字是多少,不一定是一位数。
        int num = 0;

        for (char c : chs) {
            if (Character.isDigit(c)) {
                // 1. 数字则直接计算
                num = num * 10 + c - '0';
            } else if (c == '[') {
                // 2. 左括号, 先要把前面的数字放进去, 左中括号不用入栈
                stack.push(num);
                num = 0;
            } else if (c == ']') {
                // 3. 右括号, 出栈, 获取局部字符串再根据前面的数字得到乘次数再放入stack
                String str = popAndGetString(stack);
                int times = (int) stack.pop();
                String temp = String.join("", Collections.nCopies(times, str));
                stack.push(temp);
            } else {
                // 4. 正常字符, 放String类型
                stack.push(String.valueOf(c));
            }
        }
        return new StringBuilder(popAndGetString(stack)).reverse().toString();
    }

    // 这边一种情况是,前面可能已经变成正序的了,但是后面还有,后面来了,然后在这个方法中reverse一下就又变反了
    // 那我干脆只在最终结果处reverse,其他过程不reverse
    private static String popAndGetString(Stack<Object> stack) {
        StringBuilder sb = new StringBuilder();
        while (!stack.isEmpty() && stack.peek() instanceof String) {
            sb.append(stack.pop());
        }
        return sb.toString();
    }
}

【解题思路2】双栈

一个数字栈,一个字符串栈
出栈时遇到左括号,去数字栈取出一个数字

class Solution {
   public String decodeString(String s) {
		StringBuffer ans=new StringBuffer();
		Stack<Integer> multiStack=new Stack<>();
		Stack<StringBuffer> ansStack=new Stack<>();
		int multi=0;
		for(char c:s.toCharArray()){
			if(Character.isDigit(c)){
				multi=multi*10+c-'0';
			}else if(c=='['){
				ansStack.add(ans);
				multiStack.add(multi);
				ans=new StringBuffer();
				multi=0;
			}else if(Character.isAlphabetic(c)){
				ans.append(c);
			}else{
				StringBuffer ansTmp=ansStack.pop();
				int tmp=multiStack.pop();
				for(int i=0;i<tmp;i++)ansTmp.append(ans);
				ans=ansTmp;
			}
		}
		return ans.toString();
	}
}
### 解法分析 LeetCode 394题要求实现字符串解码功能,输入格式为 `k[encoded_string]`,其中 `k` 是正整数,表示重复次数,`encoded_string` 是需要重复的内容。解码后返回重复 `k` 次的字符串。 该问题可以使用 **栈(Stack)** 来解决,主要思路是利用两个栈分别保存当前的数字和字符串状态。遍历输入字符串,遇到数字时保存到数字栈中,遇到字符时拼接到当前字符串,遇到 `[` 时保存当前字符串和数字到栈中,遇到 `]` 时从栈中取出数据进行解码拼接。 --- ### Java 解法实现 以下是一个清晰的实现方式,使用 `LinkedList` 作为栈结构来存储数字和字符串: ```java class Solution { public String decodeString(String s) { LinkedList<Integer> numStack = new LinkedList<>(); LinkedList<String> strStack = new LinkedList<>(); StringBuilder sb = new StringBuilder(); int num = 0; for (char c : s.toCharArray()) { if (Character.isDigit(c)) { num = num * 10 + (c - '0'); } else if (c == '[') { numStack.push(num); strStack.push(sb.toString()); sb = new StringBuilder(); num = 0; } else if (c == ']') { int repeat = numStack.pop(); String temp = sb.toString(); sb = new StringBuilder(); for (int i = 0; i < repeat; i++) { sb.append(temp); } sb.insert(0, strStack.pop()); } else { sb.append(c); } } return sb.toString(); } } ``` --- ### 示例输入与输出 以下是一些测试用例及输出结果: - 输入:`"3[a]2[bc]"` 输出:`"aaabcbc"` - 输入:`"3[a2[c]]"` 输出:`"accaccacc"` - 输入:`"2[abc]3[cd]ef"` 输出:`"abcabccdcdcdef"` 这些测试用例可以验证解码逻辑是否正确[^5]。 --- ### 算法解析 1. **初始化栈**:使用两个栈 `numStack` 和 `strStack` 分别保存当前数字和字符串状态。 2. **数字处理**:遇到数字字符时,将其转换为整数并保存。 3. **左括号处理**:遇到 `[` 时,将当前数字和字符串压入栈中,并重置当前字符串和数字。 4. **右括号处理**:遇到 `]` 时,取出栈顶的数字和字符串,进行解码拼接。 5. **字符处理**:遇到字母时,直接追加到当前字符串中。 --- ### 时间与空间复杂度 - **时间复杂度**:`O(n)`,其中 `n` 是输入字符串的长度。每个字符最多被处理一次。 - **空间复杂度**:`O(n)`,栈和字符串构建的空间开销。 --- ### 递归解法(可选) 除了栈结构,还可以使用递归方法实现字符串解码。递归的核心思想是每次遇到 `[` 时进入下一层递归,遇到 `]` 时返回当前层的结果。 ```java class Solution { private int ptr = 0; public String decodeString(String s) { StringBuilder sb = new StringBuilder(); int num = 0; while (ptr < s.length()) { char c = s.charAt(ptr); if (Character.isDigit(c)) { num = num * 10 + (c - '0'); } else if (c == '[') { ptr++; String decoded = decodeString(s); for (int i = 0; i < num; i++) { sb.append(decoded); } num = 0; } else if (c == ']') { ptr++; return sb.toString(); } else { sb.append(c); } ptr++; } return sb.toString(); } } ``` --- ### 总结 LeetCode 394题的核心在于字符串的嵌套解析,栈结构能够很好地保存当前状态,而递归方法则利用函数调用栈实现类似效果。两种方法都可以高效解决问题,选择哪种方式取决于个人编码习惯和性能需求。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值