基数排序的奇技淫巧

Posted Sakits

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了基数排序的奇技淫巧相关的知识,希望对你有一定的参考价值。

  据说是WC T2的子任务,ai<=2^32-1的基数排序,那么就把一个数分成几段多关键字基数排序就行了,类似后缀数组?

  分成8位/8位/8位/8位比分成16位/16位要快【丧病的底层优化

  写的是16/16的

代码如下:

 

#include<iostream>
#include<cstdlib>
#include<cstring>
#include<cstdio>
#include<algorithm>
#include<queue>
#include<cmath>
#include<map>
#define ll long long 
using namespace std;
const int maxn=500010,inf=1e9;
int n,m,x,y,z,tot;
int a[maxn],sa[maxn],sa2[maxn],sum[65536];
void read(int &k)
{
    int f=1;k=0;char c=getchar();
    while(c<\'0\'||c>\'9\')c==\'-\'&&(f=-1),c=getchar();
    while(c<=\'9\'&&c>=\'0\')k=k*10+c-\'0\',c=getchar();
    k*=f;
}
int main()
{
    read(n);m=65535;
    for(int i=1;i<=n;i++)read(a[i]);
    memset(sum,0,sizeof(sum));
    for(int i=1;i<=n;i++)sum[a[i]&((1<<16)-1)]++;
    for(int i=1;i<=m;i++)sum[i]+=sum[i-1];
    for(int i=n;i;i--)sa2[sum[a[i]&((1<<16)-1)]--]=i;
    memset(sum,0,sizeof(sum));
    for(int i=1;i<=n;i++)sum[a[i]>>16]++;
    for(int i=1;i<=m;i++)sum[i]+=sum[i-1];
    for(int i=n;i;i--)sa[sum[a[sa2[i]]>>16]--]=sa2[i];
    for(int i=1;i<=n;i++)printf("%d ",a[sa2[i]]);
    return 0;
}
View Code

 

以上是关于基数排序的奇技淫巧的主要内容,如果未能解决你的问题,请参考以下文章

数据结构-排序之基数排序(使用java代码实现)

数据结构-排序之基数排序(使用java代码实现)

基数排序:基数排序中的“组”是啥意思?

算法-java代码实现基数排序

改变基数排序基础?

LSD基数排序c++代码