提交记录 9731


用户 题目 状态 得分 用时 内存 语言 代码长度
memset0 1002i. 【模板题】多项式乘法 Accepted 100 30.208 ms 7240 KB C++11 3.97 KB
提交时间 评测时间
2019-07-08 15:28:07 2020-08-01 01:50:24
// =================================
//   author: memset0
//   date: 2019.07.07 23:10:06
//   website: https://memset0.cn/
// =================================
#include <bits/stdc++.h>
#define ll long long
#define rep(i, l, r) for (int (i) = (l), __lim = (r); (i) <= __lim; (i)++)
#define for_each(i, a) for (size_t i = 0, __lim = a.size(); i < __lim; ++i)
namespace ringo {

template <class T> inline void read(T &x) {
  x = 0; char c = getchar(); bool f = 0;
  while (!isdigit(c)) f ^= c == '-', c = getchar();
  while (isdigit(c)) x = x * 10 + c - '0', c = getchar();
  if (f) x = -x;
}
template <class T> inline void print(T x) {
  if (x < 0) putchar('-'), x = -x;
  if (x > 9) print(x / 10);
  putchar('0' + x % 10);
}
template <class T> inline void print(T x, char c) { print(x), putchar(c); }
template <class T> inline void print(T a, int l, int r, std::string s = "") {
  if (s != "") std::cout << s << ": ";
  for (int i = l; i <= r; i++) print(a[i], " \n"[i == r]);
}

const int N = 1e6 + 10, mod = 998244353;
int n, m;

inline void dec_up(int &a, int b) { a -= b; if (a < 0) a += mod; }
inline void inc_up(int &a, int b) { a += b; if (a >= mod) a -= mod; }
inline void mul_up(int &a, int b) { a = static_cast<ll>(a) * b % mod; }
inline int dec(int a, int b) { a -= b; return a < 0 ? a + mod : a; }
inline int inc(int a, int b) { a += b; return a >= mod ? a - mod : a; }
inline int mul(int a, int b) { return static_cast<ll>(a) * b % mod; }
inline int inv(int x) { return x < 2 ? 1 : mul(mod - mod / x, inv(mod % x)); }
inline int fpow(int a, int b) { int s = 1; for (; b; b >>= 1, mul_up(a, a)) if (b & 1) mul_up(s, a); return s; }

struct poly : std::vector<int> {
  using std::vector<int>::vector;
  inline void in() { for (auto &x : *this) read(x); }
  inline void out() const { for (auto x : *this) print(x, ' '); putchar('\n'); }
} f, g;

int w[N << 2], rev[N << 2];
int init(int len) {
  int lim = 1, k = 0; while (lim < len) lim <<= 1, ++k;
  for (int i = 0; i < lim; i++) rev[i] = (rev[i >> 1] >> 1) | ((i & 1) << (k - 1));
  static int base_len = 1;
  for (int len = base_len, wn; len < lim; base_len = len <<= 1) {
    wn = fpow(3, (mod - 1) / (len << 1)), w[len] = 1;
    for (int i = 1; i < len; i++) w[i + len] = mul(w[i + len - 1], wn);
  } return lim;
}

void dft(poly &a, int lim) {
  a.resize(lim);
  for (int i = 0; i < lim; i++) if (i < rev[i]) std::swap(a[i], a[rev[i]]);
  if (lim > 1) {
    for (int i = 0; i < lim; i += 2) {
      int x = a[i], y = a[i + 1];
      a[i] = inc(x, y), a[i + 1] = dec(x, y);
    }
  }
  if (lim > 2) {
    for (int i = 0; i < lim; i += 4) {
      int x = a[i], y = mul(w[2], a[i + 2]);
      a[i] = inc(x, y), a[i + 2] = dec(x, y);
      x = a[i + 1], y = mul(w[3], a[i + 3]);
      a[i + 1] = inc(x, y), a[i + 3] = dec(x, y);
    }
  }
  for (int len = 4; len < lim; len <<= 1)
    for (int i = 0; i < lim; i += (len << 1)) {
      int t0, t1, t2, t3;
      int *f = &a[i], *g = &a[i + len], *p = &w[len], *end = &w[len << 1];
      while (p < end) {
        t0 = mul(*p, *g), *g = dec(*f, t0), inc_up(*f, t0);
        t1 = mul(*(p + 1), *(g + 1)), *(g + 1) = dec(*(f + 1), t1), inc_up(*(f + 1), t1);
        t2 = mul(*(p + 2), *(g + 2)), *(g + 2) = dec(*(f + 2), t2), inc_up(*(f + 2), t2);
        t3 = mul(*(p + 3), *(g + 3)), *(g + 3) = dec(*(f + 3), t3), inc_up(*(f + 3), t3);
        f += 4, g += 4, p += 4;
      }
    }
}

void idft(poly &a, int lim) {
  a.resize(lim), std::reverse(&a[1], &a[lim]);
  dft(a, lim); int inv_lim = inv(lim);
  for (int i = 0; i < lim; i++) mul_up(a[i], inv_lim);
}

poly mul(const poly &f, const poly &g, int len = -1) {
  static poly a, b; a = f, b = g;
  int lim = init(f.size() + g.size() - 1);
  dft(a, lim), dft(b, lim);
  for (int i = 0; i < lim; i++) mul_up(a[i], b[i]);
  idft(a, lim); a.resize(~len ? len : f.size() + g.size() - 1);
  return a;
}

void main() {
  read(n), read(m), f.resize(n + 1), g.resize(m + 1);
  f.in(), g.in(), mul(f, g).out();
}

} signed main() {
#ifdef memset0
  freopen("1.in", "r", stdin);
#endif
  return ringo::main(), 0;
}

CompilationN/AN/ACompile OKScore: N/A

Subtask #1 Testcase #134.36 us48 KBAcceptedScore: 0

Subtask #1 Testcase #229.215 ms6 MB + 1016 KBAcceptedScore: 100

Subtask #1 Testcase #311.089 ms3 MB + 92 KBAcceptedScore: 0

Subtask #1 Testcase #411.098 ms3 MB + 80 KBAcceptedScore: 0

Subtask #1 Testcase #535.88 us48 KBAcceptedScore: 0

Subtask #1 Testcase #635.94 us48 KBAcceptedScore: 0

Subtask #1 Testcase #734.83 us48 KBAcceptedScore: 0

Subtask #1 Testcase #827.492 ms6 MB + 480 KBAcceptedScore: 0

Subtask #1 Testcase #927.442 ms6 MB + 480 KBAcceptedScore: 0

Subtask #1 Testcase #1025.733 ms5 MB + 964 KBAcceptedScore: 0

Subtask #1 Testcase #1130.208 ms7 MB + 72 KBAcceptedScore: 0

Subtask #1 Testcase #1222.518 ms5 MB + 976 KBAcceptedScore: 0

Subtask #1 Testcase #1335.05 us44 KBAcceptedScore: 0


Judge Duck Online | 评测鸭在线
Server Time: 2025-01-19 02:39:27 | Loaded in 1 ms | Server Status
个人娱乐项目,仅供学习交流使用 | 捐赠