C语言——oj刷题——字符串左旋和轮转数组

简介: C语言——oj刷题——字符串左旋和轮转数组

第一题:字符串左旋

问题:

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

例如:

ABCD左旋一个字符得到BCDA

ABCD左旋两个字符得到CDAB

实现:

当我们谈到字符串左旋时,我们指的是将字符串中的字符向左移动一定数量的位置。这个问题在编程中非常常见,特别是在字符串处理和算法实现中。

在C语言中,我们可以使用一种简单而有效的方法来完成字符串的左旋操作。下面是一个示例代码,演示了如何实现字符串左旋:

#include <stdio.h>
#include <string.h>
 
void reverse(char* str, int start, int end) {
    while (start < end) {
        char temp = str[start];
        str[start] = str[end];
        str[end] = temp;
        start++;
        end--;
    }
}
 
void leftRotateString(char* str, int k) {
    int len = strlen(str);
    k = k % len;  // 处理k大于字符串长度的情况
    reverse(str, 0, k - 1);  // 反转前k个字符
    reverse(str, k, len - 1);  // 反转剩余的字符
    reverse(str, 0, len - 1);  // 整体反转字符串
}
 
int main() {
    char str[] = "abcdefg";
    int k = 2;  // 左旋2个位置
 
    printf("原始字符串: %s\n", str);
    leftRotateString(str, k);
    printf("左旋后的字符串: %s\n", str);
 
    return 0;
}

在上面的示例代码中,我们定义了两个辅助函数:reverse和leftRotateString。


reverse函数用于反转字符串中指定范围内的字符。它使用两个指针(start和end)来遍历字符串,交换对应位置上的字符,直到两个指针相遇。


leftRotateString函数是实现字符串左旋的核心函数。它首先计算旋转位置k与字符串长度的余数,以处理k大于字符串长度的情况。然后,它分别对前k个字符、剩余字符和整个字符串进行反转操作,最终完成字符串的左旋。


在main函数中,我们定义了一个示例字符串"abcdefg"和旋转位置k为2。我们先打印出原始字符串,然后调用leftRotateString函数进行左旋操作,最后打印出左旋后的字符串。


运行上述代码,输出将是:

原始字符串: abcdefg

左旋后的字符串: cdefgab

这就是用C语言完成字符串左旋的方法和示例代码。希望对你有所帮助!如果有任何问题,请随时提问。

第二题:轮转数组(力扣189)

解法一:(暴力解法)

利用循环每次移动一位,代码如下:

//暴力做法

void rotate(int* nums, int numsSize, int k){
    k=k%numsSize;
    int temp;
    while(k--)
    {
        temp=nums[numsSize-1];
        for(int i=numsSize-1;i>0;i--)
        {
            nums[i]=nums[i-1];
        }
        nums[0]=temp;
    }

这种做法应该很好理解,但它的时间复杂度是O(n^2)的,会超时。

解法二:(时间换空间)

利用额外的数组实现,把移动完的数据放在一个新的数组里面去,然后再拷贝回来,代码如下:

void rotate(int*nums,int numsSize,int k)
{
    k=k%numsSize;
    int i,j;
    int a[100000];
    j=numsSize-k;
    for(i=0;i<k;i++) a[i]=nums[j++];
    j=0;
    for(;i<numsSize;i++) a[i]=nums[j++];
    for(i=0;i<numsSize;i++)
    {
        nums[i]=a[i];
    }
}

这样的解法时间复杂度符合要求,但是需要额外的空间实现。

解法三:(前后翻转)

先旋转前面的那部分,在旋转后面的那部分,然后再旋转整个数组。

void swap(int *a,int *b)
{
    int t;
    t=*a;
    *a=*b;
    *b=t;
}
void reverse(int *begin,int *end)
{
    while(begin<end)
    {
        swap(begin,end);
        begin++;
        end--;
    }
}
void rotate(int* nums, int numsSize, int k)
{
    k=k%numsSize;
    reverse(nums,nums+numsSize-k-1);
    reverse(nums+numsSize-k,nums+numsSize-1);
    reverse(nums,nums+numsSize-1);
}

解法四:(环状旋转)

解题思路:

把数组看成一个环状的。按照题目的意思,我们应当让每一个元素往后面走k步,假设x为元素下标,那么它应该到(x+k)%numsSize的地方上去,如图:

我们想把1挪到4的位置上去,直接挪的话,4就会消失,那怎么办呢?可以这样做,申请一个变量temp,先让temp=nums[0],然后temp与nums[3]交换

此时,1已经到4的位置上去了,而4被保留在temp当中,接下来当然是让4到它应当去的位置。

接下来同理:

我们要做到把所有的元素都遍历一遍,都找到它应该去的位置就可以了,我们可以用一个cnt变量来记录遍历了多少个元素,起初从nums[0]开始走,如果走了一圈之后还有元素没有遍历到的话,那就接下来从nums[1]开始走再走一圈,以此类推,显然我们需要两层循环,当cnt==numsSize的时候,就可以结束遍历了。虽然是两重循环,但是时间复杂度是O(N),因为每个元素只被遍历一次。代码如下:

void swap(int* a, int* b)
{
    int t;
    t = *a;
    *a = *b;
    *b = t;
}
void rotate(int* nums, int numsSize, int k)
{
    int cnt = 0;
    int temp, p;
    for (int start = 0; start < numsSize; start++)
    {
        temp = nums[start];
        p = start;
        do
        {
            p = (p + k) % numsSize;
            swap(&temp, &nums[p]);
            cnt++;
        } while (p != start);
        if (cnt == numsSize) break;
    }
 
}
相关文章
|
5月前
|
安全 C语言
C语言中的字符、字符串及内存操作函数详细讲解
通过这些函数的正确使用,可以有效管理字符串和内存操作,它们是C语言编程中不可或缺的工具。
335 15
|
10月前
|
存储 人工智能 程序员
一文彻底搞明白C语言的数组
本文详细介绍了C语言中的数组,包括定义、初始化(静态与动态)、存储方式、访问方法及常用操作,如遍历、修改元素和作为函数参数传递。数组是C语言中最基本的数据结构之一,掌握它对编程至关重要。下篇将介绍二维数组,敬请期待!
536 0
一文彻底搞明白C语言的数组
|
存储 C语言 开发者
【C语言】字符串操作函数详解
这些字符串操作函数在C语言中提供了强大的功能,帮助开发者有效地处理字符串数据。通过对每个函数的详细讲解、示例代码和表格说明,可以更好地理解如何使用这些函数进行各种字符串操作。如果在实际编程中遇到特定的字符串处理需求,可以参考这些函数和示例,灵活运用。
527 10
|
传感器 算法 安全
【C语言】两个数组比较详解
比较两个数组在C语言中有多种实现方法,选择合适的方法取决于具体的应用场景和性能要求。从逐元素比较到使用`memcmp`函数,再到指针优化,每种方法都有其优点和适用范围。在嵌入式系统中,考虑性能和资源限制尤为重要。通过合理选择和优化,可以有效提高程序的运行效率和可靠性。
902 6
|
存储 程序员 编译器
C 语言数组与指针的深度剖析与应用
在C语言中,数组与指针是核心概念,二者既独立又紧密相连。数组是在连续内存中存储相同类型数据的结构,而指针则存储内存地址,二者结合可在数据处理、函数传参等方面发挥巨大作用。掌握它们的特性和关系,对于优化程序性能、灵活处理数据结构至关重要。
|
3月前
|
存储 C语言
`scanf`是C语言中用于按格式读取标准输入的函数
`scanf`是C语言中用于按格式读取标准输入的函数,通过格式字符串解析输入并存入指定变量。需注意输入格式严格匹配,并建议检查返回值以确保读取成功,提升程序健壮性。
1038 0
|
11月前
|
存储 算法 C语言
【C语言程序设计——函数】素数判定(头歌实践教学平台习题)【合集】
本内容介绍了编写一个判断素数的子函数的任务,涵盖循环控制与跳转语句、算术运算符(%)、以及素数的概念。任务要求在主函数中输入整数并输出是否为素数的信息。相关知识包括 `for` 和 `while` 循环、`break` 和 `continue` 语句、取余运算符 `%` 的使用及素数定义、分布规律和应用场景。编程要求根据提示补充代码,测试说明提供了输入输出示例,最后给出通关代码和测试结果。 任务核心:编写判断素数的子函数并在主函数中调用,涉及循环结构和条件判断。
698 23
|
10月前
|
人工智能 Java 程序员
一文彻底搞清楚C语言的函数
本文介绍C语言函数:函数是程序模块化的工具,由函数头和函数体组成,涵盖定义、调用、参数传递及声明等内容。值传递确保实参不受影响,函数声明增强代码可读性。君志所向,一往无前!
436 1
一文彻底搞清楚C语言的函数
|
11月前
|
算法 C语言
【C语言程序设计——函数】利用函数求解最大公约数和最小公倍数(头歌实践教学平台习题)【合集】
本文档介绍了如何编写两个子函数,分别求任意两个整数的最大公约数和最小公倍数。内容涵盖循环控制与跳转语句的使用、最大公约数的求法(包括辗转相除法和更相减损术),以及基于最大公约数求最小公倍数的方法。通过示例代码和测试说明,帮助读者理解和实现相关算法。最终提供了完整的通关代码及测试结果,确保编程任务的成功完成。
622 15
【C语言程序设计——函数】利用函数求解最大公约数和最小公倍数(头歌实践教学平台习题)【合集】
|
11月前
|
C语言
【C语言程序设计——函数】亲密数判定(头歌实践教学平台习题)【合集】
本文介绍了通过编程实现打印3000以内的全部亲密数的任务。主要内容包括: 1. **任务描述**:实现函数打印3000以内的全部亲密数。 2. **相关知识**: - 循环控制和跳转语句(for、while循环,break、continue语句)的使用。 - 亲密数的概念及历史背景。 - 判断亲密数的方法:计算数A的因子和存于B,再计算B的因子和存于sum,最后比较sum与A是否相等。 3. **编程要求**:根据提示在指定区域内补充代码。 4. **测试说明**:平台对代码进行测试,预期输出如220和284是一组亲密数。 5. **通关代码**:提供了完整的C语言代码实现
263 24