3337|3335. 字符串转换后的长度 I(||)

news2025/5/15 13:49:20

1.字符串转换后的长度 I

1.1题目

3335. 字符串转换后的长度 I - 力扣(LeetCode)

1.2解析

递推法解析

思路框架

我们可以通过定义状态变量来追踪每次转换后各字符的数量变化。具体地,定义状态函数 f(i,c) 表示经过 i 次转换后,字符 c(对应 ASCII 码偏移量,如 a=0,b=1,…,z=25)在字符串中的出现次数。通过分析字符转换规则,我们可以建立递推关系,从而高效计算最终结果。

状态转移分析
  1. 初始状态

    • 对于给定字符串 s,初始化 f(0,c) 为字符 c 在 s 中的出现次数。
  2. 状态转移方程

    • 字符 a(对应 c=0):只能由前一次的字符 z 转换而来,因此:
      f(i,0)=f(i−1,25)
    • 字符 b(对应 c=1):可由前一次的字符 z 或 a 转换而来,因此:
      f(i,1)=f(i−1,25)+f(i−1,0)
    • 其他字符 c≥2:由前一次的字符 c−1 转换而来,因此:
      f(i,c)=f(i−1,c−1)
  3. 优化空间复杂度

    • 由于每次递推仅依赖前一层的状态,我们可以使用两个一维数组(如 cnt 和 nxt)交替存储状态,将空间复杂度从 O(t×26) 优化至 O(26)。
算法实现步骤
  1. 初始化:统计字符串 s 中每个字符的出现次数,存入数组 cnt
  2. 迭代递推
    • 执行 t 次状态转移,每次生成新数组 nxt 存储当前层状态。
    • 根据上述转移方程更新 nxt 数组。
    • 将 nxt 赋值给 cnt,为下一次迭代做准备。
  3. 结果计算:累加最终状态数组 cnt 中所有元素的和,即为转换 t 次后的字符串长度。
复杂度分析
  • 时间复杂度:O(n+t),其中 n 是字符串 s 的长度,t 是转换次数。初始化需遍历一次字符串,每次转换需处理 26 个字符。
  • 空间复杂度:O(1),仅需固定大小的数组存储状态,与输入规模无关。

1.3代码

class Solution {
public:
    int lengthAfterTransformations(string s, int t) {
        vector<int> cnt(26);
        for(auto ch:s)
        {
            ++cnt[ch-'a'];
        }

        for(int i=0;i<t;i++)
        {
            vector<int> next(26);
            next[0]=cnt[25];
            next[1]=(cnt[25]+cnt[0])%(1000000007);
            for(int i=2;i<26;i++)
            {
                next[i]=cnt[i-1];
            }

            cnt=move(next);
        }
        int ans=0;
        for(int i=0;i<26;i++)
        {
            ans=(ans+cnt[i])%(1000000007);
        }

        return ans;
    }
};

2.3337字符串转换后的长度 ||

2.1原题

3337. 字符串转换后的长度 II - 力扣(LeetCode)

2.2快速幂模板

首先我们先学一下快速幂

快速幂是一种高效计算幂运算的方法 。

原理

  • 二进制角度:将幂次 n 转化为二进制形式。例如幂次 13,其十进制数 13 转换为二进制是 1101 ,而 11012​=2^3+2^2+2^0=8+4+1 。这就把幂次的计算拆解成了与二进制位相关的形式。
  • 幂的性质利用:依据幂的性质 a2k=(ak)2 来进行计算。从 a 开始,通过不断平方得到 a,a2,a4,a8,⋯ 这些值。比如计算 a13 ,因为 13 二进制表示中第 0 位、第 2 位、第 3 位是 1 ,对应着 a1 、 a4 、 a8 ,所以 a13=a8×a4×a1 。这样就避免了像普通方法那样进行多次连乘,大幅减少了计算量。

计算过程

  1. 把幂次 n 转换为二进制数。
  2. 初始化结果变量为 1 ,底数为 a 。
  3. 从二进制数的最高位开始遍历:
    • 如果当前位是 1 ,就把当前的结果乘以对应的幂次(比如从最高位开始,第一位对应 最高位序号 )。
    • 每次遍历完一位,都将底数进行平方操作,为下一位的计算做准备。
  4. 遍历完二进制数的所有位后,得到的结果就是 an 。

优势

传统计算 an ,需要进行 n−1 次乘法,时间复杂度是 O(n) 。而快速幂通过上述方式,时间复杂度降为 O(logn) 。当 n 很大时,计算效率会有显著提升。

快速幂模板代码

#define ll long long
#define mod ......
ll qsm(ll a,ll b)
{
    ll res=0;
    while(b)
    {
        if(b&1)res=(res*a)%mod;
        a=(a*a)%mod;
        b>>=1;
    }

    return res;
}

2.3矩阵快速幂

而本题则要用到矩阵快速幂

方法一:动态规划(基础解法)

思路分析

我们可以通过动态规划的方式,逐步计算每个字母在经过多次替换后的长度。具体来说:

  1. 子问题定义:对于每个字母(如 'a', 'b', ..., 'z'),将其替换 t 次后的长度可以分解为替换 t-1 次后的子问题。例如,字母 'a' 替换一次后变为 'b' 和 'c',因此其替换 t 次后的长度等于 'b' 和 'c' 分别替换 t-1 次后的长度之和。

  2. 状态定义:定义 f[i][j] 表示字母 j(0 对应 'a',1 对应 'b',依此类推)替换 i 次后的长度。

  3. 状态转移方程:对于每个字母 j,设 c = nums[j],则有: 

  4. 即字母 j 替换 i 次后的长度等于其替换一次后生成的所有字母(共 c 个)在替换 i-1 次后的长度之和。

  5. 初始条件:当 i=0 时,即未进行任何替换,每个字母的长度为 1,因此: f[0][j] = 1

  6. 最终结果:遍历字符串 s,统计每个字母的出现次数 cnt[j],最终结果为所有字母替换 t 次后的长度之和:

这种方法适用于替换次数 t 较小的情况,但对于较大的 t 会导致超时,需要进一步优化。

方法二:矩阵快速幂优化

思路分析

当替换次数 t 非常大时,动态规划的时间复杂度会变得很高。此时,可以利用矩阵快速幂来优化计算过程。

  1. 矩阵表示:将状态转移方程转化为矩阵乘法形式。对于每个字母 j,其替换后的长度可以表示为矩阵乘法的结果。例如,在示例中,状态转移可以表示为:

  2. F[i] = M * F[i-1],其中 M 是转移矩阵。

  3. 递推关系:通过递推可得:
    F[t] = M^t * F[0]

  4. 结果计算:由于 F[0] 全为 1,因此 f[t][j] 等于矩阵 M^t的第 j 行所有元素之和。最终结果为:

  5. 这种方法通过矩阵快速幂将时间复杂度从指数级优化到对数级,能够高效处理大规模的替换次数。

2.4题解代码

class Solution {
public:
    const int MOD=1000000007;
    using Matrix = array<array<int,26>,26>;

    Matrix mul(Matrix &a,Matrix &b)//矩阵乘法
    {
        Matrix c{};
        for(int i=0;i<26;i++)
        {
            for(int k=0;k<26;k++)
            {
                if(a[i][k]==0)continue;//如果是0可以直接跳过

                for(int j=0;j<26;j++)
                {
                    c[i][j]=(c[i][j]+(long long) a[i][k]*b[k][j])%MOD;
                }
            }
        }

        return c;
    }

    Matrix pow(Matrix a,int n)//矩阵快速幂
    {
        Matrix res={};
        for(int i=0;i<26;i++)
        {
            res[i][i]=1;
        }
        while(n)
        {
            if(n&1)
            {
                res=mul(res,a);
            }
            a=mul(a,a);
            n>>=1;
        }
        return res;
    }

    int lengthAfterTransformations(string s, int t, vector<int>& nums) {
        Matrix m{};
        for(int i=0;i<26;i++)
        {
            for(int j=i+1;j<=i+nums[i];j++)
            {
                m[i][j%26]=1;
            }
        }

        Matrix mt=pow(m,t);
        int cnt[26]{};
        for(char c:s)
        {
            cnt[c-'a']++;
        }
        long long ans=0;
        for(int i=0;i<26;i++)
        {
            ans += reduce(mt[i].begin(), mt[i].end(), 0LL) * cnt[i];
        }

        return ans%MOD;
    }
};

本题解是在学习力扣题解基础上总结而来,感谢大家观看!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2376142.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

PHP黑白胶卷底片图转彩图功能 V2025.05.15

关于底片转彩图 传统照片底片是摄影过程中生成的反色图像&#xff0c;为了欣赏照片&#xff0c;需要通过冲印过程将底片转化为正像。而随着数字技术的发展&#xff0c;我们现在可以使用数字工具不仅将底片转为正像&#xff0c;还可以添加色彩&#xff0c;重现照片原本的色彩效…

字符串检索算法:KMP和Trie树

目录 1.引言 2.KMP算法 3.Trie树 3.1.简介 3.2.Trie树的应用场景 3.3.复杂度分析 3.4.Trie 树的优缺点 3.5.示例 1.引言 字符串匹配&#xff0c;给定一个主串 S 和一个模式串 P&#xff0c;判断 P 是否是 S 的子串&#xff0c;即找到 P 在 S 中第一次出现的位置。暴力匹…

基于.Net开发的网络管理与监控工具

从零学习构建一个完整的系统 平常项目上线后&#xff0c;不仅意味着开发的完成&#xff0c;更意味着项目正式进入日常运维阶段。在这个阶段&#xff0c;网络的监控与管理也是至关重要的&#xff0c;这时候就需要一款网络管理工具&#xff0c;可以协助运维人员用于日常管理&…

Python并发编程:开启性能优化的大门(7/10)

1.引言 在当今数字化时代&#xff0c;Python 已成为编程领域中一颗璀璨的明星&#xff0c;占据着编程语言排行榜的榜首。无论是数据科学、人工智能&#xff0c;还是 Web 开发、自动化脚本编写&#xff0c;Python 都以其简洁的语法、丰富的库和强大的功能&#xff0c;赢得了广大…

易学探索助手-个人记录(十)

在现代 Web 应用中&#xff0c;用户体验的重要性不断上升。近期我完成了两个功能模块 —— 语音播报功能 与 用户信息修改表单&#xff0c;分别增强了界面交互与用户自管理能力。 一、语音播报功能&#xff08;SpeechSynthesis&#xff09; 功能特点 支持播放、暂停、继续、停…

学习51单片机01(安装开发环境)

新学期新相貌.......哈哈哈&#xff0c;我终于把贪吃蛇结束了&#xff0c;现在我们来学stc51单片机&#xff01; 要求&#xff1a;c语言的程度至少要到函数&#xff0c;指针尽量&#xff01;如果c语言不好的&#xff0c;可以回去看看我的c语言笔记。 1.开发环境的安装&#x…

SpringAI

机器学习&#xff1a; 定义&#xff1a;人工智能的子领域&#xff0c;通过数据驱动的方法让计算机学习规律&#xff0c;进行预测或决策。 核心方法&#xff1a; 监督学习&#xff08;如线性回归、SVM&#xff09;。 无监督学习&#xff08;如聚类、降维&#xff09;。 强化学…

lua 作为嵌入式设备的配置语言

从lua的脚本中获取数据 lua中栈的索引 3 | -1 2 | -2 1 | -3 可以在lua的解释器中加入自己自定的一些功能,其实没啥必要,就是为了可以练习下lua

ERP系统源码,小型工厂ERP系统源码,CRM+OA+进销存+财务

ERP系统源码&#xff0c;小型工厂ERP系统源码&#xff0c;ERP计划管理系统源码&#xff0c;CRMOA进销存财务 对于ERP来说&#xff0c;最为主要的作用就是能够强调企业的计划性&#xff0c;通过以业务订单以及客户的相关需求来作为企业计划的基础&#xff0c;并且还能够对企业现…

基于EFISH-SCB-RK3576/SAIL-RK3576的矿用本安型手持终端技术方案‌

&#xff08;国产化替代J1900的矿山智能化解决方案&#xff09; 一、硬件架构设计‌ ‌本安型结构设计‌ ‌防爆防护体系‌&#xff1a; 采用铸镁合金外壳复合防爆玻璃&#xff08;抗冲击能量>20J&#xff09;&#xff0c;通过GB 3836.1-2021 Ex ib I Mb认证 全密闭IP68接口…

配置文件介绍xml、json

#灵感# 常用xml&#xff0c; 但有点模棱两可&#xff0c;记录下AI助理给我总结的。 .xml XML&#xff08;eXtensible Markup Language&#xff0c;可扩展标记语言&#xff09;是一种用于存储和传输数据的标记语言。它与 HTML 类似&#xff0c;但有以下主要特点和用途&#xf…

【PostgreSQL数据分析实战:从数据清洗到可视化全流程】附录-D. 扩展插件列表(PostGIS/PostgREST等)

&#x1f449; 点击关注不迷路 &#x1f449; 点击关注不迷路 &#x1f449; 点击关注不迷路 文章大纲 附录D. PostgreSQL扩展插件速查表一、插件分类速查表二、核心插件详解三、安装与配置指南四、应用场景模板五、版本兼容性说明六、维护与优化建议七、官方资源与工具八、附录…

Qt笔记---》.pro中配置

文章目录 1、概要1.1、修改qt项目的中间文件输出路径和部署路径1.2、Qt 项目模块配置1.3、外部库文件引用配置 1、概要 1.1、修改qt项目的中间文件输出路径和部署路径 &#xff08;1&#xff09;、为解决 “ 输出文件 ” 和 “ 中间输出文件 ”全部在同一个文件夹下的问题&am…

【Liblib】基于LiblibAI自定义模型,总结一下Python开发步骤

一、前言 Liblib AI&#xff08;哩布哩布 AI&#xff09;是一个集成了先进人工智能技术和用户友好设计的 AI 图像创作绘画平台和模型分享社区。 强大的图像生成能力 &#xff1a;以 Stable Diffusion 技术为核心&#xff0c;提供文生图、图生图、图像后期处理等功能&#xff…

CCF第七届AIOps国际挑战赛季军分享(RAG)

分享CCF 第七届AIOps国际挑战赛的季军方案&#xff0c;从我们的比赛经历来看&#xff0c;并不会&#xff0c;相反&#xff0c;私域领域问答的优秀效果说明RAG真的很重要 历经4个月的时间&#xff0c;从初赛赛道第1&#xff0c;复赛赛道第2&#xff0c;到最后决赛获得季军&…

【Cesium入门教程】第七课:Primitive图元

Cesium丰富的空间数据可视化API分为两部分&#xff1a;primitive API面向三维图形开发者&#xff0c;更底层一些。 Entity API是数据驱动更高级一些。 // entity // 调用方便&#xff0c;封装完美 // 是基于primitive的封装// primitive // 更接近底层 // 可以绘制高级图形 /…

【5分钟学Docker】Docker快速使用

目录 1. 概述 2. 基本操作 2.1. 镜像操作 2.2. 容器操作 2.3. 运行操作 2.4. 镜像保存 2.5. 镜像分享 3. 高级操作 4. 挂载 4.1. 目录挂载 4.2. 卷映射 1. 概述 Docker 镜像有镜像名称和TAG 2. 基本操作 2.1. 镜像操作 查看镜像 docker images docker image ls …

opencv 一些简单的设置

输出当前程序启动的路径 可能会出现 &#x1f527; 设置 C17 标准&#xff08;解决 std::filesystem 报错&#xff09; 在 VS 中&#xff0c;右键项目 → 属性。 选择左边的 “C/C” → “语言” 找到 C语言标准&#xff08;C Language Standard&#xff09;选项。 设置为&…

监控易运维管理软件:架构稳健,组件强大

在当今的信息化时代&#xff0c;运维管理对于企业的稳定运营至关重要。一款好的运维管理软件&#xff0c;不仅能够帮助企业高效管理IT基础设施&#xff0c;还能提升运维效率&#xff0c;降低运维成本。今天&#xff0c;我要给大家介绍的&#xff0c;就是我们公司自主研发的监控…

数字IC后端零基础入门基础理论(Day2)

数字IC后端零基础入门基础理论&#xff08;Day1&#xff09; Placement Blockage: cell摆放阻挡层。它是用来引导工具做placement的一种物理约束或手段&#xff0c;目的是希望工具按照我们的要求来做标准单元的摆放。 它主要有三种类型&#xff0c;分别是hard placement bloc…