每天一题(47) - 旋转单词顺序 + 左旋转字符串 + 右旋转字符串

这篇博客探讨了如何解决剑指Offer中关于旋转单词顺序和字符串左右旋转的问题。对于旋转单词顺序,提出了两种思路,一是先转置单词再转置句子,二是先转置句子再转置单词。对于左旋转字符串,通过将字符串分为两部分并进行旋转,然后整体翻转来实现。而右旋转字符串可以通过转换为左旋转问题来解决。
摘要由CSDN通过智能技术生成

题目来自剑指Offer

题目(1)


思路:

假设句子中的三个单词为XYZ,则我们要求的是ZYX。

给出字符串的转置性质:

可以推出:


此时可以推出思路一:

可以先对句子中的所有单词进行转置,之后再进行(句子)整体转置。

思路二:先进行句子转置,后对每一个单词转置,也是可以的。

代码:根据思路一写的:先对单词进行转置,之后对句子进行转置。

#include <iostream>
#include <assert.h>
using namespace std;

void Reverse(char* pStartStr,char* pEndStr)
{
	assert(pStartStr && pEndStr);
	assert(pEndStr >= pStartStr);
	char cTmp;
	while (pEndStr > pStartStr)
	{
		cTmp = *pStartStr;
		*pStartStr = *pEndStr;
		*pEndStr = cTmp;

		pStartStr++;
		pEndStr--;
	}
}

char* ReverseSent(char* pStr)
{
	assert(pStr);
	int nLenStr = strlen(pStr);
	if (nLenStr == 0)
	{
		return "";
	}
	
	//翻转单词
	char* pStartWord = pStr;
	char* pEndWord = pStr;
	while(*pStartWord != 0)
	{
		if (*pStartWord == ' ')
		{
			pStartWord++;
			pEndWord++;
		}
		else if (*pEndWord == 0 || *pEndWord == ' ')
		{
			pEndWord--;
			Reverse(pStartWord,pEndWord);

			pStartWord = ++pEndWord;
		}
		else
		{
			pEndWord++;
		}
	}
	//翻转句子
	Reverse(pStr,pStr + nLenStr - 1);
	return pStr;
}

int main()
{
	char Str[100] = "  I  am a Student.     ";
	cout<<ReverseSent(Str)<<endl;
	system("pause");
	return 1;
}
这里建议,使用思路一写代码,因为在处理左旋转字符串或者右旋转字符串时,就可以按照统一的思路做了,不易混淆。

左旋转字符串

题目:


所谓左旋:就是把字符串左边若干个单词移动到字符串尾部

举例:abcdefg - 左旋2位 - cdefgab

观察:把左边的字符串移动到右边时,其移动的字符串时整体移动的,内部字符也没有变化。这和问题一的思路是一样的。

思路:和第一个题目类似,可以根据旋转的单词个数,把字符串分成两部分XY。之后进行旋转。

旋转时注意:原始串为XY,目标串为YX。

步骤:

先对每一部分旋转,abcdefg 变成bagfedc

之后,对整个字符串进行翻转。bagfedc - cdefgab

代码:

char* ReverseSent(char* pStr,int nCount)
{
	assert(pStr);
	int nLenStr = strlen(pStr);
	if (nLenStr < nCount)
	{
		return pStr;
	}
	
	//处理字符串
	char* pStartFirst = pStr;
	char* pEndFirst = pStr + nCount - 1;
	char* pStartSec = pEndFirst + 1;
	char* pEndSec = pStr + nLenStr - 1;

	//翻转单词
	Reverse(pStartFirst,pEndFirst);
	Reverse(pStartSec,pEndSec);
	//翻转句子
	Reverse(pStr,pStr + nLenStr - 1);
	return pStr;
}

右旋转字符串

所谓右旋:就是把字符串右边若干个单词移动到字符串头部

举例:abcdefg- 右旋5位 - cdefgab

观察:其实和左旋差不多,区别就是旋转的单词长度变化了。

即,右旋五位,相当于左旋2位。

思路:把右旋转换为左旋做,就ok了。

代码:

char* ReverseSent(char* pStr,int nCount)
{
	assert(pStr);
	int nLenStr = strlen(pStr);
	if (nLenStr < nCount)
	{
		return pStr;
	}
	
	//处理字符串
	char* pStartFirst = pStr;
	char* pEndFirst = pStr + nLenStr - nCount - 1;// 和左旋的区别在这里
	char* pStartSec = pEndFirst + 1;
	char* pEndSec = pStr + nLenStr - 1;

	//翻转单词
	Reverse(pStartFirst,pEndFirst);
	Reverse(pStartSec,pEndSec);
	//翻转句子
	Reverse(pStr,pStr + nLenStr - 1);
	return pStr;
}



评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值