(PAT乙级) 1084 外观数列 (C语言实现,不要使用strcat)

本文介绍了一种计算给定整数d的外观数列第N项的方法,提到与字符串处理相关的问题,特别是如何通过使用index索引优化strcat操作以避免运行超时。作者提供了一个C语言代码示例,展示了如何在处理较长字符串时提高效率。
摘要由CSDN通过智能技术生成

外观数列是指具有以下特点的整数序列:

d, d1, d111, d113, d11231, d112213111, ...

它从不等于 1 的数字 d 开始,序列的第 n+1 项是对第 n 项的描述。比如第 2 项表示第 1 项有 1 个 d,所以就是 d1;第 2 项是 1 个 d(对应 d1)和 1 个 1(对应 11),所以第 3 项就是 d111。又比如第 4 项是 d113,其描述就是 1 个 d,2 个 1,1 个 3,所以下一项就是 d11231。当然这个定义对 d = 1 也成立。本题要求你推算任意给定数字 d 的外观数列的第 N 项。

输入格式:

输入第一行给出 [0,9] 范围内的一个整数 d、以及一个正整数 N(≤ 40),用空格分隔。

输出格式:

在一行中给出数字 d 的外观数列的第 N 项。

输入样例:

1 8

输出样例:

1123123111
题目分析:

本题和 PTA 1078压缩字符串一题比较类似https://pintia.cn/problem-sets/994805260223102976/exam/problems/994805262018265088

不过需要注意的是 strcat的效率问题,strcat在追加字符上细性能比较差,本题如果使用strcat实现 +=字符的效果会出现运行超时的情况

而且,本题的结果会比较长,建议使用100000以上的数组空间!

如何优化strcat?

这里我的做法是使用index索引代替,使用index索引时间上能缩小非常大的时间。下图中如果使用strcat会超过400ms,使用索引之后反而只需要4ms

image-20240322153020918

#include <ctype.h>
#include <math.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

void reverse_str(char *str){
	int left=0,right=strlen(str)-1;
	char ch;
	while(left<right){
		ch=str[left];
		str[left]=str[right];
		str[right]=ch;
		left++;
		right--;	
	}
}

char* to_string(int num){
	char* str=calloc(500,sizeof(char));
	memset(str,'\0',sizeof(str));
	int index=0;
	while(num!=0){
		str[index++]=(num%10)+'0';
		num/=10;
	}
	reverse_str(str);
	return str;
}

char* getnext(char *str){
	int num[500];
	char *nstr=calloc(100000,sizeof(char));
	memset(num,0,sizeof(num));
	int i,j,index=0;
	num[str[0]]++;
	for(i=1;str[i]!='\0';i++){
		if(str[i]!=str[i-1]){
			nstr[index++]=str[i-1];
			nstr[index++]=num[str[i-1]]+'0';
//			strcat(nstr,ch);
			num[str[i-1]]=0;
		}
		num[str[i]]++;
	}
	
	int end=strlen(str)-1;
	nstr[index++]=str[end];
	nstr[index++]=num[str[end]]+'0';
//	strcat(nstr,ch);
	return nstr;
}

int main() {
  int d,n,i;
  char str[100000]="";
  scanf("%d %d",&d,&n);
  str[0]=d+'0';
  
  char *next,*cur=str;
  for(i=2;i<=n;i++){
  	next=getnext(cur);
  	cur=next;
//  	printf("%s\n",str);
  }
  printf("%s\n",cur);
  return 0;
}

  • 5
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值