Note: 子序列,可以不连续;子字符串,必须连续。

以下题目按在我看看来的难度从易到难排列:

最大和子序列(Maximum sum subsequence)

这道题纯属娱乐...应该不会有人出这种题吧。方案是贪心法,遇到正数就放入序列。

vector<int> msseq(vector<int> &num) {
vector<int> result;
for(int i : num)
if(i > )
result.push_back(i);
return result;
}

最大和子字符串(Maximum sum substring)

暴力方案TC为O(n^2)。更好的方案为,贪心法, 设i从0到n遍历,用gmax代表全局最大和,cursum代表当前子字符串的和。cursum为负,则放弃之前的子字符串;为正则继续向后加,每遇到一个正数,都更新一次gmax。TC = O(n),SC = O(1)。

double msstr(vector<double> &nums) {
double gmax = , cur = ;
for(int i : nums) {
if(cur > )
cur += i;
else
cur = i;
gmax = max(gmax, cur);
}
return gmax;
}

最长递增子字符串(Longest increasing substring)

暴力方案TC为O(n^2)。更好的方案为,与最大和子字符串一样,贪心法,设i从0到n遍历,用gmax代表全局最长长度,len代表当前递增substring的长度。遇到递增,则len++;遇到非递增,则更新gmax,并重置len为1。TC = O(n),SC = O(1)

int listr(vector<double> &nums) {
if(nums.empty())
return ;
int gmax = , cur = ;
for(int i = ; i < nums.size(); i++) {
if(nums[i] > nums[i - ]) {
cur++;
gmax = max(gmax, cur);
}
else
cur = ;
}
return gmax;
}

最大乘积子字符串(Maximum product subsequence)

基本操作与最大乘积子序列差不多,不过是需要连续。用邻点动态规划,

double mpstr(vector<double> &nums) {
double gmax, cmax, cmin, pmax, pmin;
gmax = cmax = cmin = pmax = pmin = ;
for(double i : nums) {
cmax = max(i, max(pmax * i, pmin * i));
cmin = min(i, min(pmax * i, pmin * i));
pmax = cmax;
pmin = cmin;
gmax = max(gmax, cmax);
}
return gmax;
}

最大乘积子序列(Maximum product subsequence)

这题与最大和子序列差不多吧...如果无负数,遇到大于1的数就放入序列;不过需要考虑正负号,用邻点动态规划,

double mpseq(vector<double> &nums) {
double cmax, cmin, pmax, pmin;
cmax = cmin = pmax = pmin = ;
for(double i : nums) {
cmax = max(max(i, pmax), max(pmax * i, pmin * i));
cmin = min(min(i, pmin), min(pmax * i, pmin * i));
pmax = cmax;
pmin = cmin;
}
return pmax;
}

最长递增子序列(Longest increasing subsequence)

暴力方案TC为O(2^n)。如果用贪心法,分析知道根据第i个包含的信息,无法覆盖前i个数的情况,故第i+1个数的决策没法做;更好的方案为,全局动态规划,用opt[i]记录到i为止最长的且包含 i上字符的最长递增subsequence,opt[i]初始为1,动态转移方程为opt[i] = max(opt[j] + 1, opt[i])。

vector<int> miseq(vector<int> &num) {
vector<int> result;
if(num.empty())
return num;
vector<int> opt(num.size(), ), record(num.size(), -);
for(int i = ; i < num.size(); i++) {
for(int j = ; j < i; j++) {
if(num[i] > num[j] && opt[j] + > opt[i]) {
opt[i] = opt[j] + ;
record[i] = j;
}
}
}
int last = -, gmax = ;
for(int i = ; i < num.size(); i++) {
if(opt[i] > gmax) {
gmax = opt[i];
last = i;
}
}
while(last >= ) {
result.push_back(num[last]);
last = record[last];
}
reverse(result.begin(), result.end());
return result;
}

最长匹配子字符串(Longest common substring) 

暴力方案TC为O(m*n*max(m,n))。更好的方案为,邻点动态规划,用opt[i][j]记录到M的i - 1位置与N的j - 1位置为止,且包含i - 1,j - 1的最长匹配子字符串。将0位置设为岗哨,其中opt[0][j] = opt[i][0] = 0,动态转移方程为opt[i][j] = M[i] == N[j] ? opt[i - 1][j - 1] + 1 : 0。TC = O(m*n),SC = O(m * n),用滚动数组可以压缩到O(m) or O(n)。

string lcstr(string M, string N) {
//找到最长匹配子字符串的长度
int maxlen = INT_MIN, ri = -;
vector<vector<int> > opt(M.size() + , vector<int>(N.size() + , ));
for(int i = ; i <= M.size(); i++)
for(int j = ; j <= N.size(); j++) {
opt[i][j] = M[i - ] == N[j - ] ? opt[i - ][j - ] + : ;
if(opt[i][j] > maxlen) {
maxlen = opt[i][j];
ri = i;
}
}
//将最大值对应的匹配子字符串输出
string result = "";
int starti = ri - maxlen;
for(int i = starti; i < ri; i++)
result += M[i];
return result;
}

最长匹配子序列(Longest common subsequence) 

更好的方案为,与最长匹配子字符串类似,邻点动态规划,用opt[i][j]记录到M的i - 1位置与N的j - 1位置为止,且包含i - 1,j - 1的最长匹配子字符串。将0位置设为岗哨,其中opt[0][j] = opt[i][0] = 0,动态转移方程为opt[i][j] = M[i] == N[j] ? opt[i - 1][j - 1] + 1 : max(opt[i - 1][j], opt[i][j - 1])。TC = O(m*n),SC = O(m*n),滚动数组同样可以压缩到O(m) or O(n)。

//只是找长度很简单...
int lenlcstr(string M, string N) {
//找到最长匹配子字符串的长度
vector<vector<int> > opt(M.size() + , vector<int>(N.size() + , ));
for(int i = ; i <= M.size(); i++)
for(int j = ; j <= N.size(); j++)
opt[i][j] = M[i - ] == N[j - ] ? opt[i - ][j - ] + : max(opt[i - ][j], opt[i][j - ]);
return opt[M.size()][N.size()];
}
//还要返回一个最长匹配字符串
string lcstr(string M, string N) {
//记录路径
vector<vector<int> > track(M.size() + , vector<int>(N.size() + , -));
//找到最长匹配子字符串的长度
vector<vector<int> > opt(M.size() + , vector<int>(N.size() + , ));
for(int i = ; i <= M.size(); i++)
for(int j = ; j <= N.size(); j++) {
if(M[i - ] == N[j - ]) {
opt[i][j] = opt[i - ][j - ] + ;
track[i][j] = ;
}
else {
if(opt[i - ][j] > opt[i][j - ]) {
track[i][j] = ;
opt[i][j] = opt[i - ][j];
}
else {
track[i][j] = ;
opt[i][j] = opt[i][j - ];
}
}
}
int i = int(M.size()), j = int(N.size());
string result = "";
while(track[i][j] > -) {
if(track[i][j] == ) {
result += M[i - ];
i--;
j--;
}
else if(track[i][j] == )
i--;
else
j--;
}
reverse(result.begin(), result.end());
return result;
}

编辑距离(Edit distance)

更好的方案,与LCS类似,邻点动态规划

int edist(string s1, string s2) {
int m = int(s1.length()), n = int(s2.length());
vector<vector<int> > opt(m + , vector<int> (n + , ));
for(int i = ; i <= m; i++)
opt[i][] = i;
for(int j = ; j <= n; j++)
opt[][j] = j;
for(int i = ; i <= m; i++)
for(int j = ; j <= n; j++) {
opt[i][j] = min(min(opt[i - ][j], opt[i][j - ]), opt[i - ][j - ]) + ;
if(s1[i - ] == s2[j - ])
opt[i][j] = min(opt[i][j], opt[i - ][j - ]);
}
return opt[m][n];
}

最长无重复子字符串(Longest substring with no duplicate characters)

用一个unordered_set<char>里存已有的char,遇到加入一个新的char,则在set里找,如果没找到,则加入set,并且count++;如果找到了,则在substring的循环弹出,更新set,并循环count--,直到弹出新加入的char的重复字符。

[经典] 最X(长 | 大和 | 大积)Y(子序列 | 子字符串)的更多相关文章

  1. 《剑指offer》第四十八题(最长不含重复字符的子字符串)

    // 面试题48:最长不含重复字符的子字符串 // 题目:请从字符串中找出一个最长的不包含重复字符的子字符串,计算该最长子 // 字符串的长度.假设字符串中只包含从'a'到'z'的字符. #inclu ...

  2. 剑指offer-面试题48-最长不含重复字符的子字符串-动态规划

    /* 题目: 最长不含重复字符的子字符串. */ /* 思路: f(i) = f(i-1) + 1,(未出现过当前字符,distance > f(i-1) distance,当前字符和上一次出现 ...

  3. 剑指 Offer 48. 最长不含重复字符的子字符串 + 动态规划 + 哈希表 + 双指针 + 滑动窗口

    剑指 Offer 48. 最长不含重复字符的子字符串 Offer_48 题目详情 解法分析 解法一:动态规划+哈希表 package com.walegarrett.offer; /** * @Aut ...

  4. 剑指offer面试题48: 最长不含重复字符的子字符串

    Given a string, find the length of the longest substring without repeating characters.(请从子字符串中找出一个最长 ...

  5. 【Java】 剑指offer(48) 最长不含重复字符的子字符串

    本文参考自<剑指offer>一书,代码采用Java语言. 更多:<剑指Offer>Java实现合集   题目 请从字符串中找出一个最长的不包含重复字符的子字符串,计算该最长子字 ...

  6. 【Offer】[48] 【最长不含重复字符的子字符串】

    题目描述 思路分析 测试用例 Java代码 代码链接 题目描述 请从字符串中找出一个最长的不包含重复字符的子字符串,计算该最长子字符串的长度.假设字符串中只包含'a'~'z'的字符.例如,在字符串&q ...

  7. 剑指offer——50最长不含重复字符和子字符串

    题目: 请从字符串中找出一个最长的不包含重复字符的子字符串,计算该最长子字符串的长度.假设字符串中只包含’a~z”的字符.例如,在字符串“arabcacfr"中,最长的不含重复字符的子字符串 ...

  8. 每日一题 - 剑指 Offer 48. 最长不含重复字符的子字符串

    题目信息 时间: 2019-07-02 题目链接:Leetcode tag: 动态规划 哈希表 难易程度:中等 题目描述: 请从字符串中找出一个最长的不包含重复字符的子字符串,计算该最长子字符串的长度 ...

  9. 剑指 Offer 48. 最长不含重复字符的子字符串

    题目描述 请从字符串中找出一个最长的不包含重复字符的子字符串,计算该最长子字符串的长度. 示例1: 输入: "abcabcbb" 输出: 3 解释: 因为无重复字符的最长子串是 & ...

随机推荐

  1. C#使用Process类调用外部程序(转)

    在程序开发中,一个程序经常需要去调用其他的程序,C#中Process类正好提供了这样的功能.它提供对本地和远程进程的访问并使您能够启动和停止本地系统进程.一.启动进程实例 Process myProc ...

  2. Android开发--去掉标题栏

    Android开发中为了尽可能美观,会去掉标题栏.去掉标题栏有三种方法. 一.在Activity代码里实现 在代码中实现以下方法: this.requestWindowFeature(Window.F ...

  3. Java 6 Thread States and Life Cycle.

    Ref: Java 6 Thread States and Life Cycle This is an example of UML protocol state machine diagram sh ...

  4. linux设置中文环境

    确认当前环境 [root@Oracle11g ~]# df –h Filesystem            Size  Used Avail Use% Mounted on /dev/sda1    ...

  5. block 浅析

    最近讲了一个关于block的例子 block 可以作为一个参数 进行传递 需要注意的地方是 :block 虽然作为一个参数 但是在函数方法执行的时候 block 是不会在定义它的地方调用 除非你在后边 ...

  6. DataSet与DataTable基本用法

    http://files.cnblogs.com/files/monkeyZhong/DataSetExample.rar 在设计数据库操作或者XML操作等表格数据时,我们难免要跟DataSet和Da ...

  7. 自构BeanHandler(用BeansUtils)

    class BeanHandler<T> implements ResultSetHandler<T>{ private Class<T> clazz; publi ...

  8. Telerik_2012_Q3 破解全套下载链接

    1.Telerik_OpenAccess_ORM_2012_3_1012_SDK.zip (暂未提供下载) 2. Telerik_OpenAccess_ORM_2012_3_1012.zip 3. T ...

  9. POJ 1321-棋盘问题(DFS 递归)

    POJ 1321-棋盘问题 K - DFS Time Limit:1000MS     Memory Limit:10000KB     64bit IO Format:%I64d & %I6 ...

  10. Zephyr-开发流程

    开发流程 前提1:检查你的Linux主机满足入门指南中规定的最低要求. 具体请参考 :  物联网操作系统-Zephyr 前提2: 确保SDK的环境变量和zephyr项目的环境变量. 终端执行: $ e ...