[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 ...
随机推荐
- 网易云terraform实践
此文已由作者王慎为授权网易云社区发布. 欢迎访问网易云社区,了解更多网易技术产品运营经验. 一.terraform介绍 随着应用上云的常态化,资源栈动态管理的需求对用户也变得更加急切.资源编排(Res ...
- 【赛后补题】(HDU6228) Tree {2017-ACM/ICPC Shenyang Onsite}
这条题目当时卡了我们半天,于是成功打铁--今天回来一看,mmp,贪心思想怎么这么弱智.....(怪不得场上那么多人A了 题意分析 这里是原题: Tree Time Limit: 2000/1000 M ...
- Viewer.js 图片预览插件使用
一.简介 Viewer.js 是一款强大的图片查看器. Viewer.js 有以下特点: 支持移动设备触摸事件 支持响应式 支持放大/缩小 支持旋转(类似微博的图片旋转) 支持水平/垂直翻转 支持图片 ...
- cocos2d-x 键盘和鼠标事件
出了菜单可以响应用户事件外,cocos2d中的层(Layer)也可以响应事件.层能够自动响应窗口事件,这些事件主要是键盘和鼠标事件,cocos2d中事件处理是通过Pyglet的事件处理完成的. 1.键 ...
- KVM web管理工具——WebVirtMgr(一)
WebVirtMgr 介绍 WebVirtMgr采用几乎纯Python开发,其前端是基于Python的Django,后端是基于Libvirt的Python接口,将日常kvm的管理操作变的更加的 ...
- Vue.js---组件
详情点此连接(转载) 组件的创建和注册 vue.js的组件的使用有3个步骤:创建组件构造器.注册组件和使用组件. 1. 调用Vue.extend()方法创建组件构造器. 2. 调用Vue.compon ...
- new String(str.getBytes(“gbk”),“gbk”)的用法详解
new String(str.getBytes(“gbk”),“gbk”)的用法详解 前提是str存放的是汉字 一.如果是new String(str.getBytes(“gbk”),“gbk”)时, ...
- Win7/8, convert dynamic disk volume to basic volume.
之前不小心用了Win8自带的Disk Management 来调整磁盘分区的大小,当时跳出来一个warning窗口,说如果继续操作会变成dynamic disk,然后xxxx. 我心想都是Window ...
- 关于网站转码(SiteApp转码)
1.Siteapp页面转码的意义?在百度移动搜索引擎中为更好满足用户信息需求,会同时为用户提供pc网页和mobile网页,但目前大多数PC页在移动终端中直接浏览的体验较差(交互.兼容和流量等).因此为 ...
- hdu 1853 Cyclic Tour (二分匹配KM最小权值 或 最小费用最大流)
Cyclic Tour Time Limit: 1000/1000 MS (Java/Others) Memory Limit: 32768/65535 K (Java/Others)Total ...
