Description

  喜欢钻研问题的JS 同学,最近又迷上了对加密方法的思考。一天,他突然想出了一种他认为是终极的加密办法:把需要加密的信息排成一圈,显然,它们有很多种不同的读法。例如下图,可以读作:

  JSOI07 SOI07J OI07JS I07JSO 07JSOI 7JSOI0 把它们按照字符串的大小排序: 07JSOI 7JSOI0 I07JSO JSOI07 OI07JS SOI07J 读出最后一列字符:I0O7SJ,就是加密后的字符串(其实这个加密手段实在很容易破解,鉴于这是突然想出来的,那就^^)。但是,如果想加密的字符串实在太长,你能写一个程序完成这个任务吗?

  这道题好像也做了挺久...一个半小时吧

  首先题目很亲切 总感觉去年暑假在镇中有过类似的题 但是那道题好像有神奇的O(n)做法

  其实问题可以转化为将字符串及其循环以后的n-1个串从小到大排序,然后输出字符串首位的前一位字符

  我们要解决字符串及其循环以后一共n个串的排序

  我用后缀数组解决

  我们可以直接用后缀数组处理出1 << (trunc(ln(n)/ln(2)))长度的排序

  将后缀处理成子串的方法就在原串后面再接一次

  然后再类似于倍增一点点将剩下的长度加入

  实现方法类似于后缀数组 几个细节改动一下

  但是由于要用到中间的数据,所以sa和rank数组要在每一次之后保留不能重复利用

  还是一道非常不错的题呢^

  

 program bzoj1031;
const maxn=;
var i,x,n:longint;
ss:ansistring;
a,s,tmp,rk,sa2:array[-..maxn]of longint;
rank,sa:array[-..,-..maxn]of longint; function max(a,b:longint):longint;
begin
if a>b then exit(a) else exit(b);
end; procedure Suffix_Array;
var i,j,k,p,v0,v00,v01,v1,size:longint;
begin
size:=;
for i:= to n- do rank[,i]:=a[i];
for i:= to size- do s[i]:=;
for i:= to n- do inc(s[rank[,i]]);
for i:= to size- do inc(s[i],s[i-]);
for i:=n- downto do
begin
dec(s[rank[,i]]);
sa[,s[rank[,i]]]:=i;
end;
j:=;k:=;
size:=n;
while j<=n >> do
begin
p:=;
for i:=n-j to n- do
begin
tmp[p]:=i;
inc(p);
end;
for i:= to n- do if sa[k-,i]-j>= then
begin
tmp[p]:=sa[k-,i]-j;
inc(p);
end;
for i:= to size- do s[i]:=;
for i:= to n- do inc(s[rank[k-,i]]);
for i:= to size- do inc(s[i],s[i-]);
for i:=n- downto do
begin
dec(s[rank[k-,tmp[i]]]);
sa[k,s[rank[k-,tmp[i]]]]:=tmp[i];
end;
p:=;rank[k,sa[k,]]:=;
for i:= to n- do
begin
v0:=sa[k,i-];v1:=sa[k,i];
if v0+j<n then v00:=rank[k-,v0+j] else v00:=-;
if v1+j<n then v01:=rank[k-,v1+j] else v01:=-;
if (rank[k-,v0]=rank[k-,v1])and(v00=v01) then rank[k,sa[k,i]]:=p else
begin
inc(p);
rank[k,sa[k,i]]:=p; end;
end;
j:=j << ;inc(k);
end;
end; procedure solve;
var i,j,k,p,x,rec,v0,v1,v00,v01,size:longint;
begin
x:=trunc(ln(n >> )/ln());
size:=max(,n);
for i:= to n- do rk[i]:=rank[x,i];
for i:= to size- do s[i]:=;
for i:= to n- do inc(s[rk[i]]);
for i:= to size- do inc(s[i],s[i-]);
for i:=n- downto do
begin
dec(s[rk[i]]);
sa2[s[rk[i]]]:=i;
end;
rec:=n >> - << x;
while rec<> do
begin
x:=trunc(ln(rec)/ln());
p:=;
for i:=n- << x to n- do
begin
tmp[p]:=i;
inc(p);
end;
for i:= to n- do if sa[x,i]- << x>= then
begin
tmp[p]:=sa[x,i]- << x;
inc(p);
end;
for i:= to size- do s[i]:=;
for i:= to n- do inc(s[rk[i]]);
for i:= to size- do inc(s[i],s[i-]);
for i:= to n- do
begin
dec(s[rk[tmp[i]]]);
sa2[s[rk[tmp[i]]]]:=tmp[i];
end;
p:=;tmp[sa2[]]:=;
for i:= to n- do
begin
v0:=sa2[i-];v1:=sa2[i];
if v0+ << x<n then v00:=rank[x,v0+ << x] else v00:=-;
if v1+ << x<n then v01:=rank[x,v1+ << x] else v01:=-;
if (rk[v0]=rk[v1])and(v00=v01) then tmp[sa2[i]]:=p else
begin
inc(p);
tmp[sa2[i]]:=p;
end;
end;
for i:= to n- do rk[i]:=tmp[i];
dec(rec, << x);
end;
end; begin
readln(ss);n:=length(ss);
for i:= to n- do a[i]:=ord(ss[i+]);
for i:=n to *n- do a[i]:=a[i-n];
n:=n*;
Suffix_Array;
solve;
for i:= to n- do if sa2[i]<(n >> ) then
begin
x:=sa2[i]-;
if x=- then x:=n >> -;
write(ss[x+]);
end;
end.

[BZOJ1031][JSOI2007]字符加密Cipher 解题报告的更多相关文章

  1. [BZOJ1031] [JSOI2007] 字符加密Cipher (后缀数组)

    Description 喜欢钻研问题的JS同学,最近又迷上了对加密方法的思考.一天,他突然想出了一种他认为是终极的加密办法 :把需要加密的信息排成一圈,显然,它们有很多种不同的读法.例如下图,可以读作 ...

  2. [bzoj1031][JSOI2007]字符加密Cipher——后缀数组

    Brief Description 给定一个长度为n的字符串,你需要对其进行加密. 把字符串围成一个环 显然从任意一个位置开始都可以有一个长度为n的串 把产生的n个串按字典序排序,把这n个串的最后一个 ...

  3. BZOJ1031: [JSOI2007]字符加密Cipher

    传送门 后缀数组模板题 //BZOJ 1031 //by Cydiater //2016.9.21 #include <iostream> #include <cstring> ...

  4. [BZOJ1031][JSOI2007]字符加密Cipher(后缀数组)

    传送门 算是个模板. 题目说循环,那就再复制一串拼接上. 然后求后缀数组,再搞就可以. 虽然是求后缀,会在后面多一些字符串,然而题目中说的是循环一圈,但是没有影响. ——代码 #include < ...

  5. 【BZOJ1031】[JSOI2007]字符加密Cipher 后缀数组

    [BZOJ1031][JSOI2007]字符加密Cipher Description 喜欢钻研问题的JS同学,最近又迷上了对加密方法的思考.一天,他突然想出了一种他认为是终极的加密办法 :把需要加密的 ...

  6. BZOJ 1031: [JSOI2007]字符加密Cipher 后缀数组

    1031: [JSOI2007]字符加密Cipher Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 6014  Solved: 2503[Submit ...

  7. 【BZOJ-1031】字符加密Cipher 后缀数组

    1031: [JSOI2007]字符加密Cipher Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 5504  Solved: 2277[Submit ...

  8. bzoj 1031: [JSOI2007]字符加密Cipher 後綴數組模板題

    1031: [JSOI2007]字符加密Cipher Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 3157  Solved: 1233[Submit ...

  9. 1031: [JSOI2007]字符加密Cipher

    1031: [JSOI2007]字符加密Cipher Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 7338  Solved: 3182[Submit ...

随机推荐

  1. 环境变量 - JDK

    Linux 1. 备份并编辑配置文件 # cp /etc/profile /etc/profile.bak # vi /etc/profile 2. 设置JDK环境变量 export JAVA_HOM ...

  2. Go中的系统Signal处理

    package main import "fmt" import "os" import "os/signal" import " ...

  3. 我所认识的XPath

    实例demo 测试demo所需要xml测试数据 <?xml version="1.0" encoding="iso-8859-1"?> <bo ...

  4. webpack实践总结

    一.Loader写法及执行顺序 从webpack2起,loader的格式如下: module: { rules: [ {test: /\.css$/, use: ['style-loader','cs ...

  5. 再见NullPointerException。在Kotlin里null的处理(KAD 19)

    作者:Antonio Leiva 时间:Apr 4, 2017 原文链接:https://antonioleiva.com/nullity-kotlin/ 关于Kotlin最重要的部分之一:无效处理, ...

  6. ECharts 上传图片Example

    前端 1.为ECharts准备一个div <div id="main" style="Height:400px"></div> 2.引入 ...

  7. svm+voting

    # encoding:utf-8 import getopt from sklearn.preprocessing import MinMaxScaler import os,time from mu ...

  8. C#程序中SQL语句作为函数参数形式问题

    今天遇到一个神奇现象,目前正在写一个Demo,人事管理系统,首先肯定是初始化主页面,在初始化时,需要声明一个登陆窗体,但是当我在登陆窗体中填入登入名称和密码时直接就登陆成功了,但是发现我的status ...

  9. elasticsearch 6.2.3安装ik分词

    下载 zip文件 上传到服务器 https://github.com/medcl/elasticsearch-analysis-ik/releases unzip elasticsearch-anal ...

  10. stap用法

    sudo stap -g submit_bio.stp -D MAXACTION=100000 kern_path_locked lookup_one_len filename_create --&g ...