[BZOJ1031][JSOI2007]字符加密Cipher 解题报告
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 解题报告的更多相关文章
- [BZOJ1031] [JSOI2007] 字符加密Cipher (后缀数组)
Description 喜欢钻研问题的JS同学,最近又迷上了对加密方法的思考.一天,他突然想出了一种他认为是终极的加密办法 :把需要加密的信息排成一圈,显然,它们有很多种不同的读法.例如下图,可以读作 ...
- [bzoj1031][JSOI2007]字符加密Cipher——后缀数组
Brief Description 给定一个长度为n的字符串,你需要对其进行加密. 把字符串围成一个环 显然从任意一个位置开始都可以有一个长度为n的串 把产生的n个串按字典序排序,把这n个串的最后一个 ...
- BZOJ1031: [JSOI2007]字符加密Cipher
传送门 后缀数组模板题 //BZOJ 1031 //by Cydiater //2016.9.21 #include <iostream> #include <cstring> ...
- [BZOJ1031][JSOI2007]字符加密Cipher(后缀数组)
传送门 算是个模板. 题目说循环,那就再复制一串拼接上. 然后求后缀数组,再搞就可以. 虽然是求后缀,会在后面多一些字符串,然而题目中说的是循环一圈,但是没有影响. ——代码 #include < ...
- 【BZOJ1031】[JSOI2007]字符加密Cipher 后缀数组
[BZOJ1031][JSOI2007]字符加密Cipher Description 喜欢钻研问题的JS同学,最近又迷上了对加密方法的思考.一天,他突然想出了一种他认为是终极的加密办法 :把需要加密的 ...
- BZOJ 1031: [JSOI2007]字符加密Cipher 后缀数组
1031: [JSOI2007]字符加密Cipher Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 6014 Solved: 2503[Submit ...
- 【BZOJ-1031】字符加密Cipher 后缀数组
1031: [JSOI2007]字符加密Cipher Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 5504 Solved: 2277[Submit ...
- bzoj 1031: [JSOI2007]字符加密Cipher 後綴數組模板題
1031: [JSOI2007]字符加密Cipher Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 3157 Solved: 1233[Submit ...
- 1031: [JSOI2007]字符加密Cipher
1031: [JSOI2007]字符加密Cipher Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 7338 Solved: 3182[Submit ...
随机推荐
- 环境变量 - JDK
Linux 1. 备份并编辑配置文件 # cp /etc/profile /etc/profile.bak # vi /etc/profile 2. 设置JDK环境变量 export JAVA_HOM ...
- Go中的系统Signal处理
package main import "fmt" import "os" import "os/signal" import " ...
- 我所认识的XPath
实例demo 测试demo所需要xml测试数据 <?xml version="1.0" encoding="iso-8859-1"?> <bo ...
- webpack实践总结
一.Loader写法及执行顺序 从webpack2起,loader的格式如下: module: { rules: [ {test: /\.css$/, use: ['style-loader','cs ...
- 再见NullPointerException。在Kotlin里null的处理(KAD 19)
作者:Antonio Leiva 时间:Apr 4, 2017 原文链接:https://antonioleiva.com/nullity-kotlin/ 关于Kotlin最重要的部分之一:无效处理, ...
- ECharts 上传图片Example
前端 1.为ECharts准备一个div <div id="main" style="Height:400px"></div> 2.引入 ...
- svm+voting
# encoding:utf-8 import getopt from sklearn.preprocessing import MinMaxScaler import os,time from mu ...
- C#程序中SQL语句作为函数参数形式问题
今天遇到一个神奇现象,目前正在写一个Demo,人事管理系统,首先肯定是初始化主页面,在初始化时,需要声明一个登陆窗体,但是当我在登陆窗体中填入登入名称和密码时直接就登陆成功了,但是发现我的status ...
- elasticsearch 6.2.3安装ik分词
下载 zip文件 上传到服务器 https://github.com/medcl/elasticsearch-analysis-ik/releases unzip elasticsearch-anal ...
- stap用法
sudo stap -g submit_bio.stp -D MAXACTION=100000 kern_path_locked lookup_one_len filename_create --&g ...
