8.3 字符串中等 306 Additive Number 423 Reconstruct Original Digits from English

306 Additive Number

在这里插入图片描述

//累加数:除了前两个数,其余数都等于前两个加起来,至少包括三个数
//难点找到前两个数
//条件1:至少包括三个数–>确定前两个数字的最大长度 len = n/3 × 看下方注意1
//条件2:遇到0默认归属于他前面那个数字-------------》用于排除一些错误搭配isValid中的第一个条件语句
//主体:先分段再遍历-----------》先确定前两个数,再得到和 ,再进行比较
//难点:尝试所有可能的分段组合:分别尝试每种可能的前两个数字的长度组合,检查是否满足累加数的条件。

【默写】
注意1:确定前两个数字的长度,如果是123+1 = 124的情况,按照原思路:n = 7 n/3 = 2 但很明显是片面的。所以按照下方思路来:【积累】

在这里插入图片描述
注意2:题目中num.length 最大35 所以承载n1 n2两个数使用longlong类型。
在这里插入图片描述

class Solution {
public:
    bool isValid(const string& num , int len1 ,int len2){
        //条件2: 0不做开头 for num1 num2
        if((num[0] == '0' && len1 > 1 )|| (num[len1] == '0' && len2 > 1)){
            return false;
        }
        string num1 = num.substr(0, len1);
        string num2 = num.substr(len1,len2);
        string sum;
        //num最大长度35 所以int不得行
        long long n1 = stoll(num1) , n2 = stoll(num2);
        
        for(int start = len1+len2 ; start != num.size() ; start+=sum.size()){
            //斐波那契数列
            n2 = n1+n2;
            n1 = n2-n1;
            sum = to_string(n2);
            if((start + sum.size() > num.size())||(num.substr(start , sum.size())!=sum)){
                return false;
            }
        }
        return true;
    }
    bool isAdditiveNumber(string num) {
        int n = num.size();
        //条件1:确定前两个数的长度
        for(int i = 1 ; i <= n/2 ; i++){
            for(int j = 1 ; j <= (n-i)/2 ; j++){
                if(isValid(num , i , j)){
                    return true;
                }
            }
        }
        return false;
    }
};

c++代码学习

string.substr(start,length);
取的是string中从start位置开始长度为length的子字符串。

423 Reconstruct Original Digits from English

在这里插入图片描述
读题:打乱顺序的字母串—>建立字母表,s is valid—>s刚好可以被全部用完,res需要按顺序输出数字
难点:找到每个数字对应的个数并顺序输出 —>核心还是思考s is valid—>s刚好可以被全部用完这个条件。
【原思路】建立字母表,顺序遍历0-9的字母表示,并保证第一个字母是该数字与其他数字的不同(但肯定会有重复的 one nine 就重复了两个字母),最后依次遍历下方这个num pair<string ,char>数组,找到每个数字的英文字母出现的最低频率min,按顺序append到res。
【规范思路】建立字母表,按下方思路建立<string,char>,很明显,按照下方顺序就不能顺序输出数字了,所以多增加一个Digitcount(10,0)用于记录每个数字的min值。主体:确定字母表内容,先遍历num pair数组确定min值输入到digitcount中,最后再遍历一遍digitcount并将数字顺序append到res字符串中。
太乱的话,看最下面的图片。
在这里插入图片描述

class Solution {
public:
    string originalDigits(string s) {
        vector<pair<string, char>> num = {
            {"zero", '0'}, {"two", '2'}, {"four", '4'}, {"six", '6'}, {"eight", '8'}, 
            {"one", '1'}, {"three", '3'}, {"five", '5'}, {"seven", '7'}, {"nine", '9'}
        };
        vector<int> count(26, 0);
        vector<int> digitCount(10, 0);
        
        // 字母表的创建
        for (char ch : s) {
            count[ch - 'a']++;
        }
        
        // 遍历特殊标记数字的字母,确保数字顺序正确
        for (const auto &p : num) {
            string n = p.first;
            int min = INT_MAX;
            for (char ch : n) {
                min = min < count[ch - 'a'] ? min : count[ch - 'a'];
            }
            // 更新每个数字的出现次数
            digitCount[p.second - '0'] = min;
            // 更新 count 数组
            for (char ch : n) {
                count[ch - 'a'] -= min;
            }
        }
        
        string res = "";
        for (int i = 0; i < 10; ++i) {
            res.append(digitCount[i], '0' + i);
        }
        
        return res;
    }
};

在这里插入图片描述

  • 10
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值