字符串逆序(递归实现)

article/2025/10/24 6:43:29

目录

一、代码实现:

二、代码逐步实现过程:

1、不用递归用循环方式实现:

1、使用库函数:

2、不使用库函数:

(1)使用参数是数组的形式:

(2)使用参数是指针的形式:

2、用递归形式实现:

三、总结思路:



题目要求:

编写一个函数 reverse_string(char * string)(递归实现)

实现:将参数字符串中的字符反向排列,不是逆序打印。

要求:不能使用C函数库中的字符串操作函数

比如:

char arr[] = "abcdef";

逆序之后数组的内容变成:fedcba

一、代码实现:

#include <stdio.h>
int my_strlen(char* s)
{int count = 0;while (*s != '\0'){count++;s++;}return count;
}
void reverse_string(char* str)
{int len = my_strlen(str);char tmp = *str;*str = *(str + len - 1);*(str + len - 1) = '\0';if (my_strlen(str + 1) > 1){reverse_string(str + 1);}*(str + len - 1) = tmp;
}
int main()
{char arr[] = "abcdef";reverse_string(arr);printf("%s\n", arr);return 0;
}

二、代码逐步实现过程:

1、不用递归用循环方式实现:

1、使用库函数:

#include <stdio.h>
#include <string.h>
void reverse_string(char str[])//只需逆序不需返回
//void  reverse_string(char* str)//这两行代码都可
{int left = 0;int right = strlen(str) - 1;while (left < right){//交换char tmp = str[left];str[left] = str[right];str[right] = tmp;left++;right--;}
}
int main()
{char arr[] = "abcdef";reverse_string(arr);//用这个函数调整字符串内容,调整完应该是fedcbaprintf("%s\n", arr);//这里打印的应该是fedcbareturn 0;
}//fedcba

代码讲解:

char arr[  ] = "abcdef";//若对该数组内容逆序实现:fedcba

 对该内容逆序,即把a和f位置交换一下,b和e位置交换一下,c和d位置交换一下;
则找a,然后让它和f交换,找a,a的下标是left,f的下标是right,即把left锁定的元素和right锁定的元素交换一下;
交换完之后,left往后走一下left++找b,right往前走一下right--找e,即又找到一对元素,再让她们交换,
当left>=right的时候就没必要交换了,只有当left<right的时候才让它们交换一下。

2、不使用库函数:

不让用strlen()函数(这是库函数)则:自己模拟实现一个my_strlen()函数

(1)使用参数是数组的形式:

#include <stdio.h>
#include <string.h>
int my_strlen(char* s)
{int count = 0;while (*s != '\0'){count++;s++;}return count;
}
//参数是数组的形式:
void reverse_string(char str[])//只需逆序不需返回
//void  reverse_string(char* str)//这两行代码都可
{int left = 0;int right = my_strlen(str) - 1;while (left < right){char tmp = str[left];str[left] = str[right];str[right] = tmp;left++;right--;}
}
int main()
{char arr[] = "abcdef";reverse_string(arr);//用这个函数调整字符串内容,调整完应该是fedcbaprintf("%s\n", arr);//这里打印的应该是fedcbareturn 0;
}

(2)使用参数是指针的形式:

在(1)中left和right是下标的形式,而在这里(2)中是指针的形式

#include <stdio.h>
int my_strlen(char* s)
{int count = 0;while (*s != '\0'){count++;s++;}return count;
}
//参数是指针的形式:
void reverse_string(char* str)
//传的实参是数组名arr,是a的地址,即str里就是a的地址,所以只要找到一个指针指向a,再找到一个指针指向f就可以了
{//找到一个指针指向a:char* left = str;//left指针,把str的地址赋值过去,即把a的地址给了left,所以此时left作为指针指向了a//再找到一个指针指向f:首元素的地址+f的下标char* right = str + my_strlen(str) - 1;//即此时left指向a,right指向fwhile (left<right){//a和f交换:char tmp = *left;//对left指针解引用找到a,把a放到tmp中*left = *right;*right = tmp;left++;//找下一个元素的地址right--;//找上一个元素的地址//找完之后还要交换,所以是个循环}//因为随着数组下标的增长地址是由低到高变化的,所以left作为地址是小于right的
}
int main()
{char arr[] = "abcdef";reverse_string(arr);//用这个函数调整字符串内容,调整完应该是fedcbaprintf("%s\n", arr);//这里打印的应该是fedcbareturn 0;
}//fedcba

注意:

数组的起始地址  +  某元素的下标  =  该元素的地址

2、用递归形式实现:

用数组的递归形式实现:(先不用指针形式的递归)

因为递归是把大事化小,所以:

#include <stdio.h>
int my_strlen(char* s)
{int count = 0;while (*s != '\0'){count++;s++;}return count;
}
void reverse_string(char* str)
{int len = my_strlen(str);char tmp = *str;//a放到tmp中*str = *(str + len - 1);//f放到a位置//先不交换a和f,把f位置上放\0,这样保证bcde遇到\0才算是字符串*(str + len - 1) = '\0';//此时就可以调用reverse_string()函数去逆序中间的bcde,逆序谁就把谁的起始地址传进去,所以传参b的地址:str+1//reverse_string(str + 1);if (my_strlen(str + 1) > 1)//加限制条件{reverse_string(str + 1);}//这里交换a和f的位置:*(str + len - 1) = tmp;
}
int main()
{char arr[] = "abcdef";reverse_string(arr);//用这个函数调整字符串内容,调整完应该是fedcbaprintf("%s\n", arr);//这里打印的应该是fedcbareturn 0;
}//fedcba

若是逆序:

a   b   c   d   e   f   g   \0

采用递归则是:

a和g的交换   +   中间bcdef的逆序;

中间bcdef的逆序是:

b和f的交换   +   中间cde的逆序;

中间cde的逆序是:

c和e的交换 ;

中间元素只剩两个时需要逆序,一个字符没必要逆序了。

所以逆序的递归限制条件:从b位置向后开始需要保证有至少两个字符的字符串



三、总结思路:

循环方式:

  1. 给两个指针,left放在字符串左侧,right放在最后一个有效字符位置

  2. 交换两个指针位置上的字符

  3. left指针往后走,right指针往前走,只要两个指针没有相遇,继续2,两个指针相遇后,逆置结束

递归方式:

对于字符串“abcdefg”,递归实现的大概原理:

  1. 交换a和g,

  2. 以递归的方式逆置源字符串的剩余部分,剩余部分可以看成一个有效的字符串,再以类似的方式逆置

逆置字符串,循环的方式相对于递归实现简单。


知识穿插:与本blog相关连贯:

整形数组逆序_Bug梨哥的博客-CSDN博客



http://chatgpt.dhexx.cn/article/474wKJeN.shtml

相关文章

【C语言刷题】字符串逆序

目录 一、字符串逆序&#xff08;基础题&#xff09; 1.一个经典的错误&#xff0c;标准的零分 2.采用gets函数来修补漏洞 3.非要使用scanf怎么办&#xff1f; 4.使用指针来实现逆序函数 5.将函数修改为&#xff0c;只要传入两个地址&#xff0c;就能逆序这两个地址之间的…

使用C语言实现字符串逆序操作

这篇文章主要介绍了使用C语言实现字符串逆序操作案例,本文包含使用C语言的两种方法去实现,递归和非递归,以下就是详细内容,需要的朋友可以参考下 编写一个函数 reverse_string(char * string) 实现&#xff1a;将参数字符串中的字符反向排列。 要求&#xff1a;不能使用C函数库…

C语言实现——字符串逆序

目录 前言 如何实现 代码实现 1.设立一个数组存放输入的字符串 2.将输入的字符串整体逆序 2.1 计算字符串长度 2.2 使用函数来实现倒置 2.3 实现函数reverse 3.将其中每个单词再进行逆序 3.1 整个语句如何结束循环 3.2 每个单词的结束位置 3.3 内部实现 3.4 判断…

dtb展开成device_node

dtb展开成device_node 文章目录 dtb展开成device_node设备树是如何传递给内核的&#xff1f;设备树相关结构体举例of操作函数与查找节点有关的 OF 函数1、of_find_node_by_name 函数2、of_find_node_by_type 函数3、of_find_compatible_node 函数4、of_find_matching_node_and_…

设备树之DTS与DTB格式

目录 一、设备树 二、DTS格式 2.1 属性 2.2 节点 2.3 引用其他节点 2.4 小总结 三、DTB格式 3.1 结构 3.2 分析 一、设备树 对于点灯字符设备驱动程序可以有三种写法&#xff0c;首先是传统方法&#xff0c;这种方式直接在程序中写死&#xff0c;其次是利用总线设备驱…

Android boot.img dtb.img 编译过程

最近做RK3588案子,修改dts后,导致boot.img过大,编译出错,整体分析下boot.img过大的原因是因为在打包boot.img过程中,dbt.img过大导致,所以整体分析下boot.img编译过程,尤其是dbt.img的生成过程. boot.img生成过程 在Andorid跟目录下执行, source build/envsetup.sh 然后lunch x…

Linux 设备树(二) dtc dts/dtsi dtb的关系

在学习设备树之前&#xff0c;我们先来了解一下跟设备树相关的三个对象&#xff0c;分别是dtc、dts/dtsi、dtb。 dtc:用来编译设备树的工具 dts:设备树描述文件 dtsi:设备树头文件 dtb:编译后的二进制文件 dtc设备树编译工具 dtc是用来编译设备树的工具&#xff0c;就像gcc可以…

dtb如何转换到platform_device

分2步&#xff0c;第一步是首先转换为device_node&#xff0c;第二步device_node转换为platform_device。 第一步 /*** unflatten_device_tree - create tree of device_nodes from flat blob** unflattens the device-tree passed by the firmware, creating the* tree of st…

U-Boot 之一 零基础编译 U-Boot 过程详解、Image 镜像介绍及使用说明、DTB 文件使用说明

最近&#xff0c;工作重心要从裸机开发转移到嵌入式 Linux 系统开发&#xff0c;在之前的博文 Linux 之八 完整嵌入式 Linux 环境、&#xff08;交叉&#xff09;编译工具链、CPU 体系架构、嵌入式系统构建工具 中详细介绍了嵌入式 Linux 环境&#xff0c;接下来就是实际动手学…

dts、dtb的那些事儿

笔者最近在支持新的案子&#xff0c;过于忙碌&#xff0c;好久没更新了&#xff0c;勿怪。 1、设备树大变革故事 2011年3月17日的ARM Linux邮件列表有封邮件“this whole ARM thing is a fucking pain in the ass”引起了轩然大波&#xff0c;原来是我们的Linux之父Linus Torv…

DBT工具简介

What is DBT 在将数据加载到集中式数据仓库之前&#xff0c;必须对其进行清理、保持一致并根据需要进行组合。换句话说&#xff0c;必须转换数据&#xff0c;这就是我们所谓的 ETL&#xff08;提取、转换、加载&#xff09;和 ELT 中的“T”。 这是挖掘数据价值的关键一步。 而…

4.2uboot对设备树的支持——dtb的修改原理

本节说明在uboot中修改dtb的原理。 在uboot中&#xff0c;有一些命令支持对dtb文件进行修改。 当我们想要修改dtb文件时&#xff0c;可以直接修改dts文件&#xff0c;然后编译dts文件生成新的dtb文件&#xff0c;再将新的dtb文件载入设备。 或者&#xff0c;我们也可以在ubo…

DBT是什么

关于DBT DBT 是一种数据转换工作流&#xff0c;可帮助您完成更多工作&#xff0c;同时产生更高质量的结果。您可以使用 dbt 来模块化和集中分析代码&#xff0c;同时还为数据团队提供软件工程工作流中常见的护栏。在将数据模型安全部署到生产环境之前&#xff0c;通过监控和可见…

2.1设备树的规范(dts和dtb)——DTS格式

本节学习设备树的规范。 使用设备树时&#xff0c;需要编写dts文件&#xff0c;然后使用dtc编译dts文件&#xff0c;生成dtb文件。 所以本节分为两部分&#xff0c;第一部分讲解dts格式&#xff0c;第二部分讲解dtb格式。 首先看一下dts文件的布局。 DTS文件布局&#xff0…

关于EMUELC适配各种机型,DTB如何修改教程

很多爱好者留言&#xff0c;都想问关于EMUELC的dtb适配机型问题&#xff0c;这里我就出一个教程&#xff0c;如何修改dtb&#xff0c;然后去适配自己的机型&#xff0c;然后启动 。这里我只是提供方法&#xff0c;具体的调试是需要原理图进行配置。 首先&#xff0c;不管是aml…

DTBO简介

1、DTBO简介 设备树 (DT) 是用于描述“不可发现”硬件的命名节点和属性构成的一种数据结构。操作系统&#xff08;例如在 Android 中使用的 Linux 内核&#xff09;会使用 DT 来支持 Android 设备使用的各种硬件配置。硬件供应商会提供自己的 DT 源文件&#xff0c;接下来…

【DTB/DTBO 分区介绍】

如果你的 DTB/DTBO 位于专属的分区&#xff08;例如 dtb 和 dtbo 分区&#xff09;中&#xff0c;请使用以下表格结构和头文件格式&#xff1a; 数据结构 dt_table_header 仅适用于 dtb/dtbo 分区&#xff1b;您不能在kernel( image.gz) 末尾处附加此格式。如果您有一个 DTB/D…

2.2设备树的规范(dts和dtb)——DTB格式

本节讲述设备树的dtb格式。 上节讲述了dts格式。回顾上节&#xff0c;在dts文件和dtsi文件中&#xff0c;可以使用C语言的define和include&#xff0c;使用方法和作用也同C语言相同。 编写dts文件后&#xff0c;需要使用dtc工具将dts文件编译成dtb文件。dtc工具可以检查dts文…

「设备树」dtb给内核的两种工作模式

一&#xff0c;传递dtb给内核 对于传统bootloader提供两种工作模式&#xff1a;一是启动加载模式&#xff08;start loading&#xff09;&#xff0c;一是下载模式&#xff08;downloading&#xff09; 工作在启动加载模式时&#xff0c;bootloader会自动执行bootcmd命令&#…

设备树_dtb文件分析

前言&#xff1a;我之前的原计划是没有打算写设备树dtb文件分析&#xff0c;但是情势所迫啊&#xff01;&#xff0c;学习还是要一步一步来的。 在前面的章节提到过.dts文件以文本方式对系统设备树进行描述&#xff0c;经过Device Tree Compiler(dtc)将dts文件转换成二进制文件…