【python】Leetcode每日一题-不同的子序列
【python】Leetcode每日一题-不同的子序列
【题目描述】
给定一个字符串 s
和一个字符串 t
,计算在 s
的子序列中 t
出现的个数。
字符串的一个 子序列 是指,通过删除一些(也可以不删除)字符且不干扰剩余字符相对位置所组成的新字符串。(例如,"ACE"
是 "ABCDE"
的一个子序列,而 "AEC"
不是)
题目数据保证答案符合 32
位带符号整数范围。
示例1:
输入:s = "rabbbit", t = "rabbit"
输出:3
解释:
如下图所示, 有 3 种可以从 s 中得到 "rabbit" 的方案。
(上箭头符号 ^ 表示选取的字母)
rabbbit
^^^^ ^^
rabbbit
^^ ^^^^
rabbbit
^^^ ^^^
示例2:
输入:s = "babgbag", t = "bag"
输出:5
解释:
如下图所示, 有 5 种可以从 s 中得到 "bag" 的方案。
(上箭头符号 ^ 表示选取的字母)
babgbag
^^ ^
babgbag
^^ ^
babgbag
^ ^^
babgbag
^ ^^
babgbag
^^^
提示:
0 <= s.length, t.length <= 1000
s 和 t 由英文字母组成
【分析】
【超时】思路:
自己原先思路是对
s
和t
顺序匹配,使用dfs
遍历,最后时间超时,分析时间复杂度,代码模型为\(O(m^n)\),当然比这个小很多,并不是完整的树结构。代码:
class Solution(object):
s = ""
t = ""
num = 0
def numDistinct(self, s, t):
"""
:type s: str
:type t: str
:rtype: int
"""
self.s = s
self.t = t
self.dfs(0, 0)
return self.num def dfs(self, s_index, t_index):
if(t_index == len(self.t)):
self.num += 1
return
for x in range(len(self.s)-s_index):
if(self.s[x+s_index] == self.t[t_index]):
self.dfs(x+s_index+1, t_index+1)
【dp】思路:
dp[i][j]
表示s[0...i]
、t[0...j]
之间的子序列数。提供dp转移方程:
\[\begin{cases}
dp[i][j] = dp[i - 1][j - 1] + dp[i - 1][j];&s[i] == t[j]\\
dp[i][j] = dp[i - 1][j];&s[i] != t[j]
\end{cases}
\]- 解释:当
s[i] == t[j]
时,dp[i][j]
可分为s
串尾部是否形成匹配两种情况,如果在与t匹配中形成匹配了,则为子序列为dp[i-1][j-1]
,即除开s
、t
两串最后一个字母的子序列数,如果未形成匹配,则为dp[i-1][j]
,即s
、t
两串不包含s串最后一个字母的子序列数。 - 时间复杂度:\(O(mn)\)
AC代码:
class Solution(object):
def numDistinct(self, s, t):
"""
:type s: str
:type t: str
:rtype: int
"""
m = len(s)
n = len(t)
dp = [[0]*(n+1) for i in range(m+1)]
for i in range(m+1):
dp[i][0] = 1
for i in range(1, m+1):
for j in range(1, n+1):
if(j > i):
continue
if(s[i-1] == t[j-1]):
dp[i][j] = dp[i-1][j-1] + dp[i-1][j]
else:
dp[i][j] = dp[i-1][j]
return dp[m][n]
讨论:
看到一位兄弟的评论,图一乐
去的超时!不过63个测试用例而已,大不了我写63个case
public class Solution {
public int NumDistinct(string s, string t) {
switch(t){
case "bddabdcae":return 10582116;
case "bcddceeeebecbc":return 700531452;
case "ccdeddeabb":return 527764581;
case "baaacbceabba":return 1293119;
case "aeacbde":return 2543265;
case "rwmimatmhydhbujebqehjprrwfkoebcxxqfktayaaeheys":return 543744000;
case "rwmimatmhydhbujebqehjprarwfkoebcxxqfktayaaeheys":return 1456742400;
default:return fun(s,t,0,0);
}
}
private int fun(string s,string t,int i,int j){
int res=0;
for(;i<s.Length&&j<t.Length;i++){
if(s[i]==t[j]){
res+=fun(s,t,i+1,j);
j++;
}
}
if(j<t.Length)return 0;
return res+1;
}
}
看了官方题解,意识到动态规划边界条件的确也很重要:
当
j=n
时,t[j:]
为空字符串,由于空字符串是任何字符串的子序列,因此对任意 \(0 \le i \le m\),有 \(\textit{dp}[i][n]=1\);当
i=m
且j<n
时,s[i:]
为空字符串,t[j:]
为非空字符串,由于非空字符串不是空字符串的子序列,因此对任意 \(0 \le j<n\),有 \(\textit{dp}[m][j]=0\)。
官方题解: 戳这里
【python】Leetcode每日一题-不同的子序列的更多相关文章
- 【python】Leetcode每日一题-最长公共子序列
[python]Leetcode每日一题-最长公共子序列 [题目描述] 给定两个字符串 text1 和 text2,返回这两个字符串的最长 公共子序列 的长度.如果不存在 公共子序列 ,返回 0 . ...
- 【python】Leetcode每日一题-132模式
[python]Leetcode每日一题-132模式 [题目描述] 给定一个整数序列:a1, a2, ..., an,一个132模式的子序列 ai, aj, ak 被定义为:当 i < j &l ...
- 【python】Leetcode每日一题-寻找旋转排序数组中的最小元素
[python]Leetcode每日一题-寻找旋转排序数组中的最小元素 [题目描述] 已知一个长度为 n 的数组,预先按照升序排列,经由 1 到 n 次 旋转 后,得到输入数组.例如,原数组nums ...
- 【python】Leetcode每日一题-删除有序数组中的重复项
[python]Leetcode每日一题-删除有序数组中的重复项 [题目描述] 给你一个有序数组 nums ,请你 原地 删除重复出现的元素,使每个元素 最多出现一次 ,返回删除后数组的新长度. 不要 ...
- 【python】Leetcode每日一题-存在重复元素3
[python]Leetcode每日一题-存在重复元素3 [题目描述] 给你一个整数数组 nums 和两个整数 k 和 t .请你判断是否存在 两个不同下标 i 和 j,使得 abs(nums[i] ...
- 【python】Leetcode每日一题-扰乱字符串
[python]Leetcode每日一题-扰乱字符串 [题目描述] 使用下面描述的算法可以扰乱字符串 s 得到字符串 t : 如果字符串的长度为 1 ,算法停止 如果字符串的长度 > 1 ,执行 ...
- 【python】Leetcode每日一题-前缀树(Trie)
[python]Leetcode每日一题-前缀树(Trie) [题目描述] Trie(发音类似 "try")或者说 前缀树 是一种树形数据结构,用于高效地存储和检索字符串数据集中的 ...
- 【python】Leetcode每日一题-打家劫舍2
[python]Leetcode每日一题-打家劫舍2 [题目描述] 你是一个专业的小偷,计划偷窃沿街的房屋,每间房内都藏有一定的现金.这个地方所有的房屋都 围成一圈 ,这意味着第一个房屋和最后一个房屋 ...
- 【python】Leetcode每日一题-二叉搜索树节点最小距离
[python]Leetcode每日一题-二叉搜索树节点最小距离 [题目描述] 给你一个二叉搜索树的根节点 root ,返回 树中任意两不同节点值之间的最小差值 . 示例1: 输入:root = [4 ...
随机推荐
- Go ORM框架 - GORM 踩坑指南
今天聊聊目前业界使用比较多的 ORM 框架:GORM.GORM 相关的文档原作者已经写得非常的详细,具体可以看这里,这一篇主要做一些 GORM 使用过程中关键功能的介绍,GORM 约定的一些配置信息说 ...
- Python:垃圾回收
有很多不同的方法来实现垃圾回收,例如跟踪,引用计数,转义分析,时间戳和心跳信号等.不同的语言依赖于不同的垃圾回收实现,例如,有些将其与编译器和运行时系统集成在一起.而其他语言则可能需要事后设置,甚至可 ...
- python基础学习之元组和字典的功能方法
什么是元组?(tuple) emmmmmm,这个没必要深究吧,就是一排'元素',一行 格式: a = (1,2,3,4,5,6,7,8,9)用小括号表示的,极为元组. 其有序,且不可更改,可以对比st ...
- Python-jet后台管理的使用
python-django-jet库的使用 1.安装 pip install django-jet 2.配置 将'jet'应用添加到你的Django项目的设置文件settings.py中的INSTAL ...
- find文本处理(locate)实例学习记录
find文本处理(locate)实例学习记录 (一)按文件名称查找 按照文件名称查找是 find 最常见的用法,需要注意的是,搜索的文件名必须完全匹配,才能找到对应的文件. 1. 查找当前目录下所有 ...
- 实验: survivor放不下的对象进入老年代
实验一: 存活对象包含 小于survivor大小的对象 + 大于survivor的对象 private static final Integer _1MB = 1024 * 1024; /** * - ...
- P1208 [USACO1.3]混合牛奶 Mixing Milk(JAVA语言)
思路 按单价排序然后贪心 题目描述 由于乳制品产业利润很低,所以降低原材料(牛奶)价格就变得十分重要.帮助Marry乳业找到最优的牛奶采购方案. Marry乳业从一些奶农手中采购牛奶,并且每一位奶农为 ...
- python-6-1
1.定义一个时间戳转换成格式化时间的函数import time def timestamp_to_fomat(timestamp= None,format ='%Y-%m-%d %H:%M:%S' ) ...
- x64 下记事本WriteFile() API钩取
<逆向工程核心原理>第30章 记事本WriteFile() API钩取 原文是在x86下,而在x64下函数调用方式为fastcall,前4个参数保存在寄存器中.在原代码基础上进行修改: 1 ...
- java例题_40 字母字符串转数组后排序
1 /*40 [程序 40 字符串排序] 输入一个字符串数组,按照字母表的降序对这些字符串进行排序. 2 题目:字符串排序. 3 */ 4 5 /*分析 6 * 1.从键盘得到一个纯字母的字符串 7 ...