[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 ...
随机推荐
- 「日常训练」 Mike and Frog (CFR305D2C)
题意与分析 (Codeforces 548C) 我开始以为是一条数学题,死活不知道怎么做,无奈看题解,才知这是一条暴力,思维江化了- - 题意大概是这样的: 两个东西的初始高度分别为h1,h2&quo ...
- P2384洛谷 最短路
题目描述 给定n个点的带权有向图,求从1到n的路径中边权之积最小的简单路径. 输入输出格式 输入格式: 第一行读入两个整数n,m,表示共n个点m条边. 接下来m行,每行三个正整数x,y,z,表示点x到 ...
- LeetCode 25 —— K 个一组翻转链表
1. 题目 2. 解答 首先,利用快慢指针确定链表的总结点数. 偶数个结点时,结点个数等于 i * 2. 奇数个结点时,结点个数等于 i * 2 + 1. 然后将链表的每 K 个结点划分为一组.循环对 ...
- Python中运算符"=="和"is"的差别分析
前言 在讲is和==这两种运算符区别之前,首先要知道Python中对象包含的三个基本要素,分别是:id(身份标识).python type()(数据类型)和value(值).is和==都是对对象进行比 ...
- C++常用STL
目录 C++ 常用STL整理 容器和配接器 list(链表) stack(栈) queue(队列) priority_queue(优先队列) set(集合) vector(向量) map&&a ...
- centos7 centos6中 更改默认的系统启动级别
centos6中更改默认的启动级别 方法: 1.vi /etc/inittab 2.找到id:x:initdefault:,我的系统是id:3:initdefault:,即默认以字符模式启动. 3.将 ...
- java有几种对象(PO,VO,DAO,BO,POJO)
首先,java有几种对象(PO,VO,DAO,BO,POJO) 一.PO:persistant object 持久对象,可以看成是与数据库中的表相映射的java对象.使用Hibernate来生成PO是 ...
- ob_flush()和flush()的区别
最近写定时任务,遇到ob_flush()和flush()混淆的问题... ob_flush/flush在手册中的描述, 都是刷新输出缓冲区, 并且还需要配套使用, 所以会导致很多人迷惑- 其实, 他们 ...
- [剑指Offer] 36.两个链表的第一个公共结点
题目描述 输入两个链表,找出它们的第一个公共结点. [思路]找出两个链表的长度,然后让长的走两个链表的长度差,然后再一起走(因为两个链表用公共的尾部). /* struct ListNode { in ...
- 【python】time 和datetime类型转换,字符串型变量转成日期型变量
s1='20120125'; 6 s2='20120216'; 7 a=time.strptime(s1,'%Y%m%d'); 8 b=time.strptime( ...
