394.字符串解码

2020-03-27
字符串解码

给定一个经过编码的字符串,返回它解码后的字符串。

编码规则为: 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: 递归
暴力的一层一层解开[]
 
var decodeString = function (s) {
  let fn = (k, str) => {
    let result = ''; // 要返回的字符串
    let l, r; // 记录左右括号的位置
    while (str.indexOf('[') > -1) { // 当当前字符串还有[时 继续处理字符串
      let tmp = 1; // 临时变量 记录当前是否找到闭合的[]
      for (let i = 0; i < str.length; i++) { // 循环字符串
        if (str[i] === '[') { // 当为[时记录l的位置
          l = i;
        }
        else if (str[i] === ']') { // 当第一次出现 ] 时, tmp设为0 也就是找到了第一个闭合的[]
          tmp = 0;
          r = i;
        } else continue;
        if (tmp === 0) { // 当找到了闭合的[]时 ,l, r 分别记录的是[]的左右位置
          let j = 1; // 记录[前面几位是数字
          while (/[0-9]/.test(str[l - (j + 1)]) && l - j >= 0) {
            j++ // 如果[前面一直是数字,则继续往前找
          }
          const k = parseInt(str.slice(l - j, l)); // k是[前面数字的大小
          let sonStr = str.slice(l + 1, r); // sonStr是[]之间的字符串
          str = str.slice(0, l - j) + fn(k, sonStr) + str.slice(r + 1); // 传入的字符串等于 数字左侧的 + []之间和k乘积之后的 + ]之后的
          break; // 每处理完一个[], break出去继续找下一个 直到str中没有[]
        }
      }
    }
    for (let i = 0; i < k; i++) { // k是倍数,要把结果循环k次
      result += str;
    }
    return result; // result是每一个[]处理之后的结果
  }
  return fn(1, s); // s传入 相对于 1[s]的类型
};

 

思路2: 栈+深度优先搜索
用出栈入栈的思维解开[]
 
var decodeString = function (s) {
  let stock = [], res = '', multi = 0; // stock是辅助栈
  for (let c of s) { // 遍历字符串 s 中每个字符 c;
    if (c === '[') { // 当 c 为 [ 时,将当前 multi 和 res 入栈,并分别置空置 0
      // 记录此 [ 前的倍数 multi 至栈,用于发现对应 ] 后,获取 multi × [...] 字符串
      // 记录此 [ 前的临时结果 res 至栈,用于发现对应 ] 后的拼接操作;
      stock.push([multi, res]); // 进入到新 [ 后,res 和 multi 重新记录。
      multi = 0;
      res = '';
    } else if (c === ']') { // 当 c 为 ] 时,stack 出栈,拼接字符串 res = last_res + cur_multi * res,其中:
      let tmp = stock.pop();
      const last_res = tmp[1]; // last_res是上个 [ 到当前 [ 的字符串,例如 "3[a2[c]]" 中的 a;
      let cur_multi = tmp[0]; // cur_multi是当前 [ 到 ] 内字符串的重复倍数,例如 "3[a2[c]]" 中的 2。
      let ss = res; // ss记录循环后的res
      while(cur_multi > 1) {
        ss += res;
        cur_multi--;
      }
      res = last_res + ss; // res变为 上一次存入栈中的last_res拼接循环后的res
    } else if(/[0-9]/.test(c)) {
      multi = multi * 10 + parseInt(c); // 当 c 为数字时,将数字字符转化为数字 multi,用于后续倍数计算;
    } else {
      res += c; // 当 c 为字母时,在 res 尾部添加 c;
    }
  }
  return res;
}

 

posted @ 2020-03-27 10:57  蓝小胖纸  阅读(220)  评论(0编辑  收藏  举报