实现一个函数可以左旋字符串中的k个字符包学会!(两种办法)

简介: 实现一个函数可以左旋字符串中的k个字符包学会!(两种办法)
题目描述

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

例如:

ABCD左旋一个字符得到BCDA

ABCD左旋两个字符得到CDAB

题目分析

我们将思路先捋清楚,做任何题目之前不要盲目直接地去敲代码,可以先在自己的草稿纸上画图理解,在之后的数据结构学习中更是要养成这个学习习惯,推荐大家可以去看一本数——《剑指offer》,这个习惯就在其中有提到。

方法一

方法一,我们可以将前k个字符先逆序,然后再将后面的字符逆序,再将整体逆序,就可以得出左旋k个字符后的字符串

例如,我们将字符串ABCDE左旋2个字符:

思路如下:

方法一代码实现

首先我们下一个交换函数

因为这里是逆序字符串,所以这里的a,b的类型是char*

void swap(char* a, char* b)
{
  char temp = *a;
  *a = *b;
  *b = temp;
}

然后我们写逆序函数

当left<right的时候才逆序,等于的时候不用逆序,记住,swap函数里面的参数我们是传址调用,所以要用取地址符号&取出其字符的地址

然后left是往右移动,即进行“++”操作

right则相反,进行“- -”操作

void reverse(char* src, int left,int right)
{
  while (left < right)
  {
    swap(&src[left], &src[right]);
    left++;
    right--;
  }
}

完整代码:

注意:

当k大于字符串的长度len是,就取它的余数,左旋也是一样的效果

还有需要注意的是左旋开始的下标问题:

逆序前半段应该下标应该是从0开始,k-1结尾
后半段则是k开始,len-1结尾
整体逆序就是0开始,len-1结尾

#include<stdio.h>
#include<string.h>
void swap(char* a, char* b)
{
  char temp = *a;
  *a = *b;
  *b = temp;
}
void reverse(char* src, int left,int right)
{
  while (left < right)
  {
    swap(&src[left], &src[right]);
    left++;
    right--;
  }
}
int main()
{
  int k = 0;
  scanf("%d", &k);
  char arr[] = "ABCDEF";
  int len = strlen(arr);
  if (k > len)
  {
    k %= len;
  }
  reverse(arr, 0, k - 1);
  reverse(arr, k, len - 1);
  reverse(arr, 0, len - 1);
  printf("%s", arr);
  return 0;
}

这样,我们就完成了字符串的左旋了

方法二

我们需要左旋k个字符,那我们是不是就可以创建一个新的空间,先将后面的len-k个字符放进这个新的空间,然后再将前面的k个字符放进去,就可以实现字符串的左旋了呢?

我们用图来了解一下:

我们用开辟一个动态的内存空间temp用来存放从arr拷贝出来的字符串

然后再将temp中的内容拷贝到arr里,就实现了字符串的左旋了

方法一代码实现

首先开辟temp

字符串有多长我们就开辟多大的空间,避免出现内存的浪费

因此我们最好选择动态内存开辟(关于动态内存详解大家可以看我之前的博客)

char* temp = (char*)malloc(sizeof(char) * len);

字符串的拷贝:

我们使用memcpy函数将其放入新的空间temp中,然后再用memcpy将temp中的字符串统一放入arr中

关于memcpy函数不懂的也可以看我之前的博客

memcpy(temp, arr + len - k, sizeof(char) *(len-k));
//后面len-k个字符是从下标为len-k-1开始的,此时temp是空的,所以起始地址就是temp
memcpy(temp + len-k, arr, sizeof(char) * k);
//将arr的前k个字符下标是0开始,就是arr,但是放入temp的地址是temp的下标为len-k的地方开始的
memcpy(arr, temp, sizeof(char) * len);

放入arr后记得释放内存,并且将其置为空指针,memcpy的头文件为stdlib.h

完整代码如下:

void reverse(char* arr, int len, int k)
{
  char* temp = (char*)malloc(sizeof(char) * len);
  memcpy(temp, arr + len - k - 1, sizeof(char) * (len - k));
  memcpy(temp + len - k, arr, sizeof(char) * k);
  memcpy(arr, temp, sizeof(char) * len);
  free(temp);
  temp = NULL;
}
int main()
{
  char arr[] = "ABCDE";
  int len = strlen(arr);
  int k = 0;
  if (k > len)
    k %= len;
  scanf("%d", &k);
  reverse(arr, len, k);
  printf("%s\n", arr);
  return 0;
}

方法二的左旋字符串就完成了

这里我给大家留下一个题目:

大家可以思考一下,下期为大家解答

判断一个字符串是否为另一个字符串左旋后的字符
是的话就返回1
不是返回0

今天的分享就到这了,谢谢大家的支持!下次见!

目录
打赏
0
1
1
0
23
分享
相关文章
智能化运维在现代数据中心的应用与挑战####
本文深入探讨了智能化运维(AIOps)技术在现代数据中心管理中的实际应用,分析了其带来的效率提升、成本节约及潜在风险。通过具体案例,阐述了智能监控、自动化故障排查、容量规划等关键功能如何助力企业实现高效稳定的IT环境。同时,文章也指出了实施过程中面临的数据隐私、技术整合及人才短缺等挑战,并提出了相应的解决策略。 --- ####
150 1
【TypeScript技术专栏】TypeScript在Angular开发中的应用
【4月更文挑战第30天】本文探讨了TypeScript在Angular开发中的应用。Angular与TypeScript的结合利用了静态类型检查和ECMAScript特性,简化了大型Web应用的开发。文章涵盖组件、数据绑定、依赖注入、服务、守卫和路由以及模块化等方面,展示了如何在Angular中有效使用TypeScript。此外,还提到了TypeScript的高级应用,如泛型、高级类型和装饰器。掌握这些知识将有助于提升Angular应用的可维护性和可扩展性。
193 0
【C语言】Leetcode 88.合并两个有序数组
【C语言】Leetcode 88.合并两个有序数组
96 3
深入解析 EventLoop 和浏览器渲染、帧动画、空闲回调的关系
关于 Event Loop 的文章很多,但是有很多只是在讲「宏任务」、「微任务」
【SVN】在linux下查看svn的账号密码
【SVN】在linux下查看svn的账号密码
409 1
【SVN】在linux下查看svn的账号密码
Core Animation - 第一次使用图层来创建一个简单的项目
Core Animation - 第一次使用图层来创建一个简单的项目
108 0
线程方法:sleep( )、wait()、join( )、yield( )的区别
线程方法:sleep( )、wait()、join( )、yield( )的区别
450 0
kde
|
11天前
|
Docker镜像加速指南:手把手教你配置国内镜像源
配置国内镜像源可大幅提升 Docker 拉取速度,解决访问 Docker Hub 缓慢问题。本文详解 Linux、Docker Desktop 配置方法,并提供测速对比与常见问题解答,附最新可用镜像源列表,助力高效开发部署。
kde
7184 15
|
8天前
typora免费版,激活方法,Typora使用教程
Typora是一款简洁高效的Markdown编辑器,支持即时渲染。本教程涵盖安装方法、文件操作、视图控制、格式排版、字体样式及Markdown语法,助你快速上手使用Typora进行高效写作。
1862 1
AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等

登录插画

登录以查看您的控制台资源

管理云资源
状态一览
快捷访问