跳转至

Chirp Z 变换

与离散傅里叶变换类似,Chirp Z 变换是给出多项式 f(x)=i=0m1fixiC[x]qC{0} 求出 f(1),f(q),,f(qn1) 的一种算法,不要求 q 为单位根。也可用于数论变换。后文将介绍 Chirp Z 变换与其逆变换。

Chirp Z 变换

根据定义,Chirp Z 变换可以写作

CZTn:(f(x),q)[f(1)f(q)f(qn1)]

其中 f(x):=i=0m1fixiC[x]qC{0}

Bluestein 算法

考虑

ij=(i2)+(j2)(ij2)

其中 i,jZ,我们可以构造

G(x):=i=(m1)n1q(i2)xi,F(x):=i=0m1fiq(i2)xi.

其中 G(x)C[x,x1],且对于 i=0,,n1 我们有

[xi](G(x)F(x))=j=0m1(([xij]G(x))([xj]F(x)))=j=0m1fjq(j2)(ij2)=q(i2)f(qi)

q(i+12)=q(i2)qi(i2)=(i+12)。可以由一次多项式乘法完成求算,该算法被称为 Bluestein 算法。

模板(P6800【模板】Chirp Z-Transform
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
std::vector<uint> CZT(const std::vector<uint>& f, uint q, int n) {
  if (f.empty() || n == 0) return std::vector<uint>(n);
  const int m = f.size();
  if (q == 0) {
    std::vector<uint> res(n, f[0]);
    for (int i = 1; i < m; ++i)
      if ((res[0] += f[i]) >= MOD) res[0] -= MOD;
    return res;
  }
  // H[i] = q^(binom(i + 1, 2))
  std::vector<uint> H(std::max(m, n - 1));
  H[0] = 1;     // H[0] = q^0
  uint qi = 1;  // qi = q^i
  for (int i = 1; i < (int)H.size(); ++i) {
    qi = (ull)qi * q % MOD;
    // H[i] = q^(binom(i, 2)) * q^i
    H[i] = (ull)H[i - 1] * qi % MOD;
  }
  // F[i] = f[i] * q^(binom(i + 1, 2))
  std::vector<uint> F(m);
  for (int i = 0; i < m; ++i) F[i] = (ull)f[i] * H[i] % MOD;
  std::vector<uint> G_p(m + n - 1);
  // G[i] = q^(-binom(i, 2)), -(m - 1) ≤ i < n
  uint* const G = G_p.data() + (m - 1);
  const uint iq = InvMod(q);
  // G[-(m - 1)] = q^(-binom(-(m - 1), 2)),
  // binom(-(m - 1), 2) = (-(m - 1)) * (-(m - 1) - 1) / 2
  //                    = (m - 1) * m / 2
  G[-(m - 1)] = PowMod(iq, (ull)(m - 1) * m / 2);
  uint qmi = PowMod(q, m - 1);  // q^(-i), -(m - 1) ≤ i < n
  for (int i = -(m - 1) + 1; i < n; ++i) {
    // q^(-binom(i, 2)) = q^(-binom(i - 1, 2)) * q^(-(i - 1))
    G[i] = (ull)G[i - 1] * qmi % MOD;
    // q^(-i) = q^(-(i - 1)) * q^(-1)
    qmi = (ull)qmi * iq % MOD;
  }
  // res[i] = q^(-binom(i, 2)) * f(q^i), 0 ≤ i < n
  std::vector<uint> res = MiddleProduct(G_p, F);
  for (int i = 1; i < n; ++i) res[i] = (ull)res[i] * H[i - 1] % MOD;
  return res;
}

逆 Chirp Z 变换

逆 Chirp Z 变换可以写作

ICZTn:([f(1)f(q)f(qn1)],q)f(x)

其中 f(x)C[x]<nqC{0},并且 qiqj 对于所有 ij 成立,这是多项式插值的条件。

Bostan–Schost 算法

回顾 Lagrange 插值公式

f(x)=i=0n1(f(xi)0j<njixxjxixj)

xixj 对于所有 ij 成立。与 多项式的快速插值 中相同,我们令 M(x):=i=0n1(xxi),根据洛必达法则,有

M(xi)=limxxiM(x)xxi=0j<nji(xixj)

修正 Lagrange 插值公式 就是

f(x)=M(x)(i=0n1f(xi)/M(xi)xxi)

那么现在我们有

f(x)=M(x)(i=0n1f(qi)/M(qi)xqi)

其中 M(x)=j=0n1(xqj)。若我们设 n 为偶数,令 n=2kH(x):=j=0k1(xqj),那么

M(x)=H(x)qk2H(xqk)

这使得我们可以快速计算 M(x)。然后用 Bluestein 算法来计算 M(1),,M(qn1)。令 ci:=f(qi)/M(qi),我们有

f(x)=M(x)(i=0n1cixqi)

因为 degf(x)<n,我们只需计算 i=0n1cixqimodxn,其中 cixqiC[[x]],也就是

i=0n1cixqimodxn=i=0n1(j=0n1ciqi(j+1)xj)=j=0n1C(qj1)xj

其中 C(x)=i=0n1cixi。我们可以用 Bluestein 算法来计算 C(q1),,C(qn)

简单来说,我们分别进行下面的计算:

  1. 用减治法(decrease and conquer)计算 M(x)
  2. 用 Bluestein 算法计算 M(1),,M(qn1)
  3. 用 Bluestein 算法计算 C(q1),,C(qn)
  4. 用一次多项式乘法计算 f(x)

其中每一步的时间复杂度都等于两个次数小于等于 n 的多项式相乘的时间复杂度。

模板实现
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
std::vector<uint> InvCZT(const std::vector<uint>& f, uint q) {
  if (f.empty()) return {};
  const int n = f.size();
  if (q == 0) {
    // f[0] = f(1), f[1] = f(0)
    assert(n <= 2);
    if (n == 1) return {f[0]};                 // deg(f) < 1
    return {f[1], (f[0] + MOD - f[1]) % MOD};  // deg(f) < 2
  }
  // prod[0 ≤ i < n] (x - q^i)
  const auto DaC = [q](auto&& DaC, int n) -> std::vector<uint> {
    if (n == 1) return {MOD - 1, 1u};
    // H = prod[0 ≤ i < ⌊n/2⌋] (x - q^i)
    const std::vector<uint> H = DaC(DaC, n / 2);
    // HH = H(x / q^(⌊n/2⌋)) * q^(⌊n/2⌋^2)
    std::vector<uint> HH = H;
    const uint iqn = InvMod(PowMod(q, n / 2));
    uint qq = PowMod(q, (ull)(n / 2) * (n / 2));
    for (int i = 0; i <= n / 2; ++i)
      HH[i] = (ull)HH[i] * qq % MOD, qq = (ull)qq * iqn % MOD;
    std::vector<uint> res = Product(H, HH);
    if (n & 1) {
      res.resize(n + 1);
      const uint qnm1 = MOD - PowMod(q, n - 1);
      for (int i = n; i > 0; --i) {
        if ((res[i] += res[i - 1]) >= MOD) res[i] -= MOD;
        res[i - 1] = (ull)res[i - 1] * qnm1 % MOD;
      }
    }
    return res;
  };
  const std::vector<uint> M = DaC(DaC, n);
  // C[i] = (M'(q^i))^(-1)
  std::vector<uint> C = InvModBatch(CZT(Deriv(M), q, n));
  // C[i] = f(q^i) / M'(q^i)
  for (int i = 0; i < n; ++i) C[i] = (ull)f[i] * C[i] % MOD;
  // C(x) ← C(q^(-1)x)
  const uint iq = InvMod(q);
  uint qmi = 1;
  for (int i = 0; i < n; ++i)
    C[i] = (ull)C[i] * qmi % MOD, qmi = (ull)qmi * iq % MOD;
  C = CZT(C, iq, n);
  for (int i = 0; i < n; ++i)
    if (C[i] != 0) C[i] = MOD - C[i];
  std::vector<uint> res = Product(M, C);
  res.resize(n);
  return res;
}

参考文献

  1. Bostan, A. (2010). Fast algorithms for polynomials and matrices. JNCF 2010. Algorithms Project, INRIA.