Post

【精進2】ABC349-F

【精進2】ABC349-F

問題

F - Subsequence LCM
長さ$N$の正整数列$A$と正整数$M$が与えられる。$A$の空でない部分列であって、列の要素の最小公倍数が$M$になるようなものの個数を求めよ。

解法

$M$の約数でないような要素は選ばないので予め取り除いておく。また、空でないという条件を無視する($M=1$のときのみ空部分列を数えてしまうから、答えから$1$引く)。 $M = \prod_{i=1}^K p_i^{e_i}$と素因数分解する。ここで$M\le 10^{16}$であるから$K\le 12$である。部分列に対する条件は各$i \in[K]$について列の最小公倍数が$p_i^{e_i}$を約数として含むことになる。以前の問題同様、包除原理で求めていく。$i$について上記の条件を満たすような部分列全体の集合を$B_i$とすると求めるのは$|\bigcap_{i \in [K]}B_i| $である。包除原理から

\[\left| \bigcap_{i\in[K]}B_i \right| = \sum_{S \subseteq [K]}(-1)^{|S|}\left| \bigcap_{i\in S}\overline{B_i} \right|\]

と書ける。

これを \(\sum_{S \subseteq [K]}(-1)^{K-|S|}\left| \bigcap_{i\in [K]\setminus S}\overline{B_i} \right|\)と変形しておく。$\bigcap_{i\in [K]\setminus S}\overline{B_i}$は任意の$i\not\in S$について、$p_i^{e_i}\nmid \mathrm{lcm} $となるような部分列全体の集合である。 $\mathrm{cnt}[S]$を任意の$i \not\in S$について$p_i^{e_i}\nmid A_x $となる$x$の個数とすると、$\left|\bigcap_{i\in [K]\setminus S}\overline{B_i}\right|=2^{\mathrm{cnt}[S]}$となる。従って$\mathrm{cnt}$が求まれば、 $\mathcal{O}(2^K)$で計算できる。 $\mathrm{c}[S] $を$i\in S \Leftrightarrow p_i^{e_i}\mid A_x$となる$x$の個数とすると、$\mathrm{cnt}[S] = \sum_{T\subseteq S}\mathrm{c}[T] $である。$\mathrm{c}$は各$A_x$についてちょうど$1$つの$S$が対応するから計算することができ、$\mathrm{c}$から$\mathrm{cnt} $は高速ゼータ変換で$\mathcal{O}(K 2^K)$で計算できる。$K$が小さいので普通に$\mathcal{O}(3^K)$で計算してもよさそう。

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
49
50
51
52
53
54
55
56
#include "/home/y_midori/cp/library/template.hpp"
#include "math/pollard_rho.hpp"
#include <atcoder/modint>
using mint = atcoder::modint998244353;
void solve() {
    LL(n, m);
    auto fc = factor_count(m);
    ll k = fc.size();
    vl pc;
    for(auto [p, e] : fc) {
        pc.push_back(1);
        rep(i, e) pc.back() *= p;
    }
    vl c(1 << k);
    rep(i, n) {
        LL(ai);
        if(m % ai)
            continue;
        ll s = 0;
        rep(j, k) {
            if(ai % pc[j] == 0) {
                s |= 1 << j;
            }
        }
        c[s]++;
    }
    // 高速ゼータ変換 c[s] <- Σ[t⊆s]c[t]
    rep(j, k) {
        rep(i, 1 << k) {
            if((i >> j) & 1) {
                c[i] += c[i ^ (1 << j)];
            }
        }
    }
    mint ans = 0;
    rep(s, 1 << k) {
        mint tmp = mint(2).pow(c[s]);
        if((k - __popcount(s)) & 1)
            tmp *= -1;
        ans += tmp;
    }
    if(m == 1)
        ans--;
    print(ans.val());
}

int main() {
    ios::sync_with_stdio(false);
    std::cin.tie(nullptr);
    cout << std::setprecision(16);
    int t = 1;
    rep(_, t) {
        solve();
    }
}

別解

ユーザー解説通りにやってみる。 $ Z^{-1}(Z(X)\cdot Z(Y))_n= \sum _{\mathrm{lcm}(i,j)=n}X_i Y_j $ が重要。$(Z(X_1)\cdot Z(X_2)\cdot\dots \cdot Z(X_N))_n = 2^{c_n} $となる$c$をゼータ変換で求めて左辺の列をメビウス変換する。
(追記)ついでにlibrary checkerのgcd convolution,lcm convolutionを解いてライブラリに追加しておいた。

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
49
50
51
52
53
#include "/home/y_midori/cp/library/template.hpp"
#include "data_structure/hash-map-variable-length.hpp"
#include "math/pollard_rho.hpp"
#include <atcoder/modint>
using mint = atcoder::modint998244353;
void solve() {
    LL(n, m);
    vl d = divisors(m);
    HashMap<ll> c;
    rep(i, n) {
        LL(a);
        if(m % a)
            continue;
        c[a]++;
    }
    auto fc = factor_count(m);
    // ゼータ変換
    for(auto [p, e] : fc) {
        rep(i, d.size()) {
            if(d[i] % p == 0) {
                c[d[i]] += c[d[i] / p];
            }
        }
    }
    HashMap<mint> x;
    rep(i, d.size()) {
        x[d[i]] = mint(2).pow(c[d[i]]);
    }

    // メビウス変換
    for(auto [p, e] : factor_count(m)) {
        for(int i = ssize(d) - 1; i >= 0; i--) {
            if(d[i] % p == 0) {
                x[d[i]] -= x[d[i] / p];
            }
        }
    }
    auto ans = x[m];
    if(m == 1)
        ans--;
    cout << ans.val() << endl;
}

int main() {
    ios::sync_with_stdio(false);
    std::cin.tie(nullptr);
    cout << std::setprecision(16);
    int t = 1;
    rep(_, t) {
        solve();
    }
}

This post is licensed under CC BY 4.0 by the author.