Str 真题解(置换)
题面
对于字符串 \(s\) 定义一个变换 \(f(s)\) 表示, \(\forall 1\le k\le \lfloor|s|/2\rfloor\) ,将 \(s\) 中从后往前第 \(k\) 个字符插入从前往后第 \(k\) 个字符和第 \(k+1\) 个字符之间后得到的字符串,例如 \(\texttt{abcdef}\) 变换后得到 \(\texttt{afbecd}\) .
现在给出用 \(f(s)\) 变换过 \(k\) 次以后的字符串(即执行 \(k\) 次 \(s:=f(s)\)),求变换之前的字符串 .
(别骂我,这是原题面)
\(3\le |s|\le 10^3, 1\le k\le 10^9\) .
置换
这里没有群论
这里没有群论 .
这里的置换是狭义的,正经置换看 OI-Wiki(内含 Burnside,慎入)
置换
一个置换 \(p\) 定义为一个排列,置换相当于一个运算,将原来在位置 \(i\) 的东西变到位置 \(p_i\) .
置换的乘法(复合)
你对一个东西施加置换 \(p\) 然后施加置换 \(q\),就相当于施加 \(p\circ q\) .
显然新的置换 \(h=p\circ q\) 由下式直接表示:
\]
置换乘法的单位元
恒等置换 \(id\) 定义为 \(id_i=i\) .
显然任何置换 \(p\) 满足 \(p\circ id = id \circ p\),于是 \(id\) 就是 \(\circ\) 的单位元 .
置换乘法的结合律
结合律:
\]
为啥?
这里我们用函数表示下标,因为下标实在太多了 .
\]
Q.E.D. 是不是很显然 .
置换快速幂
有单位元,有结合律,显然可以 \(\log\) 快速幂吧 .
置换求乘法逆
显然求逆就是把置换逆过来了(类似反函数?)(从施加的角度看,真的很显然)
原来是 \(i\to p_i\),现在就是 \(p_i\to i\) .
直接模拟算就完了 .
真题解
题目要求的 \(f\) 可以看做一个置换 \(p\) .
于是 \((p^{k})^{-1}\) 就是我们要对字符串 \(s\) 施加的置换 .
直接算出来,时间复杂度 \(O(|s|\log k)\) .
一种可能的代码实现
因为求逆可以先求也可以后求甚至可以直接拿眼看出来,所以可能的代码实现有很多 .
这里是先快速幂再求逆,应该是好理解的 .
// 增加了注释
#include <iostream>
#include <algorithm>
#include <cstring>
#include <cstdio>
#include <ctime>
#include <vector>
#include <queue>
#include <cmath>
#include <map>
#include <set>
#include <bitset>
#include <cassert>
using namespace std;
const int N = 1555;
int k;
string s;
struct perm // 置换
{
perm(int x=1){n=x; for (int i=1; i<=x; i++) a[i] = i;} // 初始化为恒等置换
int operator [](const unsigned& id)const{return a[id];}
int& operator [](const unsigned& id){return a[id];} // 元素
perm operator * (const perm& rhs)const // 乘法
{
assert(n == rhs.n);
perm x(n);
for (int i=1; i<=n; i++) x[i] = rhs[a[i]];
return x;
}
perm& operator *= (const perm& rhs){return *this = *this * rhs;}
perm inv() // 求逆
{
perm ans(n);
for (int i=1; i<=n; i++) ans[a[i]] = i;
return ans;
}
inline void prt() // debug
{
for (int i=1; i<=n; i++) printf("%d ", a[i]);
puts("");
}
inline size_t size()const{return n;}
private:
int n, a[N];
};
perm qpow(perm a, int n) // 快速幂
{
perm ans(a.size());
while (n)
{
if (n&1) ans *= a;
a *= a; n >>= 1;
} return ans;
}
perm create(int n) // 生成题目说的变换 f
{
perm ans(n);
int ptr1 = 1, ptr2 = n, cc = 0;
while (ptr1 <= ptr2){ans[++cc]=ptr1; if (cc<n) ans[++cc]=ptr2; ++ptr1; --ptr2;}
return ans;
}
int main()
{
scanf("%d", &k);
cin >> s; int l = s.length(); s = "$" + s;
perm ans = qpow(create(l), k).inv();
for (int i=1; i<=l; i++) putchar(s[ans[i]]); // 对 s 施加置换
puts("");
return 0;
}
关于循环节做法
看起来这个东西有循环节?打了一发直接过了,出题人可真 sb
实际上我们可以证明这个东西有循环节且循环节是不大于 \(|s|\) 的 .
从置换的角度考虑,如果 \(i\) 能到 \(p_i\) 就连一条 \(i\to p_i\) 的有向边 .
\(n\) 个点 \(n\) 条边显然可以构成一个内向基环树森林,运算相当于在图上走一次,一直走肯定能走到环上 .
这说明任何置换 \(p\) 的方幂都是有循环节的 .
然而这题里的 \(p\) 更加特殊:
咕咕咕
Str 真题解(置换)的更多相关文章
- csps-s模拟测试60嘟嘟噜,天才绅士少女助手克里斯蒂娜,凤凰院凶真题解
题面:https://www.cnblogs.com/Juve/articles/11625190.html 嘟嘟噜: 约瑟夫问题 第一种递归的容易re,但复杂度较有保证 第二种适用与n大于m的情况 ...
- 对于JavaScript对象的prototype和__proto__的理解
一.Object和Function的关系: 刚学JavaScript的时候,看书上说JavaScript中万物皆对象,而javascript中的其他对象都是从Object继承而来,包括内置对象.瞬间觉 ...
- [Poj3128]Leonardo's Notebook
[Poj3128]Leonardo's Notebook 标签: 置换 题目链接 题意 给你一个置换\(B\),让你判断是否有一个置换\(A\)使得\(B=A^2\). 题解 置换可以写成循环的形式, ...
- python3+django2 开发易语言网络验证(中)
第四步:网络验证的逻辑开发 1.将model注册到adminx.py中 1.在apps/yanzheng目录下新建admin.py 文件,添加代码: import xadmin from xadmin ...
- 【原】Java学习笔记022 - 字符串
package cn.temptation; public class Sample01 { public static void main(String[] args) { // 字符串 // 定义 ...
- PMP备考资料和备考经验分享(基于PMP第六版)
之前有不少小伙伴私信我说,你PMP考过了,有没有报班呢,有没有自己看的资料,有没有一些经验分享,今天在这里,就统一给大家分享一下,以便大家备考和学习PMP. 先说我自己的情况,我本身是从事项目管理的, ...
- [POI2007]EGZ-Driving Exam
能到达所有路的充要条件是能到达左右两端的路 用vector反向建边对每条路左右分别求个最长不上升子序列 预处理出每条路向左向右分别需要多建多少路才能到达最左端和最右端 然后跑个\(\Theta(n)\ ...
- qsc round#2 喵哈哈村的排队(本辣鸡想七想八的,特写此博文给自己一个提醒)
该oj是qsc自己写的比赛,友情链接:http://qscoj.cn/ 喵哈哈村的排队 发布时间: 2017年2月26日 16:13 最后更新: 2017年2月26日 16:14 时间限制: ...
- Ruby Regexp类
正则表达(Regexp)类 更新:2017/06/18 改变[]集合的表格大小 80% ---> 100% 定义 正则表达: 和字符串匹配的模式(pattern)的写法 正则表达(Regexp ...
随机推荐
- vue - Vue中的ajax
只有在ajax才能找回一点点主场了,vue中的ajax一天整完,内容还行,主要是对axios的运用. 明天按理说要开始vuex了,这个从来都是只耳闻没有眼见过,明天来看看看看是个什么神奇的东西. 一. ...
- 尾递归与 memorize 优化
尾递归与 memorize 优化 本文写于 2020 年 12 月 10 日 递归 递归是一种非常常见的算法思维,在大家刚开始学编程的时候应该就会接触到. 我们可以这么理解递归: function 讲 ...
- 评估海外pop点网络质量,批量探测到整个国家运营商ip地址段时延
1 查询当地供应商所有AS号和IP地址段,如下 可以手动复制也可以爬下来,此次测试地址不多,手动复制下来再做下格式话 61.99.128.0/17 61.99.0.0/16 61.98.96.0/20 ...
- CAD图与互联网地图网页端相互叠加显示技术分析和实现
需求分析 之前相关的博文中介绍了如果在Web网页端展示CAD图形(唯杰地图云端图纸管理平台 https://vjmap.com/app/cloud),当一些CAD图纸有实际地理坐标位置时,如地形图等, ...
- linux篇-Linux逻辑卷详解总结
LVM是逻辑卷管理(Logical Volume Manager)的简称,它是建立在物理存储设备之上的一个抽象层,允许你生成逻辑存储卷,与直接使用物理存储在管理上相比,提供了更好灵活性. LVM将存储 ...
- Johnson 全源最短路
学这个是为了支持在带负权值的图上跑 Dijkstra. 为了这个我们要考虑把负的权值搞正. 那么先把我们先人已经得到的结论摆出来.我们考虑先用 SPFA 对着一个满足三角形不等式的图跑一次最短路,具体 ...
- 阻碍NB-IoT技术在智能水表发展的4个原因分析
与以往的机械水表不同,根据设备所搭载的模块,智能水表分为IC卡智能表.光电直读智能表以及无线远传智能表.随着物联网技术和工业的发展,无线远程传输智能水表开始被水务公司广泛使用. 以往的机械水表.指针式 ...
- Citus 11 for Postgres 完全开源,可从任何节点查询(Citus 官方博客)
Citus 11.0 来了! Citus 是一个 PostgreSQL 扩展,它为 PostgreSQL 添加了分布式数据库的超能力. 使用 Citus,您可以创建跨 PostgreSQL 节点集群透 ...
- python基础知识-day7(文件操作)
1.文件IO操作: 1)操作文件使用的函数是open() 2)操作文件的模式: a.r:读取文件 b.w:往文件里边写内容(先删除文件里边已有的内容) c.a:是追加(在文件基础上写入新的内容) d. ...
- CentOS中实现基于Docker部署BI数据分析
作为一个专业小白,咱啥都不懂. linux不懂,docker不懂. 但是我还想要完成领导下达的任务:在linux中安装docker后部署数据可视化工具.作为一名敬业 的打工人摆烂不可以,躺平不可以,弱 ...