题目大意:

  传送门

  给两个数列${B_i}、{C_i}$,长度均为$n$,且${B_i}$循环移位线性无关,即不存在一组系数${X_i}$使得对于所有的$k$均有$\sum_{i=0}^{n-1} X_i  B_{k-i \mod n} =0$。

  已知$C$是由$B$与$A$构造得到:

   (搬原图)。

  求所有合法的$A$序列。

题解:

  首先把式子稍加化简会得到:

  显然是个差分式,然后就会得到以下两种结果(以下$B_{i}$均为$B_{i\mod n}$):

    

  问题是,这两个式子都是对的吗?

  显然不是。

  我们考虑题目中说的${B_i}$为循环移位线性无关,但是$\Delta B_i=B_{i+1}-B_{i}$构成的${\Delta B_{i}}$我们是不知道它是否是线性无关的,如果是线性无关,那么它是正确的,反之,我们会知道必然存在一组${X_i}$使得若${A}$有解,则有无穷解,但是回带是错误的。

  但是二式我们如果可以求解可知$\Delta A_i=A_{i+1}-A{i}$是有唯一解的,然后考虑回带求$A_0$我们就可以得到最多两个解。

  所以本题就变成了求:

   

  设$B'_i=B_{-i}$,即:

    

  即,问题变为求出$\Delta C$的点值然后除去$B'_{*Z/n}$的点值再除去$-2$我们再由$\Delta A$的点值求出其系数即可。

  当然,我们可以知道$\Delta A$的系数小于$2e3$,所以我们防止卡精使用$NTT$。

  由于并不知道$B'_{*Z/n}$的长度,所以不能裸上,需要使用Bluestein's Algorithm。

  这个东西网上几乎没有讲解,好像毛爷爷的《再探》里面有说?

  具体就是:

  

  这样就可以使用一次卷积来求$B'$的点值了。($NTT$并不能直接拆成上面的形式,因为数论变换是没法消去下面除的那个2)

  所以将$ik$变为然后拆解即可。

代码:

 #include "bits/stdc++.h"

 typedef long long ll;

 inline int read() {
int s=,k=;char ch=getchar();
while (ch<''|ch>'') ch=='-'?k=-:,ch=getchar();
while (ch>&ch<='') s=s*+(ch^),ch=getchar();
return s*k;
} const int N=6e5+; ll mod,g,w[][N],W[][N]; inline ll Mult ( ll a,ll b ) {
return ( a*b - (ll)( (long double) a*b/mod )*mod + mod )% mod;
} inline ll powmod ( ll a, ll b ) {
ll ret=;
while (b) {
if (b&) ret=Mult ( ret, a );
b>>=,a=Mult ( a, a);
}return ret;
} inline ll gcd ( ll a,ll b ) { return b?gcd(b,a%b) : a; } int n,m; inline void Get_mod () {
for (m=; (m<=*n) ;m<<= );
ll lcm =1ll* n*m /gcd (n,m);
mod = lcm + ;
while ( mod < 1e5 ) mod += lcm;
while () {
int flag=true;
for (int i=;1ll*i*i<=mod;++i) if (mod%i==) {flag=false;break;}
if (flag) break;
mod+=lcm;
}
for (g=;;++g) {
int flag=true;
for (int i=;1ll*i*i<=mod;++i) if ((mod-)%i==){
if (powmod(g,i)==) {flag=false;break;}
if (powmod(g,(mod-)/i)==) {flag=false;break;}
}
if (flag) break;
}
} inline void Get_wn(){
ll w0=powmod(g,(mod-)/m);
w[][]=w[][]=;
int i;
for (i=;i<m;++i) w[][i]=Mult(w[][i-],w0);
for (i=;i<m;++i) w[][i]=w[][m-i];
w0=powmod(g,(mod-)/n);
W[][]=W[][]=;
for (i=;i<n;++i) W[][i]=Mult(W[][i-],w0);
for (i=;i<n;++i) W[][i]=W[][n-i];
} inline void NTT(ll *a,int n,int f) {
register int i,j,k,l,t;
for (i=j=;i^n;++i) {
if (i>j) std::swap(a[i],a[j]);
for (k=n>>;(j^=k)<k;k>>=);
}
for (i=;i<n;i<<=)
for (j=,t=n/(i<<);j<n;j+=i<<)
for (k=l=;k<i;++k , l+=t ) {
ll x=a[j+k],y=Mult(a[i+j+k],w[f][l]);
a[j+k]=x+y;
a[i+j+k]=x-y;
if (a[j+k]>=mod) a[j+k]-=mod;
if (a[i+j+k]<) a[i+j+k]+=mod;
}
if (f ) {
ll rev=powmod ( n,mod- );
for (i=;i<n;++i) a[i]=Mult(a[i],rev);
}
} ll Y[N]; inline void pre_Bluestein(int f) {
int i;
for (i=;i<*n;++i) Y[*n--i]=W[f][1ll*i*(i-)/%n];
for (i=*n;i<m;++i) Y[i]=;
NTT(Y,m,);
} inline void Bluestein(ll *a,int f){
static ll X[N];
register int i;
for (i=;i<n;++i) X[i]=Mult(a[i],W[f][ (n-1ll*i*(i-)/%n)%n ]);
for (i=n;i<m;++i) X[i]=;
NTT(X,m,);
for (i=;i<m;++i) X[i]=Mult(X[i],Y[i]);
NTT(X,m,);
for (i=;i<n;++i)
a[i]=Mult (X[*n--i],W[f][(n-1ll*i*(i-)/%n)%n ]);
if (f) {
ll rev=powmod(n,mod-);
for (i=;i<n;++i) a[i]=Mult(a[i],rev);
}
} int b[N],c[N];
ll rev_b[N],delta_c[N],delta_a[N],a[N]; int main() {
//freopen(".in","r",stdin);
n = read();
register int i;
for (i=;i<n;++i) b[i]=read();
for (i=;i<n;++i) c[i]=read();
Get_mod();
Get_wn();
for (i=;i<n;++i) rev_b[i]=b[i];
std::reverse(rev_b+,rev_b+n);
ll inv_2=powmod(mod-,mod-);
for (i=;i<n;++i) delta_c[i]=Mult ( ( c[(i+)%n]-c[i] + mod )%mod , inv_2 );
pre_Bluestein();
Bluestein(rev_b,);
Bluestein(delta_c,);
for (i=;i<n;++i) delta_a[i]=Mult ( delta_c[i] , powmod (rev_b[i],mod-) );
pre_Bluestein();
Bluestein(delta_a,);
for (i=;i<n;++i) {
ll v=(delta_a[i]<mod-delta_a[i])?delta_a[i]:delta_a[i]-mod;
if (abs(v)>) return puts(""),;
a[i]=v;
}
ll _c=-c[],_a=,_b=,sum=;
for (i=;i<n;++i) {
++_a;
_b+=*(sum-b[i]);
_c+=(sum-b[i])*(sum-b[i]);
sum+=a[i];
}
if (sum!=) {
puts("");
return ;
}
std::set<ll> ans;
if (_b*_b-*_a*_c>=){
ll s=ll(sqrt(_b*_b-*_a*_c) + 0.5);
if (s*s!=_b*_b-*_a*_c) return puts(""),;
if ((-_b+s)%(*_a)==) ans.insert((-_b+s)/(*_a));
if ((-_b-s)%(*_a)==) ans.insert((-_b-s)/(*_a));
}
std::set<ll>::iterator it;
printf("%d\n",ans.size());
for (it=ans.begin();it!=ans.end();++it) {
ll now=*it;
for (i=;i<n;++i){
printf("%lld ",now);
now+=a[i];
}
puts("");
}
}

codeforces901E

[codeforces 901E] Cyclic Cipher 循环卷积-Bluestein's Algorithm的更多相关文章

  1. codeforces 722F - Cyclic Cipher

    题目链接:http://codeforces.com/problemset/problem/722/F ------------------------------------------------ ...

  2. Bluestein's Algorithm

    网上很少有人提到,写的也很简单,事实上就是很简单... \(Bluestein's\ Algorithm\),用以解决任意长度\(DFT\). 考虑\(DFT\)的形式:\[\begin{aligne ...

  3. CodeForces - 156C:Cipher (不错的DP)

    Sherlock Holmes found a mysterious correspondence of two VIPs and made up his mind to read it. But t ...

  4. Codeforces - 102222C - Caesar Cipher

    https://codeforc.es/gym/102222/my 好像在哪里见过这个东西?字符的左右移还是小心,注意在mod26范围内. #include<bits/stdc++.h> ...

  5. codeforces 708ALetter Cyclic Shift

    2019-05-18 09:51:19 加油,加油,fightting !!! https://www.cnblogs.com/ECJTUACM-873284962/p/6375011.html 全为 ...

  6. 再探快速傅里叶变换(FFT)学习笔记(其三)(循环卷积的Bluestein算法+分治FFT+FFT的优化+任意模数NTT)

    再探快速傅里叶变换(FFT)学习笔记(其三)(循环卷积的Bluestein算法+分治FFT+FFT的优化+任意模数NTT) 目录 再探快速傅里叶变换(FFT)学习笔记(其三)(循环卷积的Blueste ...

  7. Codeforces Round #453 (Div. 1)

    Codeforces Round #453 (Div. 1) A. Hashing Trees 题目描述:给出一棵树的高度和每一层的节点数,问是否有两棵树都满足这个条件,若有,则输出这两棵树,否则输出 ...

  8. Codeforces Round #453

    Visiting a Friend Solution Coloring a Tree 自顶向下 Solution Hashing Trees 连续2层节点数都超过1时能异构 Solution GCD ...

  9. 【Codeforces 1083C】Max Mex(线段树 & LCA)

    Description 给定一颗 \(n\) 个顶点的树,顶点 \(i\) 有点权 \(p_i\).其中 \(p_1,p_2,\cdots, p_n\) 为一个 \(0\sim (n-1)\) 的一个 ...

随机推荐

  1. PM2 Quick Start

    PM2教程 @(Node)[负载均衡|进程管理器] [TOC] PM2简介 PM2 是一个带有负载均衡功能的Node应用的进程管理器. 当你要把你的独立代码利用全部的服务器上的所有CPU,并保证进程永 ...

  2. javascript数组特性

    数组是一段线性分配的内存, 它通过整数计算偏移并访问其中的元素. 数组是一种性能出色的数据结构. 1.数组字面量 数组字面量提供了一种非常方便地创建新数组的表示法. 多个用逗号分隔的值的表达式. 数组 ...

  3. left join 原理分析

    left join 原理分析 [转贴 2006-11-15 16:19:50]     字号:大 中 小 案例分析 user表:  id   | name --------- 1   | libk   ...

  4. NewLife.Net——开始网络编程

    网络编程的重要性就不说了,先上源码:https://github.com/nnhy/NewLife.Net.Tests 一个服务端,就是监听一些端口,接收客户端连接和数据,进行处理,然后响应. /// ...

  5. vfd折腾(二)

    这篇是前期程序部分,主要讲驱动pt6311的程序 电路见上一篇博文 #ifndef PT6311_H #define PT6311_H #include "sys.h" #incl ...

  6. Mysql 快速指南

    Mysql 快速指南 本文的示例在 Mysql 5.7 下都可以测试通过. 知识点 概念 数据库(database):保存有组织的数据的容器(通常是一个文件或一组文件). 数据表(table):某种特 ...

  7. windows下用C++修改本机IP地址

    两种方法 第一种.使用DOS命令(即时生效) 第二种.修改注册表(重启生效) 1.打开SOFTWARE\Microsoft\Windows NT\CurrentVersion\NetworkCards ...

  8. node七-required、缓存

    学会查API,远比会几个API更重要. 核心模块意义 -如果只是在服务器运行javascript代码,并没有多大意义,因为无法实现任何功能>读写文件.访问网络 -Node的用处在于它本身还提供可 ...

  9. php获取指定目录下的所有文件列表

    在我们实际的开发需求中,经常用到操作文件,今天就讲一下关于获取指定目录下的所有文件的几种常用方法: 1.scandir()函数 scandir() 函数返回指定目录中的文件和目录的数组. scandi ...

  10. Binary Search 的递归与迭代实现及STL中的搜索相关内容

    与排序算法不同,搜索算法是比较统一的,常用的搜索除hash外仅有两种,包括不需要排序的线性搜索和需要排序的binary search. 首先介绍一下binary search,其原理很直接,不断地选取 ...