zoukankan      html  css  js  c++  java
  • luogu3809 后缀排序 后缀数组

    ref and 挑战程序设计竞赛。
    主要是发现自己以前写得代码太难看而且忘光了,而且我字符串死活学不会啊,kmp这种东西我都觉得是省选+难度啊QAQ

    #include <iostream>
    #include <cstring>
    #include <cstdio>
    using namespace std;
    int n, cnt[1000005], rnk[1000005], saa[1000005], tmp[1000005], m=128, p;
    char ss[1000005];
    void sasort(){
    	for(int i=0; i<m; i++)	cnt[i] = 0;
    	for(int i=0; i<n; i++)	cnt[rnk[i]]++;
    	for(int i=1; i<m; i++)	cnt[i] += cnt[i-1];
    	for(int i=n-1; i>=0; i--)	saa[--cnt[rnk[tmp[i]]]] = tmp[i];//大约就是rnk[i](第一关键字排序后的下标数组)定了一段一段的位置以后,依据第二关键字精确确定位置
    }
    void getSuffixArray(){
    	for(int i=0; i<n; i++){
    		rnk[i] = ss[i];
    		tmp[i] = i;
    	}
    	sasort();
    	for(int j=1; p<n; j<<=1){
    		p = 0;
    		for(int i=n-j; i<n; i++)	tmp[p++] = i;
    		for(int i=0; i<n; i++)
    			if(saa[i]>=j)
    				tmp[p++] = saa[i] - j;//构造第二关键字数组,tmp[0]表示第二关键字最小(第0小)的元素位置i,(也就是i+j..n这样的后缀最小),以此类推
    		sasort();
    		swap(rnk, tmp);//因为以前的rank也要用到
    		p = 1;
    		rnk[saa[0]] = 0;
    		for(int i=1; i<n; i++)
    			if(tmp[saa[i-1]]==tmp[saa[i]] && tmp[saa[i-1]+j]==tmp[saa[i]+j])//这里是不会越界的……
    				rnk[saa[i]] = p - 1;
    			else
    				rnk[saa[i]] = p++;
    		m = p;//p代表有多少种rank,m代表字符集个数
    	}
    }
    int main(){
    	scanf("%s", ss);
    	n = strlen(ss);
    	ss[n++] = 0;
    	getSuffixArray();
    	for(int i=1; i<n; i++)
    		printf("%d ", saa[i]+1);
    	printf("
    ");
    	return 0;
    }
    
  • 相关阅读:
    oracle 释放表空间到OS(resize)
    dblink查找对应的目标端session
    oralce move和shrink释放高水位
    使用ogg实现oracle到postgresql表的实时同步
    oracle_fdw的安装和使用
    postgresql数据库升级
    ORA-00054: 資源正被使用中, 請設定 NOWAIT 來取得它, 否則逾時到期
    贪心算法应用-哈夫曼编码
    9 深拷贝学习
    8 标准库C文件
  • 原文地址:https://www.cnblogs.com/poorpool/p/8252336.html
Copyright © 2011-2022 走看看