IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 数据结构与算法 -> 牛课网编程~~倒置字符串 -> 正文阅读

[数据结构与算法]牛课网编程~~倒置字符串

前言

一道牛客网的编程题。。。
题目地址

解法1思路解析

拿到题目之后,我的思考首先是使用原空间进行倒置还是新开辟一个数组去存放倒置的结果。
这里我选择了新开辟一个空间。
因为倒置到新的数组中的时候,I like Beijing.会变成 Beijing. like I
所以在旧数组中的操作,我们选择从数组最后端开始操作
在这里插入图片描述
上述过程在放置完Beijing.之后,p要向前寻找到like,然后是I,这是个循环的过程
因为当p指向了I的时候,前面的元素不是’ ',我们无法预料,那么这个情况单独考虑,这样循环的条件就是当p!=str
代码部分实现:

	while (p != str)
	{
		char* t = p - 1;
		if (*t == ' ')
		{
			while ((*p) != '\0' && (*p) != ' ')
			{
				*q++ = *p++;//q是指向新数组的指针
			}
			p = t + 1;
			*(q) = ' ';
			q++;
		}
		p--;
	}

当p==str的情况下,依旧是赋值到新的数组中

	while ((*str) != ' ')
	{
		*q++ = *str++;
	}
	*q = '\0';
	//注意当查找到str的时候,在新数组中,相当于到了结束的时候,所以在结尾加上\0,表示结束标志

解法1完整代码

void ReserveString(char* str)
{
	int i = 0;
	int len = strlen(str);
	char b[100] = { 0 };
	char* q = b;
	char* p = str+len-1;
	while (p != str)
	{
		char* t = p - 1;
		if (*t == ' ')
		{
			while ((*p) != '\0' && (*p) != ' ')
			{
				*q++ = *p++;
			}
			p = t + 1;
			*(q) = ' ';
			q++;
		}
		p--;
	}

	while ((*str) != ' ')
	{
		*q++ = *str++;
	}
	*q = '\0';
	printf("%s", b);
}
int main()
{
	char str[100] = {0};	
	gets_s(str);
	ReserveString(str);
	return 0;
}

解法2思路解析

前面说到,在看到这个题目的时候,我的第一做法是开辟新的空间,进行倒置的存储,其实正常情况下,如果可以使用原空间,那么一定是最好的,毕竟我们要考虑时间复杂和空间复杂。倘若我们要倒置的字符串很长,新开辟空间会耗费资源的。
接下来的写法就是使用原空间的做法
首先:将这个字符串中的每一个单词逆序
然后:将整个字符串逆序一次
图解:
在这里插入图片描述
实现逆序操作的代码想必大家已经熟的不能再熟了
使用两个指针指向字符串前后两个位置,然后进行前后逆序
代码如下:

void resverse(char* left, char* right)
{
	assert(left && right);//避免left,right为空指针
	while (left < right)
	{
		char t = *left;
		*left = *right;
		*right = t;
		left++;
		right--;
	}
}

逆序三个单词的代码:

//start和end代表一个单词首尾指针
while (*end != '\0')//条件是不遇到\0,不然之间还是有单词的
	{
		while (*end != ' ' && *end!='\0')
		{
			end++;
		}
		//逆序这两个指针之间的内容,相当于逆序单词
		//注意end-1,因为此刻是end=' '了,单词是在' '前面的 
		resverse(start, end - 1);
		
		//指向下一个单词
		//如果到了最后一个\0,start就不能往后走了
		if (*end == '\0')
		{			
			start = end;
		}
		//如果没有遇到就是正常的指向下一个单词
		else
		{
			start = end + 1;
		}
		end = start;
	}

逆序整个单词的代码:

	resverse(a, a + strlen(a) - 1);

解法2完整代码

#include<stdio.h>
#include<string.h>
#include<assert.h>
void resverse(char* left, char* right)
{
	assert(left && right);//避免left,right为空指针
	while (left < right)
	{
		char t = *left;
		*left = *right;
		*right = t;
		left++;
		right--;
	}
}
int main()
{
	char a[100] = { 0 };
	gets_s(a);
	char* start = a;
	char* end = start;
	while (*end != '\0')//条件是不遇到\0,不然之间还是有单词的
	{
		while (*end != ' ' && *end!='\0')
		{
			end++;
		}
		//逆序这两个指针之间的内容,相当于逆序单词
		resverse(start, end - 1);
		//指向了下一个单词
		if (*end == '\0')
		{
			//如果到了最后一个\0,start就不能往后走了
			start = end;
		}
		else
		{
			start = end + 1;
		}
		end = start;
	}

	//接下来整个串再逆序一下
	resverse(a, a + strlen(a) - 1);
	printf("%s", a);
	return 0;
}

最后的总结

当时写题目的时候是有时间限制的,我没有在规定的时间内完成代码,可以给大佬们看看我的代码

#include<stdio.h>
#include<string.h>
void ReserveString(char* str)
{
	int i = 0;
	int len = strlen(str);
	char b[100] = { 0 };
	char* q = b;
	char* p = str+len-1;
	while (p != str)
	{
		char* t = p - 1;
		if (*t == ' ')
		{
			while ((*p) != '\0' && (*p) != ' ')
			{
				*q++ = *p++;
			}
			p = t + 1;
			*(q) = ' ';
			q++;
		}
		p--;
	}

	while ((*str) != ' ')
	{
		*q++ = *str++;
	}
	*q = '\0';
	str = b;
}
int main()
{
	char str[100] = {0};	
	gets_s(str);
	ReserveString(str);
	printf("%s", str);
	return 0;
}

就是题解一的过程,不同的是当时在考虑的时候,想着指针是传递地址的,所以在倒置函数内部,我把新数组的空间赋给了str,然后返回到main函数中str找到b的首地址,完美的把后面的内容打印出来,当时以为是这么回事,后来截至时间过了之后调试了一把。。。。
函数结束后,函数内部的空间会被自动释放,b那个空间是个什么啊,还赋值给str了。。。。
在这里插入图片描述
各位大佬观看时如果发现错误,务必要及时指正我啊!

ps:说说最近,开通博客呢是想记录自己重学C语言的历程,但是距离上次博客已经是多久以前了,其实我花在C上的时间远远少于当时心中所想。所以说我在干什么呢。。。对接口。。感觉在写shi,循环里面发请求加异步还传多组参数。c,我究竟是什么丧心病狂。
然后又到了考试周,明天还有数据结构摸底测,希望快点放假,寒假我要好好学习!

  数据结构与算法 最新文章
【力扣106】 从中序与后续遍历序列构造二叉
leetcode 322 零钱兑换
哈希的应用:海量数据处理
动态规划|最短Hamilton路径
华为机试_HJ41 称砝码【中等】【menset】【
【C与数据结构】——寒假提高每日练习Day1
基础算法——堆排序
2023王道数据结构线性表--单链表课后习题部
LeetCode 之 反转链表的一部分
【题解】lintcode必刷50题<有效的括号序列
上一篇文章      下一篇文章      查看所有文章
加:2021-11-29 16:33:23  更:2021-11-29 16:33:27 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/9 16:01:09-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码