《PTA——C语言源代码》之第1008题

本文介绍了一种在不使用额外数组的情况下,实现整数数组循环右移M位的算法,并给出了具体的实现步骤与源代码。重点在于如何通过有限次的数据移动达到目标效果。

题目介绍

一个数组A中存有N(N&gt0)个整数,在不允许使用另外数组的前提下,将每个整数循环向右移M(M>=0)个位置,即将A中的数据由(A~0~ A~1~……A~N-1~)变换为(A~N-M~ …… A~N-1~ A~0~ A~1~……A~N-M-1~)(最后M个数循环移至最前面的M个位置)。如果需要考虑程序移动数据的次数尽量少,要如何设计移动的方法?

输入格式:每个输入包含一个测试用例,第1行输入N ( 1<=N<=100)、M(M>=0);第2行输入N个整数,之间用空格分隔。

输出格式:在一行中输出循环右移M位以后的整数序列,之间用空格分隔,序列结尾不能有多余空格。

输入样例:

6 2
1 2 3 4 5 6

输出样例:

5 6 1 2 3 4

源代码

#include<stdio.h>
#define MAXSIZE 101

void move(int A[], int N, int M){
  for(; M < N - 1; ++M){
    A[N-M] = A[N-M-1];
  }
  A[1] = A[0];
}

int main(){
  int N, M, A[MAXSIZE]={0};  //A[0]用来存储需要移动的元素
  scanf("%d %d", &N, &M);
  for(int i = 1; i <= N; ++i){
    scanf("%d", &A[i]);
  }
  M = M % N;  //极其重要
  if(N == M){
    for(int i = N; i >= 1; --i){
    if( i == 1 )
      printf("%d", A[i]);
    else printf("%d ", A[i]);
     }
  }
  else {
    for(int i = 1; i <= M;++i){
      A[0] = A[N - i + 1];
      A[N - i + 1] = A[N - M];
      move(A, N, M);
    }
    for(int i = 1; i <= N; ++i){
      if( i == N )
        printf("%d", A[i]);
      else printf("%d ", A[i]);
    }
  }
  return 0;
}

知识分析

思路

拿到此题后的思路为,开辟一个数组大小为N+1,A[0]用来存储元素,下标1——N则为数组正常元素。要移动M个元素,先将最后一个值拿出来存储,然后数组其他的元素往后移动N-M个位置,最后将拿出来的元素放到A[0]即可。具体思路如下:
1. 最后一个元素A[N]赋值给A[0];
2. 令A[N - i + 1] = A[N - M];这样做的目的是将移动M个元素分解为移动1个元素。//晚点用好一点的术语描述,1≤i≤M;
3. 将A[1]到A[N-M]的值均往后移1位;移动完毕之后A[1] = A[0]
4. 重复上述3个步骤,直至i == M即可。

出错分析

  M = M % N;  //极其重要
  if(N == M){
    for(int i = N; i >= 1; --i){
        if( i == 1 )
        printf("%d", A[i]);
        else printf("%d ", A[i]);
      }
  }

提交完整的正确答案之前没有考虑极端情况,即
1. M>N的时候,此时应该对N求模。
2. M==N的时候,直接倒序输出即可。

### PTA C语言编程练习与解决方案 PTA程序设计类实验辅助教学平台是一个在线编程平台,旨在为学生提供良好的编程实践环境,帮助他们掌握编程语言和编程思想,从而提升程序设计能力[^1]。该平台上提供了大量的编程练习目和实验目,涵盖了C语言的基础语法、数据结构以及算法等内容。 以下是几个常见的PTA C语言编程问及其解决方案: #### 1. 输出格式严格要求 在某些情况下,PTA会对输出格式有严格的限制。例如,在处理数组输出时,可能会遇到如下情况:当使用`printf("%d ", A[n])`输出最后一个数后带有多余的空格,这可能导致测试用例无法通过。此时可以通过单独处理最后一项来避免多余空格的发生[^4]。 ```c for (int i = 0; i < n - 1; i++) { printf("%d ", A[i]); } if (n > 0) { // 防止输入为空的情况 printf("%d", A[n - 1]); // 单独打印最后一项 } ``` #### 2. 素数判定问 素数是指除了1和它本身以外再能被其他自然数整除的大于1的自然数。在PTA上的素数相关目通常会涉及如何高效地判断一个数是否为素数,并且需要注意输出格式的要求,比如数字间需留有间隔而行末得多出空格[^5]。 一种简单的实现方法如下所示: ```c #include <stdio.h> #include <math.h> // 判断是否为素数函数 int isPrime(int num) { if (num <= 1) return 0; int sqrtNum = sqrt(num); for (int i = 2; i <= sqrtNum; i++) { if (num % i == 0) return 0; } return 1; } void printPrimes(int start, int end) { int firstFlag = 1; // 控制首项标志位 for (int i = start; i <= end; i++) { if (isPrime(i)) { if (!firstFlag) printf(" "); // 如果是首个元素则先加空格 printf("%d", i); // 打印当前素数值 firstFlag = 0; // 设置后续非首次标记 } } putchar('\n'); // 行结束符 } int main() { int a, b; scanf("%d%d", &a, &b); printPrimes(a, b); return 0; } ``` 此代码片段定义了一个用于检测给定范围内所有素数的功能模块,并妥善解决了连续输出间的分隔及结尾无冗余字符的问。 #### 3. 学生管理系统初步构建 为了验证学习者对C语言的理解程度,许多课程都会布置开发简易的学生管理系统的作业作为综合考核项目之一。这类系统一般包括但限于以下几个功能点:录入学员基本信息、查询特定条件下的记录列表、修改已有条目属性值以及删除指定对象等功能操作[^3]。 下面展示的是一个非常基础版本框架示意: ```c typedef struct Student { char name[20]; int id; }Student; #define MAX_STUDENTS 100 Student students[MAX_STUDENTS]; // 定义存储空间大小 int studentCount = 0; // 当前已存入数量计数器变量声明初始化为零 void addStudent(Student s){ if(studentCount >=MAX_STUDENTS ){ puts("No more space!"); return ; } students[studentCount++]=s; } void displayAll(){ for(int i=0;i<studentCount;i++){ printf("ID:%d Name:%s\n",students[i].id,students[i].name); } } int main(){ Student temp={}; while(1){ printf("\nEnter command(add/display/exit):"); char cmd[10]; scanf("%s",cmd); if(strcmp(cmd,"add")==0){ printf("Enter ID and NAME:"); scanf("%d%s",&temp.id,temp.name); addStudent(temp); }else if(strcmp(cmd,"display")==0){ displayAll(); }else if(strcmp(cmd,"exit")==0){ break; } } return 0; } ``` 上述例子仅展示了最简单的核心逻辑架构,实际应用中还需要考虑更多边界状况处理机制等问。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值