【C语言】字符串左旋的三种解题方法详细分析

news2024/11/27 12:08:59

在这里插入图片描述

博客主页: [小ᶻ☡꙳ᵃⁱᵍᶜ꙳]
本文专栏: C语言

文章目录

  • 💯前言
  • 💯题目描述
  • 💯方法一:逐字符移动法
  • 💯方法二:使用辅助空间法
  • 💯方法三:三次反转法
  • 💯方法对比与总结
    • 拓展思考
  • 💯总结

在这里插入图片描述


在这里插入图片描述


💯前言

  • 在这篇文章中,我们将深入探讨字符串左旋的三种解决方案,系统分析每种方法的算法设计时间复杂度空间复杂度,以及其适用场景局限性。这些方法从简单到高效,分别展示了逐字符移动法使用辅助空间法三次反转法的不同特点及其理论基础
    通过对这些方法的详细对比讨论,读者将对字符串操作基本原理有更为深入的理解,掌握在不同应用场景下如何选择最优的解决方案,以提升代码的效率鲁棒性
    C语言
    在这里插入图片描述

💯题目描述

实现一个函数,能够左旋字符串中的 k 个字符。

例如:

  • 给定字符串 ABCD,如果左旋 1 个字符,得到 BCDA
  • 如果左旋 2 个字符,得到 CDAB

左旋是指将字符串的前面部分字符移到字符串的末尾。我们将使用三种不同的实现方式来解决这一问题:逐字符移动法、使用辅助空间法、以及三次反转法。


💯方法一:逐字符移动法

思路

逐字符移动法是一种直观的实现方式,通过多次迭代将字符串的第一个字符移动到末尾,直到完成所需的旋转次数。该方法依赖于不断移动字符的位置以达到最终目标。其本质是一种线性搬移操作,容易理解,但在大数据量的情况下效率较低。
在这里插入图片描述

代码实现

void leftRound(char* str, int k) {
    int len = strlen(str);
    int time = k % len;

    for (int i = 0; i < time; i++) {
        char tmp = str[0];
        int j = 0;
        for (; j < len - 1; j++) {
            str[j] = str[j + 1];
        }
        str[j] = tmp;
    }
}

int main() {
    char str[] = "abcdef";
    leftRound(str, 7);
    printf("%s\n", str);
    return 0;
}

解释

  • 获取字符串长度 len
  • 计算实际旋转次数 time = k % len,这样可以避免 k 超过字符串长度导致的冗余操作。
  • 外层循环用于执行 time 次旋转,每次只移动一个字符。
  • 使用内部的 for 循环,将每个字符依次前移一个位置,最后将原来的第一个字符移到末尾。

时间复杂度和空间复杂度

  • 时间复杂度:O(k * len),对于每次旋转都需要遍历整个字符串,因而效率较低。
  • 空间复杂度:O(1),只需一个额外的字符变量来暂存被移动的字符。

优缺点

  • 这种方法的优点是实现简单且易于理解,适合初学者用于理解字符串旋转的基本概念。但其缺点在于效率较低,尤其在 k 较大和字符串长度较长时,由于需要多次逐字符移动,时间开销会显著增加。

💯方法二:使用辅助空间法

思路

使用辅助空间法通过将旋转后的字符串临时保存,再将结果复制回原字符串。借助辅助空间将两个部分拼接,可以有效避免逐字符移动带来的低效问题。该方法的核心在于利用辅助数组,快速完成字符串的局部重组与合并
在这里插入图片描述

代码实现

void leftRound(char* str, int k) {
    int len = strlen(str);
    int time = k % len;
    char tmp[256] = { 0 };

    strcpy(tmp, str + time); 
    strncat(tmp, str, time); 
    strcpy(str, tmp);
}

int main() {
    char str[] = "abcdef";
    leftRound(str, 7);
    printf("%s\n", str);
    return 0;
}

解释

  • 通过 strlen 获取字符串的长度。
  • 计算实际的旋转次数 time = k % len,以应对 k 过大时的情况。
  • 使用临时数组 tmp 存储旋转后的结果。
  • 利用 strcpy 函数将原字符串从 time 索引位置开始的部分复制到 tmp 中。
    • strcpy(tmp, str + time)str + time 表示指向原字符串从第 time 个位置开始的指针,strcpy 函数将从这个位置开始的所有字符(直到字符串结束符 )复制到 tmp 中。因此,tmp 中最初会存储字符串的后半部分。例如,若 str = "ABCD"time = 2strcpy(tmp, str + time)"CD" 复制到 tmp,结果为 tmp = "CD"
  • 使用 strncat 函数将原字符串的前 time 个字符拼接到 tmp 的末尾。
    • strncat(tmp, str, time)strncat 用于将源字符串的指定数量字符拼接到目标字符串的末尾。在这里,str 表示原字符串,time 表示从 str 开头开始的 time 个字符。因此,strncat(tmp, str, time) 将原字符串的前 time 个字符(即 "AB")拼接到 tmp 的末尾,形成最终结果。例如,当 tmp = "CD"time = 2 时,拼接后结果为 tmp = "CDAB"
  • 最后使用 strcpytmp 的内容复制回原字符串 str,完成旋转操作。

时间复杂度和空间复杂度

  • 时间复杂度:O(n),只需遍历字符串几次即可完成旋转。
  • 空间复杂度:O(n),需要额外的存储空间来保存旋转后的结果。

优缺点

  • 辅助空间法的优势在于其高效的操作,尤其适合处理较大的字符串,但需要额外的空间来存储结果。因此,对于内存资源受限的情况,该方法可能不太适合。

💯方法三:三次反转法

思路

三次反转法通过对字符串的部分片段进行反转,从而达到整体左旋的效果。核心思想是将字符串划分为两部分,分别反转,再对整体反转,以实现最终的左旋

这种方法以极少的操作步骤完成字符位置的重新排列,具有较高的效率数学美感

在这里插入图片描述

代码实现

void ReverseRange(char* str, int start, int end) {
    int left = start;
    int right = end;
    while (left < right) {
        char tmp = str[left];
        str[left] = str[right];
        str[right] = tmp;
        left++;
        right--;
    }
}

void leftRound(char* str, int k) {
    int len = strlen(str);
    int time = k % len;
    ReverseRange(str, 0, time - 1);    // 反转前 time 个字符
    ReverseRange(str, time, len - 1);  // 反转剩余部分
    ReverseRange(str, 0, len - 1);     // 反转整个字符串
}

int main() {
    char str[] = "abcdef";
    leftRound(str, 7);
    printf("%s\n", str);
    return 0;
}

解释

  1. 反转前 time 个字符

    • 调用 ReverseRange(str, 0, time - 1),反转字符串的前 time 个字符。
    • 例如,对 "ABCD" 反转前两个字符,得到 "BACD"
  2. 反转剩余部分

    • 调用 ReverseRange(str, time, len - 1),反转剩余部分。
    • "BACD" 中的后两个字符反转,得到 "BADC"
  3. 反转整个字符串

    • 调用 ReverseRange(str, 0, len - 1),反转整个字符串。
    • "BADC" 反转得到 "CDAB",从而完成左旋操作。

时间复杂度和空间复杂度

  • 时间复杂度:O(n),三次反转操作均为线性时间复杂度,因此总时间复杂度为 O(n)
  • 空间复杂度:O(1),只需进行交换操作,没有额外的空间开销。

优缺点

  • 三次反转法的优势在于其高效性,时间复杂度为 O(n),空间复杂度为 O(1),适合处理大规模字符串。该方法通过反转实现字符串的重新排列,以最小的时间和空间成本实现左旋,非常适合实际开发中的高效实现。

💯方法对比与总结

方法时间复杂度空间复杂度优点缺点
逐字符移动法O(k * n)O(1)实现简单,直观效率较低
辅助空间法O(n)O(n)实现高效,代码简洁需要额外空间
三次反转法O(n)O(1)高效且无额外空间开销实现稍复杂,需要三次反转
  • 逐字符移动法 适合初学者理解字符串旋转的概念,但效率不高。在字符较多或者旋转次数较大时,可能会因为需要逐次移动而显得低效
  • 辅助空间法 则通过一次性拼接字符串提升了效率,但需要额外的空间存储,对于空间资源紧张的场景可能不适合
  • 三次反转法最优的方法,尤其适合大规模字符串。它的空间复杂度最低效率高,是一个非常经典的字符串操作技巧。该方法不仅巧妙而且具有一定的数学美感,通过三次反转将字符串重新排列到目标位置。

拓展思考

  • 这些字符串旋转的实现方法可以拓展到其他类型的数据结构,例如数组的左旋、右旋操作。对于数组来说,三次反转法同样有效,可以通过类似的思路对数组元素进行重新排列。
  • 三次反转法的思想不仅可以用于字符串旋转,还可以用于链表数组等其他序列的数据结构。它是一种通用的思想,可以帮助解决许多涉及顺序调整的问题。
  • 对于多种编程语言,我们也可以找到相似的实现方法。Python 中可以使用字符串切片操作来轻松实现左旋,而 Java 中可以借助 StringBuilder 进行高效的字符串操作。Python 的切片方法非常直观,例如可以使用 s = s[k:] + s[:k] 来实现左旋操作,简洁而高效。
  • 对于面试场景,理解这些方法的时间复杂度空间复杂度,以及不同方法在实际应用中的适用性非常重要。面试官通常希望看到你对基础方法的掌握以及对最优解法的深入理解。

💯总结

  • 在这里插入图片描述
    字符串左旋是一个非常基础但又非常经典的字符串操作问题。通过这篇文章的深入解读,我们详细探讨了三种不同的解决方案,并对每种方法的算法复杂度适用场景优缺点进行了深入分析。掌握这些方法有助于我们在面试中应对字符串操作类的问题,也能帮助我们在实际开发中写出更优雅、高效的代码。这三种方法展示了从基础到高效的不同解决思路,是学习和掌握字符串操作的宝贵经验
    希望这篇文章能够激发你对字符串操作的更深入思考与理解。未来的学习和工作中,愿你能够灵活应用这些技巧,优化代码的效率与可读性,深入理解不同方法背后的原理,并能够在解决问题时选择最合适的工具。这正是编程的艺术,也是不断提升的动力所在

在这里插入图片描述


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

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

相关文章

【346】Postgres内核 Startup Process 通过 signal 与 postmaster 交互实现 (5)

1. Startup Process 进程 postmaster 初始化过程中, 在进入 ServerLoop() 函数之前,会先通过调用 StartChildProcess() 函数来开启辅助进程,这些进程的目的主要用来完成数据库的 XLOG 相关处理。 如: 核实 pg_wal 和 pg_wal/archive_status 文件是否存在Postgres先前是否发…

大数据面试SQL题-笔记02【查询、连接、聚合函数】

大数据面试SQL题复习思路一网打尽&#xff01;(文档见评论区)_哔哩哔哩_bilibiliHive SQL 大厂必考常用窗口函数及相关面试题 大数据面试SQL题-笔记01【运算符、条件查询、语法顺序、表连接】大数据面试SQL题-笔记02【查询、连接、聚合函数】​​​​​​​ 目录 01、查询 01…

【Python中while循环】

一、深拷贝、浅拷贝 1、需求 1&#xff09;拷贝原列表产生一个新列表 2&#xff09;想让两个列表完全独立开&#xff08;针对改操作&#xff0c;读的操作不改变&#xff09; 要满足上述的条件&#xff0c;只能使用深拷贝 2、如何拷贝列表 1&#xff09;直接赋值 # 定义一个…

51单片机从入门到精通:理论与实践指南入门篇(二)

续51单片机从入门到精通&#xff1a;理论与实践指南&#xff08;一&#xff09;https://blog.csdn.net/speaking_me/article/details/144067372 第一篇总体给大家在&#xff08;全局&#xff09;总体上讲解了一下51单片机&#xff0c;那么接下来几天结束详细讲解&#xff0c;从…

【pyspark学习从入门到精通20】机器学习库_3

目录 使用 ML 预测婴儿生存几率 加载数据 创建转换器 创建估计器 创建管道 拟合模型 使用 ML 预测婴儿生存几率 在这一部分&#xff0c;我们将使用前一章中的数据集的一部分来介绍 PySpark ML 的概念。 在这一部分&#xff0c;我们将再次尝试预测婴儿的生存几率。 加载…

【计算机网络】核心部分复习

目录 交换机 v.s. 路由器OSI七层更实用的TCP/IP四层TCPUDP 交换机 v.s. 路由器 交换机-MAC地址 链接设备和设备 路由器- IP地址 链接局域网和局域网 OSI七层 物理层&#xff1a;传输设备。原始电信号比特流。数据链路层&#xff1a;代表是交换机。物理地址寻址&#xff0c;交…

LLamafactory 批量推理与异步 API 调用效率对比实测

背景 在阅读 LLamafactory 的文档时候&#xff0c;发现它支持批量推理: 推理.https://llamafactory.readthedocs.io/zh-cn/latest/getting_started/inference.html 。 于是便想测试一下&#xff0c;它的批量推理速度有多快。本文实现了 下述两种的大模型推理&#xff0c;并对…

【自动化Selenium】Python 网页自动化测试脚本(上)

目录 1、Selenium介绍 2、Selenium环境安装 3、创建浏览器、设置、打开 4、打开网页、关闭网页、浏览器 5、浏览器最大化、最小化 6、浏览器的打开位置、尺寸 7、浏览器截图、网页刷新 8、元素定位 9、元素交互操作 10、元素定位 &#xff08;1&#xff09;ID定位 &…

Table 滚动条始终停靠在可视区域的底部

1. 话题引入 存在这样一个场景&#xff1a;当页面尺寸发生变化时&#xff0c;希望滚动条能够随之动态调整&#xff0c;始终展示在 table 的可视区域的最下方&#xff0c;而不是整个 table 本身的最底部。 这种行为可以提升用户的使用体验&#xff0c;尤其是在处理大数据表格时…

【漏洞复现】CVE-2020-13925

漏洞信息 NVD - CVE-2020-13925 Similar to CVE-2020-1956, Kylin has one more restful API which concatenates the API inputs into OS commands and then executes them on the server; while the reported API misses necessary input validation, which causes the hac…

基于Springboot的心灵治愈交流平台系统的设计与实现

基于Springboot的心灵治愈交流平台系统 介绍 基于Springboot的心灵治愈交流平台系统&#xff0c;后端框架使用Springboot和mybatis&#xff0c;前端框架使用Vuehrml&#xff0c;数据库使用mysql&#xff0c;使用B/S架构实现前台用户系统和后台管理员系统&#xff0c;和不同级别…

快速理解微服务中Gateway的概念

一.基本概念 定义&#xff1a; 在微服务架构中&#xff0c;Spring Cloud Gateway 是一个用于API网关的框架&#xff0c;它是一个基于 Spring Framework 的高效、可扩展的路由器和反向代理&#xff0c;它能够将外部请求转发到适当的微服务&#xff0c;并提供一些与请求处理相关…

Java【多线程】(1)进程与线程

目录 1.前言 2.正文 2.1什么是进程 2.2PCB&#xff08;进程控制块&#xff09; 2.2.1进程id 2.2.2内存指针 2.2.3文件描述符表 2.2.4进程状态 2.2.4.1就绪状态 2.2.4.2阻塞状态 2.2.5进程优先级 2.2.6进程上下文 2.2.7进程的记账信息 2.3CPU操作进程的方法 2.4什…

计算机毕业设计Python+大模型美食推荐系统 美食可视化 美食数据分析大屏 美食爬虫 美团爬虫 机器学习 大数据毕业设计 Django Vue.js

温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 作者简介&#xff1a;Java领…

华为鸿蒙内核成为HarmonyOS NEXT流畅安全新基座

HDC2024华为重磅发布全自研操作系统内核—鸿蒙内核&#xff0c;鸿蒙内核替换Linux内核成为HarmonyOS NEXT稳定流畅新基座。鸿蒙内核具备更弹性、更流畅、更安全三大特征&#xff0c;性能超越Linux内核10.7%。 鸿蒙内核更弹性&#xff1a;元OS架构&#xff0c;性能安全双收益 万…

kafka生产者和消费者命令的使用

kafka-console-producer.sh 生产数据 # 发送信息 指定topic即可 kafka-console-producer.sh \ --bootstrap-server bigdata01:9092 \ --topic topicA # 主题# 进程 29124 ConsoleProducer kafka-console-consumer.sh 消费数据 # 消费数据 kafka-console-consumer.sh \ --boo…

构造函数的相关

文章目录 一、构造函数 今天我们要来讲解类的默认成员函数之一的构造函数。 一、构造函数 构造函数是特殊的成员函数&#xff0c;需要注意的是&#xff0c;构造函数虽然名称叫构造&#xff0c;但是构造函数的主要任务并不是开空间创建对象(我们常使用的局部对象是栈帧创建时&…

云服务器部署WebSocket项目

WebSocket是一种在单个TCP连接上进行全双工通信的协议&#xff0c;其设计的目的是在Web浏览器和Web服务器之间进行实时通信&#xff08;实时Web&#xff09; WebSocket协议的优点包括&#xff1a; 1. 更高效的网络利用率&#xff1a;与HTTP相比&#xff0c;WebSocket的握手只…

计算机网络八股整理(一)

计算机网络八股文整理 一&#xff1a;网络模型 1&#xff1a;网络osi模型和tcp/ip模型分别介绍一下 osi模型是国际标准的网络模型&#xff0c;它由七层组成&#xff0c;从上到下分别是&#xff1a;应用层&#xff0c;表示层&#xff0c;会话层&#xff0c;传输层&#xff0c;…

【Qt】控件7

1.QTextEdit的简单使用 使用简单的QTextEdit,获取到的内容显示到标签上 使用textChanged信号 在槽函数中需要获取QTextEdit的内容&#xff0c;对应操作是&#xff1a; QString curorui->textEdit->toPlainText();然后显示到标签上&#xff0c;对应操作是&#xff1a; …