首页 > 其他分享 >C语言王国——数组的旋转(轮转数组)三种解法

C语言王国——数组的旋转(轮转数组)三种解法

时间:2024-06-12 19:03:10浏览次数:26  
标签:numsSize arr 轮转 nums int C语言 数组 printf size

一、题目

给定一个整数数组 nums,将数组中的元素向右轮转 k 个位置,其中 k 是非负数。

示例 1:

输入: nums = [1,2,3,4,5,6,7], k = 3
输出: [5,6,7,1,2,3,4]
解释:
向右轮转 1 步: [7,1,2,3,4,5,6]
向右轮转 2 步: [6,7,1,2,3,4,5]
向右轮转 3 步: [5,6,7,1,2,3,4]

示例 2:

输入:nums = [-1,-100,3,99], k = 2
输出:[3,99,-1,-100]
解释: 
向右轮转 1 步: [99,-1,-100,3]
向右轮转 2 步: [3,99,-1,-100]

二、分析 

2.1 暴力求解法

这题是旋转字符串但是它的实质是将前面n-1个数据往后移一位,然后将最后一个数据移到第一个,旋转几次则执行几次这个步骤。为了变量不被覆盖,变量采取从后向前移动,而最后一个数据利用一个空变量temp去拷贝一份,在前面数据移动完成后则拷贝到第一个数据。如下面代码:

void rotate(int* num ,int k , int size)
{
	k %= size;//size次一次循环
	while (k)
	{
		int temp = num[size-1];
		for (int i = size-1; i > 0; i--)
		{
			num[i] = num[i - 1];
		}
		num[0] = temp;
		k--;
	}
}

int main()
{

	int arr[] = { 7,1,2,3,4,5,6 };
	int size = sizeof(arr) / sizeof(arr[0]);
	int k = 1;
	printf("旋转几次数组:");
	scanf("%d", &k);
	printf("旋转前:");
	for (int i = 0; i < size; i++)
	{
		printf("%d ", arr[i]);
	}
	rotate(arr, k, size);
	printf("\n旋转后为:");
	for (int i = 0; i < size; i++)
	{
		printf("%d ", arr[i]);
	}

	return 0;
}

2.2  找规律

我们发现暴力求解虽然可以解出此题,但是时间复杂度O(N^{2}),那我们有什么办法去降低时间复杂度呢?我们可以尝试降低它的循环次数,找找看他们有什么规律?

如图,将前面n-k个数字逆置然后将后面n个逆置,然后将他们整体逆置。(红色位将要进行逆置的数)

代码如下:

 

void Inversion(int* arr, int l, int r)
{
	while (l < r)
	{
		int temp = arr[l];
		arr[l] = arr[r];
		arr[r] = temp;
		l++;
		r--;
	}
}

void rotate(int* nums, int numsSize, int k) {
	k %= numsSize;//不让数组越界,完成一次数组长度的旋转数组不变
	Inversion(nums, 0, numsSize - k - 1);
	Inversion(nums, numsSize - k, numsSize - 1);
	Inversion(nums, 0, numsSize - 1);
}

int main()
{

	int arr[] = { 7,1,2,3,4,5,6 };
	int size = sizeof(arr) / sizeof(arr[0]);
	int k = 1;
	printf("旋转几次数组:");
	scanf("%d", &k);
	printf("旋转前:");
	for (int i = 0; i < size; i++)
	{
		printf("%d ", arr[i]);
	}
	rotate(arr, size, k);
	printf("\n旋转后为:");
	for (int i = 0; i < size; i++)
	{
		printf("%d ", arr[i]);
	}

	return 0;
}

*这里要注意数组越界访问,我们发现数组完成一次数组长度的旋转,数组不变,所以我们取模于数组长度。 

2.3 追求时间效率,以空间换时间

这里我们不讨论空间复杂度只优化时间复杂度,所以我们可以新开辟一段空间,将后n个旋转的数字先放入我们开辟的空间,然后再将前面的n-k个数字放入空间后面。

如代码:

void rotate(int* nums, int numsSize, int k) {
	k %= numsSize;
	int* temp = (int*)malloc(sizeof(int) * numsSize);
	memcpy(temp, nums + numsSize - k, sizeof(int) * k);
	memcpy(temp + k, nums, sizeof(int) * (numsSize - k));
	memcpy(nums, temp, sizeof(int) * numsSize);
	free(temp);//释放临时空间
	temp = NULL;
}

int main()
{

	int arr[] = { 7,1,2,3,4,5,6 };
	int size = sizeof(arr) / sizeof(arr[0]);
	int k = 1;
	printf("旋转几次数组:");
	scanf("%d", &k);
	printf("旋转前:");
	for (int i = 0; i < size; i++)
	{
		printf("%d ", arr[i]);
	}
	rotate(arr, size, k);
	printf("\n旋转后为:");
	for (int i = 0; i < size; i++)
	{
		printf("%d ", arr[i]);
	}

	return 0;
}

三、结论 

每一道题都有属于自己的空间和时间复杂度,当你写出一段代码的时候去想想还能不能继续去优化它,使它的时间和空间复杂度更小。姜糖在这里就是不断在优化它的时间复杂度,从O(N^2)到最后的O(N)。如果大家还有什么不同的看法可以跟姜糖展开讨论哦。

标签:numsSize,arr,轮转,nums,int,C语言,数组,printf,size
From: https://blog.csdn.net/2301_78955228/article/details/139601425

相关文章

  • C语言字符串处理函数strstr的用法
    C语言字符串处理函数strstr的用法在C语言中,strstr函数是一个字符串处理函数,用于在一个字符串(称为“主字符串”)中查找另一个字符串(称为“子字符串”)的首次出现。如果找到子字符串,则该函数返回一个指向主字符串中子字符串首次出现位置的指针;如果没有找到,则返回NULL。函数的原型定......
  • (nice!!!)LeetCode 2865. 美丽塔 I(数组、单调栈)
    2865.美丽塔I思路:方法一,时间复杂度0(n^2)。枚举每一个点i作为当前山脉数组的最高点。然后我们通过预处理遍历其前面和后面,来更新两个数组f1、f2。数组f1[i]:表示在满足非递减的情况下,区间0~i,以点i的高度heighs[i]为最高点所能形成的最大高度和。数组f2[i]:表示在满足非......
  • c语言开发 php扩展 sm4
    首先php可以直接调用openssl直接进行sm4sm3的加密如:openssl_encrypt($plaintext,'sm4-cbc',$key,OPENSSL_RAW_DATA,$iv);openssl_digest('123','sm3')php如果直接调用sm2需要统一使用openssl的evp接口openssl1.1的源码在sm2_crypt文件里面此处只是学习/*gmteste......
  • c语言实现密码学算法应用
    一实验目的   1、掌握对称密钥密码体制的基本概念;   2、掌握对称密钥密码体制DES加密/解密的工作原理;   3、掌握非对称密码算法RSA加密/解密的基本原;   4、通过用DES和RSA算法对实际的数据进行加密/解密运算深刻理解加密算法原理。二实验内容   根据给......
  • 指针和数组-1
    目录1、指针的算术运算指针加上整数:指针减去整数:两个指针相减:2、指针用于数组处理1.访问数组元素:2.遍历数组:3.修改数组元素:​ 4.传递数组到函数:​5.动态内存分配(先了解后面章节会详解):3、指针比较1.检查两个指针是否相等2.检查一个指针是否在另一个之前或之后......
  • c语言目录操作
    在shell中我们可以直接输入命令pwd来显示当前的工作目录,在C程序中调用getcwd函数可以获取当前的工作目录。函数声明:char*getcwd(char*buf,size_tsize);需要头文件:#includegetcwd函数把当前工作目录存入buf中,如果目录名超出了参数size长度,函数返回NULL,如果成功,返回buf......
  • C语言详解(编译和链接)
    Hi~!这里是奋斗的小羊,很荣幸您能阅读我的文章,诚请评论指点,欢迎欢迎~~......
  • Vue 中数组常用方法的使用和示例
    Find查找数组元素Find用来遍历查找数组元素,当找到符合条件的元素时,直接返回。所以Find元素只会返回符合条件的第一个元素//数据types:[{"value":0,"label":"头像素材"},{"value":1,......
  • 请编写一个函数void fun(char a[],char b[],int n),其功能是:删除以各字符串中指定下标
    请编写一个函数voidfun(chara[],charb[],intn),其功能是:删除以各字符串中指定下标的字符。其中,a指向原字符串,删除后的字符串存放在b所指的数组中,n中存放指定的下标。#include<stdio.h>voidfun(chara[],charb[],intn){inti,j=0;for(i=0;a[i]......
  • 【C语言】12.C语言内存函数
    文章目录1.memcpy使用和模拟实现2.memmove使用和模拟实现3.memset函数的使用4.memcmp函数的使用memcpy:内存拷贝memmove:内存移动memset:内存设置memcmp:内存比较1.memcpy使用和模拟实现memcpy:内存拷贝void*memcpy(void*destination,constvoid*source,......