信息学奥赛初赛天天练-56-CSP-J2019完善程序2-双关键字排序、计数排序、前缀和、前缀自增、后缀自增、数组下标自增

PDF文档公众号回复关键字:20240805

1 完善程序 (单选题 ,每小题3分,共30分)

计数排序

计数排序是一个广泛使用的排序方法。下面的程序使用双关键字计数排序,将 n 对 10000以内的整数,从小到大排序。

例如有三对整数 (3,4)、(2,4)、(3,3),那么排序之后应该是 (2,4)、(3,3)、(3,4) 。

输入第一行为 n,接下来 n 行,第 i 行有两个数 a[i] 和 b[i],分别表示第 i 对整数的第一关键字和第二关键字。

从小到大排序后输出。

数据范围 1<n<10^7,1<a[i],b[i]< 10^4。

提示:应先对第二关键字排序,再对第一关键字排序。数组 ord[] 存储第二关键字排序的结果,数组 res[] 存储双关键字排序的结果。

试补全程序。

01 #include <cstdio>
02 #include <cstring>
03 using namespace std;
04 const int maxn = 10000000;
05 const int maxs = 10000;
06 
07 int n;
08 unsigned a[maxn], b[maxn],res[maxn], ord[maxn];
09 unsigned cnt[maxs + 1];
10 int main() {
11     scanf("%d", &n);
12     for (int i = 0; i < n; ++i) 
13         scanf("%d%d", &a[i], &b[i]);
14     memset(cnt, 0, sizeof(cnt));
15     for (int i = 0; i < n; ++i)
16         ①; // 利用 cnt 数组统计数量
17     for (int i = 0; i < maxs; ++i) 
18         cnt[i + 1] += cnt[i];
19     for (int i = 0; i < n; ++i)
20         ②; // 记录初步排序结果
21     memset(cnt, 0, sizeof(cnt));
22     for (int i = 0; i < n; ++i)
23         ③; // 利用 cnt 数组统计数量
24     for (int i = 0; i < maxs; ++i)
25         cnt[i + 1] += cnt[i];
26     for (int i = n - 1; i >= 0; --i)
27         ④ // 记录最终排序结果
28     for (int i = 0; i < n; i++)
29         printf("%d %d", ⑤);
30 
31     return 0;
32}

1 ①处应填( )[3分]

A ++cnt [i]

B ++cnt[b[i]]

C ++cnt[a[i] * maxs + b[i]]

D ++cnt[a[i]]

2 ②处应填( )[3分]

A ord[--cnt[a[i]]] = i

B ord[--cnt[b[i]]] = a[i]

C ord[--cnt[a[i]]] = b[i]

D ord[--cnt[b[i]]] = i

3 ③处应填( )[3分]

A ++cnt[b[i]]

B ++cnt[a[i] * maxs + b[i]]

C ++cnt[a[i]]

D ++cnt [i]

4.④处应填( )[3分]

A res[--cnt[a[ord[i]]]] = ord[i]

B res[--cnt[b[ord[i]]]] = ord[i]

C res[--cnt[b[i]]] = ord[i]

D res[--cnt[a[i]]] = ord[i]

5 ⑤处应填( )[3分]

A a[i], b[i]

B a[res[i]], b[res[i]]

C a[ord[res[i]]]j b[ord[res[i]]]

D a[res[ord[i]]]j b[res[ord[i]]]

2 相关知识点

1) 计数排序

计数排序又称为鸽巢原理,是对哈希直接定址法的变形应用,具体思路为
统计相同元素出现次数
根据统计的结果将序列回收到原来的序列中

#include<bits/stdc++.h>
using namespace std;

int main(){
	int a[10]={3,4,2,7,5,4,3,3,3,5};
	int cnt[7]={0};//cnt数组记录对应下标出现次数 
	for(int i=0;i<10;i++){
		cnt[a[i]]++;
	}
	
	for(int i=0;i<=7;i++){//枚举对应范围的数 从最小到最大,本示例从0~7即可 
		while(cnt[i]>0){//一个数字出现多次时,cnt[i]为对应的数为出现几次
			cout<<i<<" ";
			cnt[i]--;
		}
	}
	
	return 0;
}

/*
输出
2 3 3 3 3 4 4 5 5 7 
*/ 

2) 前缀和

前缀和就是数组的前i项之和

s[1]=a[1]

s[2]=a[1]+a[2]

s[3]=a[1]+a[2]+a[3]

s[4]=a[1]+a[2]+a[3]+a[4]

s[5]=a[1]+a[2]+a[3]+a[4]+a[5]

#include<bits/stdc++.h>
using namespace std;

int main(){
	int a[10]={1,2,3,4,5,6,7,8,9,10};
	int s[10]={0};//存放前i项和数组 
	int sum=0;//累加变量 
	for(int i=0;i<10;i++){
		sum+=a[i];
		s[i]+=sum;
	}
	for(int i=0;i<10;i++){
		cout<<s[i]<<" ";
	}
	return 0;
}

/*
输出
1 3 6 10 15 21 28 36 45 55 
*/

3) 计数排序与前缀和

计数排序和前缀和结合,把通过一个排序数组记录待排序数组的下标

#include<bits/stdc++.h>
using namespace std;
/*
  计数排序和前缀和结合,可以和原数组的数一一对应 
*/ 
int main(){
	int a[10]={3,4,2,7,5,4,3,3,3,5};
	int b[10]={0};//从小到大记录对应数在a数组的下标 
	int cnt[15]={0};//cnt数组记录对应下标出现次数 
	for(int i=0;i<10;i++){//累加a数组每个数出现次数 
		cnt[a[i]]++;
	}//0 0 1 4 2 2 0 1 0 0
	
	for(int i=0;i<15;i++){//前缀和计算 为a中每个数计算一个对应下标 
		cnt[i+1]+=cnt[i];
	}//0 0 1 5 7 9 9 10 10 10

	for(int i=0;i<10;i++){//把a数组数放入b数组通过下标排序,值为a数组的下标 
		b[--cnt[a[i]]]=i;
	}//2 3 3 3 3 4 4 5 5 7  
	return 0;
}

/*
输出
2 3 3 3 3 4 4 5 5 7 
*/ 

4) 后缀自增和前缀自增

前缀自增 ++i,后缀自增i++

i++只有i变量加+1,i++的表达式不加+1

++i则变量加1,++i表达式加1

#include<bits/stdc++.h>
using namespace std;

int main(){
	int a[5]={0,1,2,3,4};
	int i=0;
	cout<<a[i++]<<endl;//i++表达式不加,输出a[0]的值,对应为0 
	i=0;
	cout<<a[++i]<<endl;//++i表达式加1,输出a[1]的值,对应为1 
	i=0;
	i++;
	cout<<a[i]<<endl;//i++后i变量加1, 出a[1]的值,对应为1 
	i=0;
	++i;
	cout<<a[i]<<endl;//++i后i变量加1, 出a[1]的值,对应为1 
	return 0;
}

3 思路分析

1 ①处应填( B )[3分]

A ++cnt [i]

B ++cnt[b[i]]

C ++cnt[a[i] * maxs + b[i]]

D ++cnt[a[i]]

分析

通过cnt数组对b数组进行排序,如果有重复会累加到cnt当前元素中
例如
int b[3]={4,4,3}
cnt数组的值如下图

2 ②处应填( D )[3分]

A ord[--cnt[a[i]]] = i

B ord[--cnt[b[i]]] = a[i]

C ord[--cnt[a[i]]] = b[i]

D ord[--cnt[b[i]]] = i

分析

在ord数组中,从小到大记录b数组的位置
ord[0]=2 对应b[2]=3
ord[1]=1 对应b[1]=4
ord[2]=0 对应b[0]=4

3 ③处应填( C )[3分]

A ++cnt[b[i]]

B ++cnt[a[i] * maxs + b[i]]

C ++cnt[a[i]]

D ++cnt [i]

分析

进行这里之前,对cnt数组进行了清0
21     memset(cnt, 0, sizeof(cnt));

通过cnt数组对a数组进行排序,如果有重复会累加到cnt当前元素中
例如
int a[3]={3,2,3}
cnt数组的值如下图

4.④处应填( A )[3分]

A res[--cnt[a[ord[i]]]] = ord[i]

B res[--cnt[b[ord[i]]]] = ord[i]

C res[--cnt[b[i]]] = ord[i]

D res[--cnt[a[i]]] = ord[i]

分析

在res数组中,从小到大记录b数组的下标,b数组下标和a数组下标对应
    
a[ord[i]] ord[]已经对b进行了排序,n-1~0 从大到小找对应的a
cnt[a[ord[i]]] 找到a后,以下标放入res数组,进行计数排序

res[--cnt[a[ord[i]]]] = ord[i] --用ord[i]赋值,记录res[]存放的b数组下标

例如
int a[3]={3,2,3}
int b[3]={4,4,3}
res[--cnt[a[ord[2]]]]= res[--cnt[a[0]]]=res[--cnt[3]]=res[2]
=ord[2]=0
res[--cnt[a[ord[1]]]]= res[--cnt[a[1]]]=res[--cnt[2]]=res[0]
=ord[1]=1
res[--cnt[a[ord[0]]]]= res[--cnt[a[2]]]=res[--cnt[3]]=res[1]
=ord[0]=2

26     for (int i = n - 1; i >= 0; --i)
因为ord[]是从小到大,从n-1~0,从大到小循环,a相同时,取出的第1次取出的下标大,可以对应
如果从0~n-1循环,正好相反,导致b排序不对

5 ⑤处应填( B )[3分]

A a[i], b[i]

B a[res[i]], b[res[i]]

C a[ord[res[i]]]j b[ord[res[i]]]

D a[res[ord[i]]]j b[res[ord[i]]]

分析

res[]对a和b数组进行计数排序,按a数组从小到大,如果a相同按b数组从小到大
res[]数组的值分别位a和b数组的下标,a和b数组下标对应
所以按res下标从小到大输出a和b的值即可

posted @ 2024-08-05 18:46  new-code  阅读(1)  评论(0编辑  收藏  举报