C语言题目的多种解法分享 2之字符串左旋和补充题

news2024/10/6 12:24:58

前言

有的时候,这个系列专栏中的解法之间并无优劣,只是给大家提供不同的解题思路

我决定将代码实现的过程写成注释,方便大家直接找到对应的函数,只有需要补充说明的知识才会单拿出来强调

这个系列的文章会更的比较慢,因为多种解法的需要慢慢收集、整理

字符串左旋

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

例如:

ABCD左旋一个字符得到BCDA

ABCD左旋两个字符得到CDAB

暴力求解法1:数组传参

#include<stdio.h>
#include<string.h>
 void left_move(char arr[], int k)
{
	int i = 0;
	int len = strlen(arr);
	for (i = 0; i < k; i++)
	{
		char tmp = arr[0];
		int j = 0;
		for (j = 0; j < len - 1; j++)
		{
			arr[j] = arr[j + 1];
		}
		arr[len - 1] = tmp;
	}
}

int main()
{
	int k = 0;
	scanf("%d", &k);
	char arr[] = "abcdef";
	printf("%s\n", arr);
	left_move(arr,k);
	printf("%s\n", arr);
	return 0;
}

在此简单的解释一下:先将前面的元素提取出来存在别的地方,然后后面的元素向前移动一位,再将前面的元素放在后面
通过循环即可实现

暴力求解法2:指针传参


#include<stdio.h>
#include<string.h>
#include<assert.h>

//函数主体(应用场景)

void left_move(char *arr, int k)
{
	assert(arr);
	//为了良好的代码风格(前面在《实用调试技巧》中有提到过),我们可以在函数体开头使用断言,防止arr是空指针
	int i = 0;
	for (i = 0; i < k; i++)//旋转一个字符
	{
		int len = strlen(arr);//求数组长度

		char* tmp = arr;//存储第一个元素
		int j = 0;
		for (j = 0; j < len - 1; j++)//把后面的元素都向前移动一位
		{
			*(arr + j) = *(arr + j + 1);
			//因为此处会访问到j+1,所以如果j<len的话,会出现数组越界的情况,所以在for循环的判断那里要注意一下
		}
		*(arr + len - 1) = tmp;
		//将第一个元素存储到最后
	}

}

int main()
{
	char arr[] = "abcdef";
	int k = 0;
	scanf("%d", &k);
	left_move(arr, k);//左旋函数
	printf("%s\n", arr);//输出结果

	return 0;
}

提示:使用assert

使用assert可以改善我们的代码风格,具体原因在我的另一篇文章中有详细说明,感兴趣的朋友可以去看一看

三步翻转法

思路:

先逆序前k个元素,再逆序后面的元素,最后再逆序整体
也就是同一个逆序函数要调用三次,我们为了整洁,就单独封装出一个reverse函数

需求:
想要实现逆序,就需要知道两端的地址

reverse函数

void reverse(char* left, char* right)
{
	assert(left);//建议存在指针传参的时候,就使用断言,反正也不吃亏
	assert(right);
	while (left<right)
	{
		char tmp = *left;
		*left = *right;
		*right = tmp;
		left++;
		right--;
	}

}

left_move函数

void left_move(char* arr, int k)
{
	assert(arr);
	int len = strlen(arr);
	assert(len);//判断k是否超出数组大小
	reverse(arr, arr + k - 1);//逆序左边k个元素
	reverse(arr + k, arr + len - 1);//逆序右边的元素
	reverse(arr, arr + len - 1);//逆序整体

}

代码

#include<stdio.h>
#include<string.h>
#include<assert.h>

void reverse(char* left, char* right)
{
	assert(left);
	assert(right);
	while (left<right)
	{
		char tmp = *left;
		*left = *right;
		*right = tmp;
		left++;
		right--;
	}

}


void left_move(char* arr, int k)
{
	assert(arr);
	int len = strlen(arr);
	assert(len);
	reverse(arr, arr + k - 1);//逆序左边k个元素
	reverse(arr + k, arr + len - 1);//逆序右边的元素
	reverse(arr, arr + len - 1);//逆序整体

}

int main()
{
	char arr[] = "abcdef";
	int k = 0;
	scanf("%d", &k);
	left_move(arr, k);//左旋函数
	printf("%s\n", arr);//输出结果

	return 0;
}

拓展

做一道类似的题,帮助大家巩固

写一个函数,判断一个字符串是否为另外一个字符串旋转之后的字符串。

例如:给定s1 =AABCD和s2 = BCDAA,返回1

给定s1=abcd和s2=ACBD,返回0.

AABCD左旋一个字符得到ABCDA

AABCD左旋两个字符得到BCDAA

AABCD右旋一个字符得到DAABC

解法一:穷举,列出所有可能

void reverse(char* left, char* right)
{
	assert(left);
	assert(right);
	while (left<right)
	{
		char tmp = *left;
		*left = *right;
		*right = tmp;
		left++;
		right--;
	}

}

int left_move(char* arr, int k)
{
	assert(arr);
	int len = strlen(arr);
	assert(len);
	reverse(arr, arr + k - 1);//逆序左边k个元素
	reverse(arr + k, arr + len - 1);//逆序右边的元素
	reverse(arr, arr + len - 1);//逆序整体

}

int is_left_move(char* s1, char* s2)
{
	int len = strlen(s1);//求字符串长度,来判断有多少种可能性
	int i = 0;
	for (i = 0; i < len; i++)
	{
		left_move(s1, 1);//左旋的每种可能
		int ret =strcmp(s1, s2);//比较二者是否相等
		if (ret == 0)
		{
			return 1;
		}
	}

		return 0;
}


int main()
{
	char arr1[] = "abcdef";//此处不能使用指针,因为使用指针变量的话,它就是常量,是不能改变的
	char arr2[] = "cdefab";
	int ret = is_left_move(arr1, arr2);//接收返回值来判断是否是
	if (ret == 1)
	{
		printf("yes\n");//是
	}
	else
	{
		printf("no\n");//不是
	}

	return 0;
}

解法二:双倍数组

创建两个数组,如果要对比的字符串,是这个大的数组的子集,那就符合要求,不然就不符合要求

is_left_move函数的实现


int is_left_move(char* s1, char* s2)
{

	int len2 = strlen(s2);
	//分为两步:
	//1.在str1字符串中再放一个str1字符串,要确保str1足够大,并且str1一定要确定数组大小,不然在使用srtncat函数的时候会报错
	// strnact函数:字符串操作函数
	int len1 = strlen(s1);

	if (len1 != len2)
	{
		return 0;
	}
	strncat(s1, s2, len1);

	//2.判断str2指向的字符串是否是str1指向的字符串的子串(类似子集)
	//strstr:找子串的函数
	char* ret = strstr(s1, s2);
	if (ret == NULL)
	{
		return 0;
	}
	else
	{
		return 1;
	}

	return 0;
}

单独说明:

此处单独说明三个比较重要的函数:

stract与strnact

strcat:字符串操作函数,需要包含头文件<string.h>
语法格式:

stract(s1, s2);//把后者追加到前者的后面

但是,要注意自己追加自己是不能使用这个函数的,程序会直接报错

简单的解释一下:
我们创建两个数组,分别存储abc\0和def\0

想要追加的话,首先,要找到第一个字符串中的\0,然后用’d‘取代\0,之后存入ef\0,遇到\0说明追加结束,

而如果想要自己追加自己,我们还是要先找到\0,然后再放入abc,但当我们想放入\0时,却发现原来字符串中的\0已经被我替换成’a‘了,这样追加永远无法结束,程序崩溃~

在这里插入图片描述

那么我们可以使用strnact函数

语法格式
strnact(s1, s2, len)

此处简单的说明一下吧,strnact比stract多了一个参数
下图是二者的函数
在这里插入图片描述
在这里插入图片描述
strnact函数多了一个count参数
前者是直接追加,遇到\0,就停止追加
后者则是追加count位字符之后停止追加

strstr

字符串操作函数,找后者是不是前者的子串
语法格式:
strstr(s1, s2)

返回值的说明:
如果是子串,就返回s2的首元素地址

如果不是子串,就返回空指针NULL

小疑问

代码此时看起来已经可以运行了,但是小明这时候提出了一个疑问:
如果s2是cdef,输出结果是什么

输出结果是yes

这就是问题所在,所以我们在使用那些库函数之前,先要进行一个判断:s1和s2的长度是否相等,如果相等再去判断,如果不等,直接返回0结束判断。

最终代码:

#define _CRT_SECURE_NO_WARNINGS 1

#include<stdio.h>
#include<string.h>
#include<assert.h>


int is_left_move(char* s1, char* s2)
{
	int len1 = strlen(s1);
	int len2 = strlen(s2);

	if (len1 != len2)
	{
		return 0;
	}
	strncat(s1, s2, len1);

	char* ret = strstr(s1, s2);
	if (ret == NULL)
	{
		return 0;
	}
	else
	{
		return 1;
	}

}

int main()
{
	char arr1[50] = "abcdef";//此处不能使用指针,因为使用指针变量的话,它就是常量,是不能改变的
	char arr2[] = "cdefab";
	int ret = is_left_move(arr1, arr2);//接收返回值来判断是否是
	if (ret == 1)
	{
		printf("yes\n");//是
	}
	else
	{
		printf("no\n");//不是
	}

	return 0;
}

结语

本来是想整理五道题再发的,但现在刚写两道就四千多字了,那我想还是发出去吧,毕竟字符串左旋也比较经典,就单独列出一篇来。

希望对各位有帮助,我们下次见

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

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

相关文章

级联(数据字典)

二级级联&#xff1a; 一&#xff1a;新建两个Bean 父级&#xff1a; /*** Description 数据字典* Author WangKun* Date 2023/7/25 10:15* Version*/ Data AllArgsConstructor NoArgsConstructor TableName("HW_DICT_KEY") public class DictKey implements Seri…

学习笔记整理-JS-06-函数

一、函数基本使用 1. 什么是函数 函数就是语句的封装&#xff0c;可以让这些代码方便地被复用。函数具有"一次定义&#xff0c;多次调用"的优点。使用函数&#xff0c;可以简化代码&#xff0c;让代码更具有可读性。 2. 函数的定义和调用 和变量类似&#xff0c;函…

C++:模拟实现list及迭代器类模板优化方法

文章目录 迭代器模拟实现 本篇模拟实现简单的list和一些其他注意的点 迭代器 如下所示是利用拷贝构造将一个链表中的数据挪动到另外一个链表中&#xff0c;构造两个相同的链表 list(const list<T>& lt) {emptyinit();for (auto e : lt){push_back(e);} }void test_…

【ES】【elasticsearch】分布式搜索

文章目录 ☀️安装elasticsearch☀️1.部署单点es&#x1f338;1.1.创建网络&#x1f338;1.2.下载镜像&#x1f338;1.3.运行 ☀️2.部署kibana&#x1f338;2.1.部署&#x1f338;2.2.DevTools ☀️3.安装IK分词器&#x1f338;3.1.在线安装ik插件&#xff08;较慢&#xff0…

ARM汇编快速入门

本文主要分享如何快速上手ARM汇编开发的经验、汇编开发中常见的Bug以及Debug方法、用的Convolution Dephtwise算子的汇编实现相对于C版本的加速效果三方面内容。 前言 神经网络模型能够在移动端实现快速推理离不开高性能算子&#xff0c;直接使用ARM汇编指令来进行算子开发无疑…

ad+硬件每日学习十个知识点(32)23.8.12 (元器件封装、PCB封装、3D的PCB封装)

文章目录 1.元器件封装属性值说明2.PCB封装标准说明&#xff08;M、N、L&#xff09;3.电阻的PCB封装&#xff08;阻焊层&#xff09;4.电感的PCB封装&#xff08;CD、CDRH&#xff09;1.CD31的意思是&#xff0c;直径3mm&#xff0c;高度1mm![在这里插入图片描述](https://img…

【SQL应知应会】索引(二)• MySQL版

欢迎来到爱书不爱输的程序猿的博客, 本博客致力于知识分享&#xff0c;与更多的人进行学习交流 本文收录于SQL应知应会专栏,本专栏主要用于记录对于数据库的一些学习&#xff0c;有基础也有进阶&#xff0c;有MySQL也有Oracle 索引 • MySQL版 前言一、索引1.简介2.创建2.1 索引…

Gradio——快速部署可视化人智能应用

前言 Gradio是一个开源的Python库&#xff0c;用于快速构建机器学习和数据科学演示的应用。它可以帮助你快速创建一个简单漂亮的用户界面&#xff0c;以便向客户、合作者、用户或学生展示你的机器学习模型。此外&#xff0c;还可以通过自动共享链接快速部署模型&#xff0c;并获…

IntelliJ IDEA热部署:JRebel插件的安装与使用

热部署 概述JRebel 概述 热部署&#xff0c;指修改代码后&#xff0c;无需停止应用程序&#xff0c;即可使修改后的代码生效&#xff0c;其有利于提高开发效率。 热部署方式&#xff1a; 手动热部署&#xff1a;修改代码后&#xff0c;重新编译项目&#xff0c;然后启动应用程…

【软件测试】Linux系统下安装jdk配置环境变量(详细步骤)

目录&#xff1a;导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结&#xff08;尾部小惊喜&#xff09; 前言 1、安装环境 操作…

UGUI事件系统EventSystem

一. 事件系统概述 Unity的事件系统具有通过鼠标、键盘、游戏控制柄、触摸操作等输入方式&#xff0c;将事件发送给对象的功能。事件系统通过场景中EventSystem对象的组件EventSystem和Standalone Input Module发挥功能。EventSystem对象通常实在创建画布的同时被创建的&#xf…

c++杂谈-5

目录 一、一些c11新特性1. decltype关键字及函数后置返回类型2. 函数后置返回类型 二、c的内存模型三、函数指针和回调函数四、函数模版的注意事项 一、一些c11新特性 1. decltype关键字及函数后置返回类型 在C11中&#xff0c;decltype操作符&#xff0c;用于查询表达式的数…

HarmonyOS NEXT新能力,一站式高效开发HarmonyOS应用

2023年8月6日华为开发者大会2023&#xff08;HDC.Together&#xff09;圆满收官&#xff0c;伴随着HarmonyOS 4的发布&#xff0c;华为向开发者发布了汇聚所有最新开发能力的HarmonyOS NEXT开发者预览版&#xff0c;并分享了围绕“一次开发&#xff0c;多端部署” “可分可合&a…

通过Microsoft Loopback Adapter实现虚拟机和物理机的通信

问题 问&#xff1a;不借助路由器或交换机的情况下&#xff0c;能不能实现主机和虚拟及之间两个软件的通信呢&#xff1f;要求主机和虚拟及均有独立的ip地址&#xff0c;从而进行指定源的组播通信。 答&#xff1a;可以。通过借助虚拟网络适配器&#xff0c;不需要路由器或交…

深度思考rpc框架面经系列之三

6 一个rpc框架的请求调用的流程&#xff08;小红书面试&#xff09; 6.1 讲讲rpc调用原理&#xff0c;比如服务怎么发现&#xff0c;怎么调用&#xff0c;提供者怎么响应。怎么去请求&#xff0c;又怎么回来的 一个RPC&#xff08;远程过程调用&#xff09;框架的核心目的是允…

百度飞浆实战-手写数字识别

目录 参考建模过程1、数据加载和预处理2、模型的网络设计和开发模型组网 3、模型训练 代码实战1、打开aistudio找到项目 参考 视频教程 PaddleAPI DOC 建模过程 1、数据加载和预处理 飞桨框架帮助我们将MNIST数据集进行了内置 数据集名称&#xff1a; MNIST 数据集官网 &am…

2023-08-14 LeetCode每日一题(合并二叉树)

2023-08-14每日一题 一、题目编号 617. 合并二叉树二、题目链接 点击跳转到题目位置 三、题目描述 给你两棵二叉树&#xff1a; root1 和 root2 。 想象一下&#xff0c;当你将其中一棵覆盖到另一棵之上时&#xff0c;两棵树上的一些节点将会重叠&#xff08;而另一些不会…

Android布局【RelativeLayout】

文章目录 介绍常见属性根据父容器定位根据兄弟组件定位 通用属性margin 设置组件与父容器的边距padding 设置组件内部元素的边距 项目结构主要代码 介绍 RelativeLayout是一个相对布局&#xff0c;如果不指定对齐位置&#xff0c;都是默认相对于父容器的左上角的开始布局 常见…

3D沉浸式旅游网站开发案例复盘【Three.js】

Plongez dans Lyon网站终于上线了。 我们与 Danka 团队和 Nico Icecream 共同努力&#xff0c;打造了一个令我们特别自豪的流畅的沉浸式网站。 这个网站是专为 ONLYON Tourism 和会议而建&#xff0c;旨在展示里昂最具标志性的活动场所。观看简短的介绍视频后&#xff0c;用户…

[足式机器人]Part5 机械设计 Ch00/01 绪论+机器结构组成与连接 ——【课程笔记】

本文仅供学习使用 本文参考&#xff1a; 《机械设计》 王德伦 马雅丽课件与日常作业可登录网址 http://edu.bell-lab.com/manage/#/login&#xff0c;选择观摩登录&#xff0c;查看2023机械设计2。 机械设计-Ch00Ch01——绪论机器结构组成与连接 Ch00-绪论0.1 何为机械设计——…