小X与队列

时间限制: 1 Sec 内存限制: 128 MB
题目描述
小X正和同学们做列队的练习。
有n名同学排成一路纵队,编号为i的同学排在从前往后数第i个位置上,即:初始时的队列为1, 2, 3, …, n。
接下来小X会发出若干条指令,每条指令形如“请编号为x的同学排到最前面来”。(例如:若当前时刻的队列为5, 4, 3, 2, 1,发出一条x=2的指令后,队列变成了2, 5, 4, 3, 1。)
小X发出了很多很多指令,同学们晕头转向不知道该怎么排列。于是就请你算一算,执行完这些指令后,队列应该变成什么样?

输入
第一行两个用空格隔开的正整数n和m,分别表示人数和指令数。
第二行m个用空格隔开的正整数x[i],按顺序表示每次指令的x值。

输出
输出仅有一行包含n个正整数,相邻两个数之间用一个空格隔开,表示执行完所有指令后的队列。

样例输入
4 3
2 3 2
样例输出
2 3 1 4

提示
样例解释
第一条指令前:1 2 3 4
第一条指令后(x=2):2 1 3 4
第二条指令后(x=3):3 2 1 4
第三条指令后(x=2):2 3 1 4

数据范围
对于30%的数据,1<=n,m<=1000
对于另外30%的数据,n=m,且1~n每个数在x[i]中恰好出现一次。
对于100%的数据,1<=n,m<=100000

这个代码是缓冲区溢出,不知道为什么?怎么解决?

#include <iostream>
#include <algorithm>
#include <string.h>
#include <stdio.h>
int a[100005];
int b[100005];
using namespace std;
int main()
{
	int n,m;
	cin>>n>>m;
	for(int i=0;i<n;i++) a[i]=i+1;
	
	for(int i=0;i<m;i++){
		int x;
		scanf("%d",&x);
		int *k;
		int t;
		k=find(a,a+n,x);
		t=*k;
		b[0]=x;
		memcpy(&b[1],a,sizeof(int)*(t-1));
		memcpy(&b[t],&a[t],sizeof(int)*(n-t));
		memcpy(a,b,sizeof(int)*n);
		memset(b,0,sizeof(b));
	}
	for(int i=0;i<n;i++){
		if(i==n-1) printf("%d ",a[i]);
		else printf("%d ",a[i]);
	}
	return 0;
}

这个是借鉴别人的

#include<bits/stdc++.h>
using namespace std;
int a[100005],b[100005],c[100005];
int main()
{
	int n,m;
	int k=0;
	cin>>n>>m;
	for(int i=1;i<=m;i++){     //从1 开始因为编号从1开始
		cin>>a[i];
	}
	for(int i=m;i>=1;i--){   //从后向前是因为最后一个编号要放在最前面
		if(b[a[i]]==0){
			c[k++]=a[i];     
			b[a[i]]=1;   //使提到前面的置1,在后面输出
		}
	}
	for(int i=0;i<k;i++){      //被提到前面
		printf("%d ",c[i]);
	}
	for(int i=1;i<=n;i++){    //没有被提到前面的
		if(b[i]==0)	printf("%d ",i);
	}
}

猜你喜欢

转载自blog.csdn.net/weixin_42868863/article/details/85220034
今日推荐