leetcode-38:外观数列

简介: leetcode-38:外观数列

题目:

题目链接

给定一个正整数 n ,输出外观数列的第 n 项。

「外观数列」是一个整数序列,从数字 1 开始,序列中的每一项都是对前一项的描述。

你可以将其视作是由递归公式定义的数字字符串序列:

  • countAndSay(1) = "1"
  • countAndSay(n) 是对 countAndSay(n-1) 的描述,然后转换成另一个数字字符串。

前五项如下:

1.     1
2.     11
3.     21
4.     1211
5.     111221
第一项是数字 1 
描述前一项,这个数是 1 即 “ 一 个 1 ”,记作 "11"
描述前一项,这个数是 11 即 “ 二 个 1 ” ,记作 "21"
描述前一项,这个数是 21 即 “ 一 个 2 + 一 个 1 ” ,记作 "1211"
描述前一项,这个数是 1211 即 “ 一 个 1 + 一 个 2 + 二 个 1 ” ,记作 "111221"

要 描述 一个数字字符串,首先要将字符串分割为 最小 数量的组,每个组都由连续的最多 相同字符 组成。然后对于每个组,先描述字符的数量,然后描述字符,形成一个描述组。要将描述转换为数字字符串,先将每组中的字符数量用数字替换,再将所有描述组连接起来。

例如,数字字符串 “3322251” 的描述如下图:

示例 1:

输入:n = 1
输出:"1"
解释:这是一个基本样例。

示例 2:

输入:n = 4
输出:"1211"
解释:
countAndSay(1) = "1"
countAndSay(2) = 读 "1" = 一 个 1 = "11"
countAndSay(3) = 读 "11" = 二 个 1 = "21"
countAndSay(4) = 读 "21" = 一 个 2 + 一 个 1 = "12" + "11" = "1211"

解题

方法一:递归

def countAndSay(self, n: int) -> str:
    if n == 1:
        return '1'
    s = self.countAndSay(n-1)
    i, res = 0, ''
    for j, c in enumerate(s):
        if c != s[i]:
            res += str(j-i) + s[i]
            i = j
    res += str(len(s) - i) + s[-1]  # 最后一个元素莫忘统计
    return res

方法二:迭代

def countAndSay(self, n: int) -> str:
    res = '1'
    for _ in range(n-1):  # 控制循环次数
        i, tmp = 0, ''
        for j, c in enumerate(res):
            if c != res[i]:
                tmp += str(j-i) + res[i]
                i = j
        res = tmp + str(len(res) - i) + res[-1]
    return res

方法三:正则表达式:提取元素

def countAndSay(self, n: int) -> str:
    if n == 1:
        return '1'
    s = self.countAndSay(n-1)
    p = r'(\d)\1*'
    pattern = re.compile(p)
    res = [_.group() for _ in pattern.finditer(s)]  # 提取结果
    return ''.join(str(len(c)) + c[0] for c in res)  # join 内部的 str(len(c)) + c[0] for c in res 是生成器类型

字符串 (\d)\1* 可以用来匹配结果。这里用来提取连在一块的元素, 如 '111221',提取出的元素是 res = ['111', '22', '1']。

然后再返回我们要组装的字符串。

方法四:正则表达式:元素替换

def countAndSay(self, n: int) -> str:
    res = '1'
    p = r'(\d)\1*'
    pattern = re.compile(p)
    for _ in range(n-1):
        res = pattern.sub(lambda x: str(len(x.group())) + x.group(1), res)  # 替换
    return res

对于该句:res = pattern.sub(lambda x: str(len(x.group())) + x.group(1), res),还是拿上面的举例,对于 111221 的第一个匹配项 111,其中 group() 匹配的是 111(全局匹配),而 group(1) 匹配到的是 1 (第一个捕获组,三个 1 中的第一个)。

相关文章
|
算法 安全 Swift
LeetCode - #38 外观数列
不积跬步,无以至千里;不积小流,无以成江海,Swift社区 伴你前行。如果大家有建议和意见欢迎在文末留言,我们会尽力满足大家的需求。
LeetCode - #38 外观数列
|
存储 Python
LeetCode 38. 外观数列
给定一个正整数 n ,输出外观数列的第 n 项。
82 0
LeetCode 665.非递减数列
LeetCode 665.非递减数列
77 0
LeetCode 665.非递减数列
|
算法
LeetCode 38外观数列&39组合总和
给定一个正整数 n(1 ≤ n ≤ 30),输出外观数列的第 n 项。
85 0
LeetCode 38外观数列&39组合总和
|
存储 算法 C语言
想要去欺负Leetcode的这些年——第二次,看看咱们是怎么把数组玩转,把数列理解透彻的~
想要去欺负Leetcode的这些年——第二次,看看咱们是怎么把数组玩转,把数列理解透彻的~
127 0
想要去欺负Leetcode的这些年——第二次,看看咱们是怎么把数组玩转,把数列理解透彻的~
LeetCode:665. 非递减数列
LeetCode:665. 非递减数列
79 0
Leetcode_Python 665 非递减数列
分析:题目让最多改变一个情况下,满足非递减数列。遍历数组,可能存在以下情况。
101 0
☆打卡算法☆LeetCode 38、外观数列 算法解析
“给定一个正整数n,输出外观数列的第n项。”
|
算法
​LeetCode刷题实战38: 外观数列
算法的重要性,我就不多说了吧,想去大厂,就必须要经过基础知识和业务逻辑面试+算法面试。所以,为了提高大家的算法能力,这个公众号后续每天带大家做一道算法题,题目就从LeetCode上面选 !
64 0
​LeetCode刷题实战38: 外观数列