首页 > 其他分享 >Codeforces Round 856 (Div2)

Codeforces Round 856 (Div2)

时间:2023-04-16 20:00:54浏览次数:53  
标签:... 856 const int res 质数 底数 Codeforces Div2

Counting Factorizations

任何一个正整数 \(m\) 都可以被唯一的分解为 \(p_1^{e_1} \cdot p_2^{e_2} \ldots p_k^{e_k}\) 的形式。将正整数 \(m\) 的唯一质数分解转化为一个长度为 \(2k\) 的 可重集合 记为 \(f(m)\)。

\[f(m)=\{p_1,e_1,p_2,e_2,p_3,e_3, \ldots ,p_k,e_k \} \]

给定正整数 \(n\) 和一个长度为 \(2n\) 的可重集 \(A\)。求出满足 \(f(m) = A\) 的正整数 \(m\) 的个数。答案对 \(998\ 244\ 353\) 取模。

\(1≤n≤2022\)

\(1≤a_i≤10^6\)

题解:欧拉筛 + 组合计数 + \(DP\)

只有质数才能作为底数,也就是说我们要在\(2n\)个数中选择\(n\)个的不同的质数作为底数,多余\(n\)个的质数和非质数作为指数,设有\(s_1\)个非质数,每个非质数数量为\(b_i\),有\(s_2\)个质数,每个质数的数量为\(c_i\) ,我们在\(s_2\)个质数中选择n个质数作为底数也就是使对应的数量减\(1\),\(c_i':=c_i-1\),根据组合计数的原理我们得到答案为

\[\sum\frac{n!}{b_1!b_2!...b_{s_1!}c_1'c_2'...c_{s_2}'} \]

实际上左半部分贡献是固定的\(\frac{n!}{b_1!b_2!...b_{s_1!}}\),我们只需求出\(\sum\frac{1}{c_1'c_2'...c_{s_2}'}\)即可,那么对于这部分贡献我们可以通过\(DP\)求出

状态表示:\(f[i][j]\):在前\(i\)个质数中选择\(j\)个作为底数的方案数 \(O(n^2)\)

状态属性:数量

状态计算:

  • 不选择第\(i\)个质数作为底数

\[f[i][j] = f[i][j] + f[i-1][j]*c[i]!,j<=i \]

  • 选择第\(i\)个质数作为底数

\[f[i][j] = f[i][j] + f[i-1][j-1]*(c[i] - 1)!,j<=i \]

状态初始:\(f[0][0]=1\)

答案呈现:从\(s_2\)个质数中选\(n\)个质数作为底数的方案数,\(f[s_2][n]\)

我们应该提前利用快速幂来预处理阶乘的逆元\(O(nlogn)\),同时利用欧拉筛提前预处理出\(1e6\)范围内的质数

#include <bits/stdc++.h>
#define Zeoy std::ios::sync_with_stdio(false), std::cin.tie(0), std::cout.tie(0)
#define debug(x) cerr << #x << '=' << x << endl
#define all(x) (x).begin(), (x).end()
#define rson id << 1 | 1
#define lson id << 1
#define int long long
#define mpk make_pair
#define endl '\n'
using namespace std;
typedef unsigned long long ULL;
typedef long long ll;
typedef pair<int, int> pii;
typedef pair<ll, ll> pll;
const int inf = 0x3f3f3f3f;
const ll INF = 0x3f3f3f3f3f3f3f3f;
const int mod = 998244353;
const double eps = 1e-9;
const int N = 1e6 + 10, M = 5e3 + 10;

int n;
int p[N], vis[N], idx;
int fact[M], inv[M], c[M];
int f[M][M];
unordered_map<int, int> uprime, prime;

int qpow(int a, int b, int p)
{
    int res = 1;
    while (b)
    {
        if (b & 1)
            res = res * a % p;
        b >>= 1;
        a = a * a % p;
    }
    return res % p;
}

void get_primes(int n)
{
    vis[1] = 1;
    for (int i = 2; i <= n; ++i)
    {
        if (!vis[i])
            p[++idx] = i;
        for (int j = 1; j <= idx && p[j] * i <= n; ++j)
        {
            vis[p[j] * i] = 1;
            if (i % p[j] == 0)
                break;
        }
    }
}

void solve()
{
    cin >> n;
    get_primes(1e6);
    int tot = 0;
    for (int i = 1; i <= 2 * n; ++i)
    {
        int x;
        cin >> x;
        if (!vis[x])
            prime[x]++;
        else
            uprime[x]++;
    }
    fact[0] = inv[0] = 1;
    for (int i = 1; i <= n; ++i)
    {
        fact[i] = fact[i - 1] * i % mod;
        inv[i] = inv[i - 1] * qpow(i, mod - 2, mod) % mod;
    }
    int ans = fact[n];
    for (auto [x, y] : uprime)
        ans = ans * inv[y] % mod;
    int cnt = 0;
    for (auto [x, y] : prime)
        c[++cnt] = y;
    f[0][0] = 1;
    for (int i = 1; i <= cnt; ++i)
    {
        for (int j = 0; j <= i; ++j)
            f[i][j] = (f[i][j] % mod + f[i - 1][j] * inv[c[i]] % mod) % mod;
        for (int j = 1; j <= i; ++j)
            f[i][j] = (f[i][j] % mod + f[i - 1][j - 1] * inv[c[i] - 1] % mod) % mod;
    }
    cout << (ans % mod * f[cnt][n] % mod) % mod << endl;
}
signed main(void)
{
    Zeoy;
    int T = 1;
    // cin >> T;
    while (T--)
    {
        solve();
    }
    return 0;
}

标签:...,856,const,int,res,质数,底数,Codeforces,Div2
From: https://www.cnblogs.com/Zeoy-kkk/p/17323921.html

相关文章

  • Codeforces Round 866 (Div. 2) A~C
     这场,非常快落!是难得对中国选手友好的时间(17:05) A观察一下,发现在连续的___中插入^就好,然后特判一下首尾,发现很像小学奥数的那个植树问题哇(注意特判一下只有一个^#include<bits/stdc++.h>usingnamespacestd;voidsolve(){strings;cin>>s;intn=s.len......
  • Codeforces Round 628 (Div. 2) A-D
    CodeforcesRound628(Div.2)A.EhAbAnDgCdvoidsolve(){intn=read();for(inti=1;i*i<=n;i++){intg=__gcd(i,n-i);if(g*g+i*(n-i)==n*g){cout<<i<<""<<n-i<<endl;bre......
  • Codeforces Round 862 (Div. 2)
    Preface补题ing这场思路挺顺的,早上上课的时候口胡了前5题下午都一发写过了然后想了30min的F1也Rush出来了,不过F2还是有点仙的做不动A.WeNeedtheZeroSB题,首先判断是否所有数的异或和等于\(0\)若不为\(0\)且\(n\)为偶数则无解,否则答案就是这个异或和本身#include<cstdio......
  • [Educational Codeforces Round 118 (Rated for Div. 2)]题解
    A题意:给定两个数,每一个数有两个属性,第一个属性是p1,第二个属性是p2.表示这个数有p2个后缀0.这个数本身等于p1后面加p2个0.问给你两个这种数,判断大小。思路:赛场上想到的:如果最终的长度不一样,可以直接根据长度判断。如果相等,就把后缀0加上直接比较大小就可以(比较字典序的大小),但......
  • Educational Codeforces Round 110 (Rated for Div. 2) C. Unstable String(状态机)
    https://codeforces.com/contest/1535/problem/C题目大意:给定一个字符串s,由10?组成:?每次都可以任意替换成0或者1问我们这个子字符串中,能够组成010101这样两两互不相等的字符串的数量最大是多少?input30?10????10??1100output8625#include<bits/stdc++.h>usin......
  • Educational Codeforces Round 120 (Rated for Div. 2)
    题目链接C核心思路这是一个很好的二分的题目,首先我们判断题目可不可二分,很显然是可以的把。因为假设我们x是可以的话,x+1...肯定也是可以的,但是x-1,x-2....这些又是不可以的。好,接下来思考二分刚开始的左右边界,左边届很好想,关键是右边界。这个其实也不难。因为我们最坏肯定是全......
  • Codeforces Round #289 Div. 2 解题报告 A.B.C.E
    A-MaximuminTable纯递推。代码如下:#include<iostream>#include<string.h>#include<math.h>#include<queue>#include<algorithm>#include<stdlib.h>#include<map>#include<set>#include<stdio.h>usingn......
  • Codeforces Round #290 (Div. 2) 解题报告 A.B.C.D.
    A-FoxAndSnake模拟。代码如下:#include<iostream>#include<string.h>#include<math.h>#include<queue>#include<algorithm>#include<stdlib.h>#include<map>#include<set>#include<stdio.h>usingnames......
  • Codeforces Round #287 (Div. 2) 解题报告 A.B.C.D.E
    这次的CF挺水的,当时B题犯了一个很SB的错误,浪费了好多时间,所以D和E也没来得及看。sad,。。A-AmrandMusic水题,从小的开始选。代码如下:#include<iostream>#include<string.h>#include<math.h>#include<queue>#include<algorithm>#include<stdlib.h>#include<map>......
  • Codeforces Round #286 (Div. 2) C题 Mr. Kitayuta, the Treasure Hunter (DFS+记忆化D
    题目地址:http://codeforces.com/contest/505/problem/C从d点开始,每个点都有三个方向,形成了一棵树,那么就从跟结点开始进行dfs查找,dp数组记录当前的点和长度,当这两个条件相同的时候,显然,后面的子树是完全相同的,于是用记忆化来优化。代码如下:#include<iostream>#include<string.h>#......