剑指offer—day 10(中级dp—2)

1.剑指 Offer 46. 把数字翻译成字符串【中等】

给定一个数字,我们按照如下规则把它翻译为字符串:0 翻译成 “a” ,1 翻译成 “b”,……,11 翻译成 “l”,……,25 翻译成 “z”。一个数字可能有多个翻译。请编程实现一个函数,用来计算一个数字有多少种不同的翻译方法。

示例 1:
输入: 12258
输出: 5
解释: 12258有5种不同的翻译,分别是"bccfi", “bwfi”,“bczi”,“mcfi"和"mzi”

dp,你怎么变难了😶‍🌫️😶‍🌫️😶‍🌫️😶‍🌫️😶‍🌫️😶‍🌫️😶‍🌫️

//若第n-1,n两个数字小于等于25,则
f(n)=f(n-1)+f(n-2)  		//f(n)为n个数字翻译的数量
//反之
f(n)=f(n-1)

先用递归,简单易懂:

int translateNum(int n) {
    if (n < 10) 
        return 1;
    if (n % 100 < 26 && n % 100 > 9) 
        return translateNum(n / 10) + translateNum(n / 100);     //若两个均可 f(n)=f(n-1)+f(n-2)
    else 
        return translateNum(n / 10);                          //若仅仅一个 f(n)=f(n-1)
}

思想同上:

string src = to_string(num);
    int p = 0, q = 0, r = 1;
    for (int i = 0; i < src.size(); ++i) {
        p = q; 
        q = r; 
        r = 0;
        r += q;
        if (i == 0) {
            continue;
        }
        auto pre = src.substr(i - 1, 2);
        if (pre <= "25" && pre >= "10") {
            r += p;
        }
    }
    return r;
}

\qquad

2.剑指 Offer 48. 最长不含重复字符的子字符串【中等】

请从字符串中找出一个最长的不包含重复字符的子字符串,计算该最长子字符串的长度。

示例 1:
输入: “abcabcbb”
输出: 3
解释: 因为无重复字符的最长子串是 “abc”,所以其长度为 3。

//滑动窗口+hash集合
int lengthOfLongestSubstring(string s) {   //滑动窗口+hash集合
    unordered_set<char> occ;
    int rk=-1,ans=0;         //rk为右指针,ans为最大长度
    for(int i=0;i<s.size();i++){
        if(i!=0)
            occ.erase(s[i-1]);      //每次移动把hash集中左端的元素擦除
        while(rk+1<s.size() && !occ.count(s[rk+1])){  //右指针右移
            occ.insert(s[rk+1]);              //新的元素进入集合
            rk++;
        }
        ans=max(ans, rk-i+1);
    }
    return ans;
}
//快速dp
int lengthOfLongestSubstring(string s) {
     vector<int> dp(128,-1);//存储每个字符最后出现的位置
     int i=0,j=0,res=0;
     for(;j<s.size();j++){    
         if(dp[s[j]]<i)//前面的子串不含新加的字符
             res=max(res,j-i+1);
         else//当前字符在之前的子串中出现过            
             i=dp[s[j]]+1;//更新i,使得i到j没有重复字符
         dp[s[j]]=j;//更改当前字符出现的位置                               
     }
     return res;
 }
滑动窗口 (双指针)

假设我们选择字符串中的第 k个字符作为起始位置,并且得到了不包含重复字符的最长子串的结束位置为 r k r_k rk 。那么当我们选择第 k+1个字符作为起始位置时,首先从k+1 到 r k r_k rk的字符显然是不重复的,并且由于少了原本的第 k个字符,我们可以尝试继续增大,直到右侧出现了重复字符为止。

我们使用两个指针表示字符串中的某个子串(或窗口)的左右边界,其中左指针代表着上文中「枚举子串的起始位置」,而右指针即为上文中的 r k r_k rk.
在每一步的操作中,我们会将左指针向右移动一格,表示我们开始枚举下一个字符作为起始位置,然后我们可以不断地向右移动右指针,但需要保证这两个指针对应的子串中没有重复的字符。在移动结束后,这个子串就对应着
以左指针开始的,不包含重复字符的最长子串。我们记录下这个子串的长度; 在枚举结束后,我们找到的最长的子串的长度即为答案。

判断重复:

unordered_set<char>  occ; //hash集合
occ.count(s[rk+1]);      //判断是否在集合中
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值