【C语言练习】字符串旋转你会嘛?

这篇具有很好参考价值的文章主要介绍了【C语言练习】字符串旋转你会嘛?。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

【C语言练习】字符串旋转你会嘛?
【C语言练习】字符串旋转你会嘛?

🍬题目描述:

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

  • ABCD左旋一个字符得到BCDA
  • ABCD左旋两个字符得到CDAB

🍭思路一:

 要左旋 k 个字符,我们首先应该考虑左旋 1 1 1 个字符怎么做。左旋一个字符分为以下的三步:

  • 取出字符串中最左边的一个字符
  • 将字符串中剩下的字符按从左到右的顺序依次左移一位
  • 把第一步取出来的字符放到第二步结束得到的字符串的末尾

【C语言练习】字符串旋转你会嘛?

 知道左旋 1 1 1 个字符的做法,那左旋 k 个字符就只需要把上面的步骤重复执行 k 次就可以。分析的差不多了,接下来上代码:

void left_move(char arr[], int k)//k为左旋次数
{
	int len = strlen(arr);//计算字符串的长度
	int i = 0;
	for (i = 0; i < k; i++)//循环执行k次
	{
		//1:取出字符串中最左边的一个字符
		char tmp = arr[0];
		//2:将字符串中剩下的字符按从左到右的顺序依次左移一位
		int j = 0;
		for (j = 0; j < len - 1; j++)//除了第一个字符,还剩len-1个字符
		{
			arr[j] = arr[j + 1];
		}
		//3:把第一步取出来的字符放到第二步结束得到的字符串的末尾
		arr[len - 1] = tmp;
	}
}

int main()
{
	char arr[] = "ABCDE" ;
	left_move(arr, 2);
	printf("%s\n", arr);
	return 0;
}

🍡代码优化:

 上面的代码已经可以满足题目的要求,但是还可以进行优化。上面的字符串一共只有 5 5 5 个字符串,如果我们要左旋 7 7 7 个字符,那其实就相当于左旋 2 2 2 个字符。如果就按照上面的代码,那左旋 7 7 7 个字符可是要比左旋 2 2 2 个字符多浪贵一些时间,但它们两个最终得到的结果却是一样的,有没有什么办法能够把左旋 7 7 7 个字符变成左旋 2 2 2 个字符呢?答案是有的。这里问题的关键出在循环次数 k 的身上,当左旋 7 7 7 个字符时,只要让 k=2 就可以。因此我们可以在进入 for 循环之前加入下面这句代码: k%=len;。这样一来就可以大大优化代码的运行时间。分析的差不了,接下来上代码:

void left_move(char arr[], int k)//k为左旋次数
{
	int len = strlen(arr);//计算字符串的长度
	k %= len;//重中之重
	int i = 0;
	for (i = 0; i < k; i++)
	{
		//1:取出字符串中最左边的一个字符
		char tmp = arr[0];
		//2:将字符串中剩下的字符按从左到右的顺序依次左移一位
		int j = 0;
		for (j = 0; j < len - 1; j++)//出了第一个字符,还剩len-1个字符
		{
			arr[j] = arr[j + 1];
		}
		//3:把第一步取出来的字符放到第二步结束得到的字符串的末尾
		arr[len - 1] = tmp;
	}
}

int main()
{
	char arr[] = "ABCDE" ;
	left_move(arr, 7);
	printf("%s\n", arr);
	return 0;
}

 虽然优化后的代码,在某些情况下运行时间会有所下降,但整体来看,这种方法还是存在大量的字符移动,效率还是比较低的,接下来介绍一种更加高效的方法。

🍭思路二:

 思路二只需要对字符串进行三次逆序就可以。

  • 先逆序左边的 k 个字符
  • 再逆序剩下的 len-k 个字符
  • 最后逆序整个字符串

【C语言练习】字符串旋转你会嘛?
 以上动图就是左旋两个字符的过程。分析的差不多了,接下来上代码:

void reverse(char* left, char* right)//逆序函数,只要知道待逆序字符串中第一个字符的地址和最后一个地址就可以
{
	assert(left && right);//判断指针的有效性
	while (left < right)
	{
		char tmp = *left;
		*left = *right;
		*right = tmp;
		left++;
		right--;
	}
}

void left_move(char arr[], int k)
{
	int len = strlen(arr);
	k %= len;
	//1:先逆序左边的 `k` 个字符
	reverse(arr, arr + k - 1);
	//2:再逆序剩下的 `len-k` 个字符
	reverse(arr + k, arr + len - 1);
	//3:最后逆序整个字符串
	reverse(arr, arr + len - 1);
}

int main()
{
	char arr[] = "ABCDE";
	left_move(arr, 7);
	printf("%s\n", arr);
	return 0;
}

 本题到这里就结束了,在本题的基础上还衍生出来了下面这道题,还请各位看官继续往下看。

🍬题目描述:

 写一个函数,判断一个字符串是否为另外一个字符串旋转之后的字符串。例如:给定s1 =AABCD和s2=BCDAA,返回1;给定s1=abcd和s2=ACBD,返回0。AABCD左旋一个字符得到ABCDA,AABCD左旋两个字符得到BCDAA,AABCD右旋一个字符得到DAABC。


🍭思路一:

 我们首先判断两个字符串的长度是否相等,如果长度不相等,那必不可能通过旋转一个字符串而得到另一个字符串。如果对于两个字符串 s1s2 ,它们的长度相等,那只需要取其中的一个字符串每次旋转一个字符,然后与另一个字符串比较,看是否相等,此时就可以用到上面的字符串左旋函数了。分析的差不多了,接下来上代码:

void reverse(char* left, char* right)//逆序函数,只要知道待逆序字符串中第一个字符的地址和最后一个地址就可以
{
	assert(left && right);//判断指针的有效性
	while (left < right)
	{
		char tmp = *left;
		*left = *right;
		*right = tmp;
		left++;
		right--;
	}
}

void left_move(char arr[], int k)
{
	int len = strlen(arr);
	k %= len;
	//1:先逆序左边的 `k` 个字符
	reverse(arr, arr + k - 1);
	//2:再逆序剩下的 `len-k` 个字符
	reverse(arr + k, arr + len - 1);
	//3:最后逆序整个字符串
	reverse(arr, arr + len - 1);
}

int is_left_move(char* s1, char* s2)
{
	//先判断两个字符串的长度是否相等
	int len1 = strlen(s1);
	int len2 = strlen(s2);
	if (len1 != len2)//如果长度不相等就直接返回0
	{
		return 0;
	}
	int i = 0;
	for (i = 0; i < len1; i++)//对s1旋转,然后与s2比较
	{
		left_move(s1, 1);//每次旋转一个字符
		if (strcmp(s1, s2) == 0)
		{
			return 1;
		}
	}
	return 0;//代码执行到这里说明s1旋转了一轮和s2都不相等
}

int main()
{
	char s1[] = "AABCD";
	char s2[] = "BCDAA";
	int ret = is_left_move(s1, s2);
	if (ret == 1)
	{
		printf("YES\n");
	}
	else
	{
		printf("NO\n");
	}
	return 0;
}

 上面的代码可以实现题目的要求,但是这种方法比较依赖字符串旋转函数,接下来介绍一种更加巧妙地方法。

🍭思路二:

 在字符串 s1 的后面追加上 s1 ,得到一个新的字符串 s3 ,看 s2 是不是 s3 的一个字串,如果是,那说明 s1 可以通过旋转得到 s2 ,如果不是,那说明无法通过旋转 s1 得到 s2

【C语言练习】字符串旋转你会嘛?
 接下来上代码:

int is_left_move(char* s1, char* s2)
{
	int len1 = strlen(s1);
	int len2 = strlen(s2);
	if (len1 != len2)//如果两个字符串的长度不相等就直接返回0
	{
		return 0;
	}
	char* s3 = strncat(s1, s1, len1);//strncat是字符串追加函数
	if (strstr(s3, s2) != NULL)//strstr是子串查找函数
	{
		return 1;//如果s2是s3的一个子串,说明可以通过旋转s1得到s2,就返回1
	}
	return 0;//如果代码执行到这里,说明s2不是s3的一个子串,说明不能通过旋转s1得到s2,就返回0
}

int main()
{
	char s1[20] = "AABCDF";
	char s2[] = "BCDFAA";
	int ret = is_left_move(s1, s2);
	if (ret == 1)
	{
		printf("YES\n");
	}
	else
	{
		printf("NO\n");
	}
	return 0;
}

🎶结语:
 今天首先介绍了两种旋转字符串的方法,思路一是比较常规的方法,思路二就比较巧妙,通过三次逆序就实现了字符串旋转,大大提高了效率。紧接着,在字符串旋转的基础上又介绍了:一个字符串能否通过另一个字符串旋转得到的问题,利用字符串自身追加结合查找子串巧妙地解决了这一问题。
 今天的分享到这里就结束啦,以上的内容如果对你有帮助的话,可以动动小手点赞、评论、收藏,你的支持就是我前进路上最大的动力!


【C语言练习】字符串旋转你会嘛?文章来源地址https://www.toymoban.com/news/detail-499427.html

到了这里,关于【C语言练习】字符串旋转你会嘛?的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 找凶手,定名次,字符串旋转,杨氏矩阵

    1.找凶手问题: //题目名称: //猜凶手 //题目内容: //日本某地发生了一件谋杀案,警察通过排查确定凶手必为4个嫌疑犯的一个。 //以下为4个嫌疑犯的供词: //A说:不是我 //B说:是C //C说:是D //D说:C在胡说 //已知3个人说的是真话,1个人说的是假话。 //请根据这些信息,写

    2023年04月23日
    浏览(33)
  • 判断一个字符串是否为另一个字符串旋转之后的字符串 (arr1是arr2右旋得到)

    问题: 1.判断函数传参时忘记给arr2加[] 2.把if放在for之外,导致判断不出,程序报错

    2024年02月16日
    浏览(40)
  • Python练习分割字符串

     

    2024年02月09日
    浏览(44)
  • Java中的String字符串练习

    目录 Java中的String字符串练习 01-用户登录 02-遍历字符串并统计字符个数 03-字符串拼接 04-字符串反转 注意点 05-金额转化(简单) 代码解释: 06-手机号屏蔽 07-身份证号码查看 易错点: 08-敏感词替换 注意点 toCharArray() 是Java中的一个方法,它用于将字符串转换为字符数组。 方法签

    2024年03月28日
    浏览(66)
  • 算法练习-右旋字符串(思路+流程图+代码)

            难度:简单         分类:字符串         难度与分类由我所参与的培训课程提供,但需要注意的是,难度与分类仅供参考。以下内容均为个人笔记,旨在督促自己认真学习。         字符串的【右旋转】操作是把字符串尾部的若干个字符转移到字符串的前

    2024年01月22日
    浏览(41)
  • 练习题 替换子串得到平衡字符串

    题目 有一个只含有  \\\'Q\\\', \\\'W\\\', \\\'E\\\', \\\'R\\\'  四种字符,且长度为  n  的字符串。 假如在该字符串中,这四个字符都恰好出现  n/4  次,那么它就是一个「平衡字符串」。 给你一个这样的字符串  s ,请通过「替换一个 子串 」的方式,使原字符串  s  变成一个「平衡字符串」。

    2024年01月18日
    浏览(54)
  • 【C++ OJ练习】5.字符串最后一个单词的长度

    字符串最后一个单词的长度_牛客题霸_牛客网 倒着找第一个空格的下标 用下标之间的差来计算  【C++ OJ练习】5.字符串最后一个单词的长度 完

    2024年02月13日
    浏览(63)
  • 第5章:5.4.5 字符串数组的综合练习(MATLAB入门课程)

    ​讲解视频:可以在bilibili搜索《MATLAB教程新手入门篇——数学建模清风主讲》。​ MATLAB教程新手入门篇(数学建模清风主讲,适合零基础同学观看)_哔哩哔哩_bilibili 案例 1 : 下表左侧给出了四名同学在三次测试中的成绩数据,请对所有同学的测试成绩进行排名,并生成一

    2024年01月24日
    浏览(49)
  • 字符串变换最小字符串(C语言)

    给定一个字符串s,最多只能进行一次变换,返回变换后能得到的最小字符串(按照字典序进行比较)。 变换规则:交换字符串中任意两个不同位置的字符。 一串小写字母组成的字符串s 按照要求进行变换得到的最小字符串。 输入 abcdef 输出 abcdef 说明 abcdef已经是最小字符串

    2024年02月22日
    浏览(37)
  • c语言字符串转数字,数字转字符串

    目录 一、字符串转数字 1. 标准库转换 atoi(),字符串转换成整型数 atol(),字符串转换成长整型数 atof()字符串转换为浮点值 atoll(),字符串转换成长长整型数 strtod(),字符串转换为浮点数  strtol(),将字符串换成长整型数 二、数字转字符串 1. 通过C语言标准库函数 itoa(),整数转换成字符

    2024年02月02日
    浏览(39)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包