The 2022 ICPC Asia Regionals Online Contest (II) 2022ICPC第二场网络赛 ABEFGJKL题解

A Yet Another Remainder【费马小定理】

首先明确x是无法求解的,当x大于1e6,一共有1e4个方程组,不可能求解1e6个未知数。
因为题目将x的各个十进制位拆开,并且按照不同的步长分组,因此可能是通过计算组内的值,再将各组的值拼接起来得到 x   m o d   p x\ mod\ p x mod p的结果。
p p p都是小于100的数, m o d   p mod\ p mod p的结果必然都是100以内的值,因此推测 m o d   p mod\ p mod p的结果可能会产生某种规律,比如形成循环节出现。
到这里没有什么新的思路,因此可以翻翻书找找跟质数和模数有关的定理来辅助解题…最常见最相关的就是费马小定理了!

费马小定理: 如果p是一个质数,而整数a不是p的倍数,则有 a ( p − 1 ) ≡ 1 a^{(p-1)}≡1 ap11(mod p)。

费马小定理中的 p p p就是我们题目中的 p p p,那么 a a a可以是什么呢?题目将 x x x按照10进制进行拆分,因此 a a a可以是 1 0 i 10^i 10i,可惜发现 p = 5 p=5 p=5 1 0 i 10^i 10i是5的倍数,但是!题目非常细心地将 p = 5 p=5 p=5的情况给剔除掉了!因此可以确定这个方向应该是正确的!

接下来就是利用费马小定理给 x   m o d   p x\ mod\ p x mod p进行分组求和了。因为 1 0 p − 1 ≡ 1 ( m o d   p ) 10^{p-1}≡1(mod\ p) 10p11(mod p),因此有 1 0 2 ( p − 1 ) ≡ ( m o d   p ) ,   1 0 3 ( p − 1 ) ≡ ( m o d   p ) . . .   1 0 s ( p − 1 ) ≡ ( m o d   p ) 10^{2(p-1)}≡(mod\ p),\ 10^{3(p-1)}≡(mod\ p)...\ 10^{s(p-1)}≡(mod\ p) 102(p1)(mod p), 103(p1)(mod p)... 10s(p1)(mod p),其中s表示最后一项。因此可以将 1 0 k ( p − 1 ) ≡ 1 ( m o d   p ) 10^{k(p-1)}≡1(mod\ p) 10k(p1)1(mod p)当成一组同时求解。而在第 p − 1 p-1 p1行,给定的恰好是以 p − 1 p-1 p1为步长的十进制位的和。
因为 x = 1 0 n − 1 ∗ a 1 + 1 0 n − 2 ∗ a 2 + . . . + 1 0 n − i ∗ a i + 1 0 0 ∗ a n x=10^{n-1}*a^1+10^{n-2}*a^2+...+ 10^{n-i}*a^i+10^0*a^n x=10n1a1+10n2a2+...+10niai+100an,可知找到 1 0 s ( p − 1 ) 10^{s(p-1)} 10s(p1)对应的是a数组的第 n − s ( p − 1 ) n-s(p-1) ns(p1)项,即 b p − 1 , n − s ( p − 1 ) b_{p-1,n-s(p-1)} bp1,ns(p1)

1 0 k ( p − 1 ) + 1 ,   k = 0 , 1 , 2 , . . . , s 10^{k(p-1)+1},\ k=0,1,2,...,s 10k(p1)+1, k=0,1,2,...,s的项作为一组,将 1 0 k ( p − 1 ) + 2 ,   k = 0 , 1 , 2 , . . . , s 10^{k(p-1)+2},\ k=0,1,2,...,s 10k(p1)+2, k=0,1,2,...,s作为一组。

即将x拆解为 ( a 1 + a 1 + ( p − 1 ) + a 1 + 2 ( p − 1 ) + . . . + a 1 + s ( p − 1 ) ) ∗ 1 0 ( n − 1 ) % ( p − 1 ) ) + ( a 2 + a 2 + ( p − 1 ) + a 2 + 2 ( p − 1 ) + . . . + a 2 + s ( p − 1 ) ) ∗ 1 0 ( n − 2 ) % ( p − 1 ) ) + . . . + ( a p − 1 + a p − 1 + ( p − 1 ) + a p − 1 + 2 ( p − 1 ) + . . . + a p − 1 + s ( p − 1 ) ) ∗ 1 0 ( n − p + 1 ) % ( p − 1 ) ) (a_1+a_{1+ (p-1)}+a_{1+2(p-1)}+...+a_{1+s(p-1)})*10^{(n-1)\%(p-1)})+\\(a_2+a_{2+ (p-1)}+a_{2+2(p-1)}+...+a_{2+s(p-1)})*10^{(n-2)\%(p-1)})+\\...+\\(a_{p-1}+a_{p-1+ (p-1)}+a_{p-1+2(p-1)}+...+a_{p-1+s(p-1)})*10^{(n-p+1)\%(p-1)}) (a1+a1+(p1)+a1+2(p1)+...+a1+s(p1))10(n1)%(p1))+(a2+a2+(p1)+a2+2(p1)+...+a2+s(p1))10(n2)%(p1))+...+(ap1+ap1+(p1)+ap1+2(p1)+...+ap1+s(p1))10(np+1)%(p1))
用b表示为 b p − 1 , 1 ∗ 1 0 ( n − 1 ) % ( p − 1 ) + b p − 1 , 2 ∗ 1 0 ( n − 2 ) % ( p − 1 ) + . . . + b p − 1 , p − 1 ∗ 1 0 ( n − p + 1 ) % ( p − 1 ) b_{p-1,1}*10^{(n-1)\%(p-1)}+b_{p-1,2}*10^{(n-2)\%(p-1)}+...+b_{p-1,p-1}*10^{(n-p+1)\%(p-1)} bp1,110(n1)%(p1)+bp1,210(n2)%(p1)+...+bp1,p110(np+1)%(p1)

#include<bits/stdc++.h>
#define LL long long
using namespace std;
const int N = 110;
int b[N][N];
int main()
{
    int t; scanf("%d", &t);
    while(t --)
    {
        int n; scanf("%d", &n);
        for(int i = 1; i <= min(n, 100); i ++)
            for(int j = 1; j <= i;j ++)
                scanf("%d", &b[i][j]);
        if(n <= 100)
        {
            int q; scanf("%d", &q);
            while(q --)
            {
                int p; scanf("%d", &p);
                int ans = 0, pow = 1;
                for(int i = n; i; i --)
                {
                    ans = (ans + b[n][i] * pow) % p;
                    pow = 10 * pow % p;
                }
                printf("%d\n", ans);
            }
            continue;
        }
        int q; scanf("%d", &q);
        while(q --)
        {
            int p; scanf("%d", &p);
            int ans = 0;
            int pow10[100] = {0};
            pow10[0] = 1;
            for(int i = 1; i <= p; i ++) pow10[i] = pow10[i - 1] * 10 % p;
            for(int i = 1; i < p; i ++)
                ans = (ans + pow10[(n - i) % (p - 1)] * b[p - 1][i] % p) % p;
            printf("%d\n", ans);
        }

    }
    return 0;
}

B Non-decreasing Array【线性DP】

赛时没过题,一直往区间dp想,复杂度是 n 4 n^4 n4的,很难优化。但是dp最简单的就是从头到尾推的线性dp啊…下次思路卡住就看看书吧!dp入门!

先简单地证明一下,将一个数字删除总是会得到更好的结果
假设现在的数组是 a , b , c ( c > = b > = a ) a,b,c(c>=b>=a) a,b,c(c>=b>=a),三个数对答案的贡献是 ( c − b ) 2 + ( b − a ) 2 (c-b)^2+(b-a)^2 (cb)2+(ba)2,而将中间的数删除之后,对答案的贡献变成 ( c − a ) 2 = ( ( c − b ) + ( b − a ) ) 2 = ( c − b ) 2 + ( b − a ) 2 + 2 ( c − b ) ( b − a ) (c-a)^2=((c-b)+(b-a))^2=(c-b)^2+(b-a)^2+2(c-b)(b-a) (ca)2=((cb)+(ba))2=(cb)2+(ba)2+2(cb)(ba),因为 c > = b , b > = a c>=b,b>=a c>=b,b>=a,因此每删除一个中间数,答案就相比原来增加2*中间数与前数的差*后数与前数的差。删除一个数,答案增加如上所述,而改变一个数,其实最佳方式就是直接将其变成它的某个相邻数,因为相邻差变为0,对答案不再有贡献,其实和删除的操作是一样的。因此第 k k k次操作,就是选两个数删除掉。

先记录起始的答案,接下来只有相邻差有用,而原来的数已经没有用了,因此直接记录它们的相邻差数组 d i f dif dif即可。接下来的问题就像石子合并:一开始有 n − 1 n-1 n1堆石子,每堆石子的数量为 d i f [ i ] dif[i] dif[i],每次选择相邻的两堆石子 i , j i,j i,j,答案增加 2 ∗ d i f [ i ] ∗ d i f [ j ] 2*dif[i]*dif[j] 2dif[i]dif[j],然后将两堆石子合并起来。

f [ i ] [ j ] f[i][j] f[i][j]表示从第一堆石子到第 i i i堆石子,已经合并了 j j j堆石子获得的最大答案。
接下来思考如何更新 f [ i ] [ j ] f[i][j] f[i][j]。只需要考虑最后第 i i i堆和前面多少堆连续合并在一起即可。

  1. i i i堆不合并,直接从 f [ i − 1 ] [ j ] f[i-1][j] f[i1][j]更新过来。
  2. i i i堆只和第 i − 1 i-1 i1合并,答案为 f [ i − 2 ] [ j − 1 ] f[i-2][j-1] f[i2][j1]再加上第 i i i堆和第 i − 1 i-1 i1堆合并的新答案。
  3. i i i堆和第 i − 1 , i − 2 i-1,i-2 i1,i2堆合并,一共合并2次,答案为 f [ i − 3 ] [ j − 2 ] f[i-3][j-2] f[i3][j2]再加上后面三堆合并的新答案。
    依次类推即可。

note: 可预处理出数组 g [ i ] [ j ] g[i][j] g[i][j]表示从 i i i j j j连续段石子合并的答案。

#include<bits/stdc++.h>
#define LL long long
using namespace std;
const int N = 110;
LL g[N][N], f[N][N], dif[N], pre[N];
int a[N];
int main()
{
    int n; cin >> n;
    for(int i = 1; i <= n; i ++) cin >> a[i];
    LL ans = 0;
    for(int i = 1; i < n; i ++) dif[i] = a[i + 1] - a[i], ans += dif[i] * dif[i], pre[i] = pre[i - 1] + dif[i];
    //预处理出从i到j连续段的答案贡献
    n --;
    for(int i = 1; i < n; i ++)
        for(int j = i + 1; j <= n; j ++)
            g[i][j] = g[i][j - 1] + 2 * dif[j] * (pre[j - 1] - pre[i - 1]);
    for(int i = 1; i <= n; i ++)
        for(int j = 0; j < i; j ++)
            for(int k = 1; k <= i; k ++)
                if(j - i + k >= 0)
                    f[i][j] = max(f[i][j], f[k - 1][j - i + k] + g[k][i]);
    for(int i = 1; i <= n + 1; i ++)
    {
        printf("%lld\n", ans + f[n][min(i * 2, n - 1)]);
    }
    return 0;
}

E An Interesting Sequence【签到】

找到第一个和k互质的数,然后一直放2323…或者3232…即可。

F Infinity Tree 【签到】

观察发现,第n秒树的大小为 ( k + 1 ) n (k+1)^n (k+1)n,不难推出,节点x的父亲节点为 ( x − t − 1 ) / k + 1 (x - t - 1) / k + 1 (xt1)/k+1,其中 t t t为第一个小于x的 ( k + 1 ) n (k+1)^n (k+1)n,每次暴力找到第一个小于x和小于y的节点,一直往上跳即可。

G Good Permutation【排列组合,树形结构】

考虑最简单的情况,只有 [ 1 , n ] [1,n] [1,n]一个区间,答案就是 n ! n! n!。如果 [ 1 , n ] [1,n] [1,n]中包含 3 3 3个区间限制,恰好覆盖 [ 1 , n ] [1,n] [1,n]整个区间,长度分别为 l e n 1 , l e n 2 , l e n 3 len1,len2,len3 len1,len2,len3,答案是 3 ! ∗ l e n 1 ! ∗ l e n 2 ! ∗ l e n 3 ! 3!*len1!*len2!*len3! 3!len1!len2!len3! 3 ! 3! 3!表示将3个限制区间在 [ 1 , n ] [1,n] [1,n]之间进行全排列, l e n 1 ! len1! len1!表示被分到第1个区间的连续数字的排列方案。但是如果区间1还有2个小区间,长度分别为 l e n 11 , l e n 12 len11,len12 len11len12,这时区间1的方案数就变成 2 ! ∗ l e n 11 ! ∗ l e n 12 2!*len11!*len12 2!len11!len12,全区间的方案数也会从 3 ! ∗ l e n 1 ! ∗ l e n 2 ! ∗ l e n 3 ! 3!*len1!*len2!*len3! 3!len1!len2!len3!变成 3 ! ∗ 2 ! ∗ l e n 11 ! ∗ l e n 12 ∗ l e n 2 ! ∗ l e n 3 ! 3!*2!*len11!*len12*len2!*len3! 3!2!len11!len12len2!len3!。因此应该先将小区间更新,更新完小区间再用小区间更新大区间。

因为每个区间只有包含和并列关系,因此构成了树形结构,可以给每个区间一个编号。然后排序乱搞一通获得一棵树结构。先利用set将区间去重,然后按照左端点从小到大,右端点从大到小排序。这样排序可保证后面的区间必然不是前面的区间的父亲,因此每次记录上一次排序结束的区间节点 p r e pre pre,当前区间的节点为 c u r cur cur,则 c u r cur cur的父亲就是 c u r cur cur p r e pre pre的最近公共祖先,因为 p r e pre pre c u r cur cur区间中间已经没有隔着其他区间了。

#include<bits/stdc++.h>
#define LL long long
using namespace std;
const int N = 1e6 + 10, mod = 1e9 + 7;
struct Seg
{
    int l, r, len, fa;
    bool operator<(const Seg &x)const
    {
        if(l == x.l) return x.r < r;
        return l < x.l;
    }
    bool operator=(const Seg &x)const
    {
        return l == x.l && r == x.r;
    }
}seg[N];
set<Seg> tmp_seg;
int n, m;
int fac[N];
vector<int> son[N];
int dfs(int u)
{
    int ans = 1, len = 0;
    for(auto v : son[u])
    {
        ans = 1ll * ans * dfs(v) % mod;
        len += seg[v].len;
    }
    int left = seg[u].len - len, sz = left + son[u].size();
    ans = 1ll * fac[sz] * ans % mod;
    return ans;
}
int main()
{
    scanf("%d%d", &n, &m);
    for(int i = 1; i <= m; i ++)
    {
        int l, r; scanf("%d%d", &l, &r);
        if(l == 1 && r == n) continue;
        tmp_seg.insert({l, r, r - l + 1, 0});
    }
    int cnt = 1;
    seg[cnt].l = 1, seg[cnt].r = n, seg[cnt].len = n;
    for(auto it : tmp_seg)
    {
        int pre = cnt; cnt ++;
        seg[cnt].l = it.l, seg[cnt].r = it.r, seg[cnt].len = it.len;
        while(seg[pre].r < seg[cnt].r) pre = seg[pre].fa;
        seg[cnt].fa = pre;
        son[pre].push_back(cnt);
    }
    fac[0] = 1;
    for(int i = 1; i <= max(m, n); i ++) fac[i] = 1ll * fac[i - 1] * i % mod;
    printf("%d\n", dfs(1));
    return 0;
}

J A Game about Increasing Sequences【博弈】

可以发现,只有数列从左向右最长的一段递增序列和从右向左最长的一段递增序列有用。
考虑这两段序列的起始元素,如果两个元素大小相同,且两段序列中存在一段奇数长度的序列则先手必胜。
否则:
如果大的那个元素对应的序列长度为奇数,那么先手必胜;
不然先手必然选小的那个元素,然后轮到后手选;
我们直接模拟一遍即可得出结果。

K Black and White Painting【模拟】

因为整个网格的大小只有200*200,因此考虑标记每个网格的有贡献的边和圆弧来统计答案。用左下角的点来表示网格。对于每个网格,直线边有上下左右四条边,网格内可能存在四种圆弧。因此只需要考虑边和圆弧的所有组合情况即可。
在这里插入图片描述

  1. 某个网格里圆弧组合为02 或者13,则四条边和所有圆弧都会被覆盖失效。
    在这里插入图片描述

  2. 若整个网格都某个正方形覆盖,那么不管网格里的所有圆弧都会失效,枚举网格四条边看是否存在一条正方形边,且不会被其他的图形覆盖。比如判断网格左边是否有贡献,要检查:网格左侧有正方形边,当前网格左侧的网格没有整个网格被覆盖且没有23形状的圆弧。

  3. 存在两段圆弧(除了02组合和13组合的其他圆弧组合情况),都会有1/3的圆弧贡献

  4. 存在一段圆弧,都会有1/2的圆弧贡献。

#include<bits/stdc++.h>
using namespace std;
const int N = 210, mod = 998244353;
bool a[N][N], b[N][N][4], c[N][N][2];
int qpow(int a, int b)
{
    int ans = 1;
    while(b)
    {
        if(b & 1) ans = 1ll * ans * a % mod;
        a = 1ll * a * a % mod;
        b >>= 1;
    }return ans;
}
bool ckl(int i, int j){return (!a[i][j - 1] && !b[i][j - 1][2] && !b[i][j - 1][3]) && c[i][j][0];}
bool ckr(int i, int j){return (!a[i][j + 1] && !b[i][j + 1][0] && !b[i][j + 1][1]) && c[i][j + 1][0];}
bool cku(int i, int j){return (!a[i - 1][j] && !b[i - 1][j][0] && !b[i - 1][j][3]) && c[i - 1][j][1];}
bool ckd(int i, int j){return (!a[i + 1][j] && !b[i + 1][j][1] && !b[i + 1][j][2]) && c[i][j][1];}
int main()
{
    int n; cin >> n;
    for(int i = 1; i <= n; i ++)
    {
        int op, x, y; cin >> op >> x >> y;
        x += 102, y += 102;
        if(op == 1)
            a[x][y] = a[x + 1][y] = a[x + 1][y - 1] = a[x][y - 1] = true,
            c[x][y - 1][0] = c[x][y + 1][0] = c[x + 1][y - 1][0] = c[x + 1][y + 1][0] = true,
            c[x - 1][y - 1][1] = c[x - 1][y][1] = c[x + 1][y - 1][1] = c[x + 1][y][1] = true;
        else b[x][y][0] = b[x + 1][y][1] = b[x + 1][y - 1][2] = b[x][y - 1][3] = true;
    }
    int r1 = 0, r2 = 0;
    for(int i = 1; i < N - 1; i ++)
    {
        for(int j = 1; j < N - 1; j ++)
        {
            if((b[i][j][0] && b[i][j][2]) || (b[i][j][1] && b[i][j][3])) continue;
            else if(a[i][j]) r1 += ckl(i, j) + ckr(i, j) + cku(i, j) + ckd(i, j);
            else if((b[i][j][0] || b[i][j][2]) && (b[i][j][1] || b[i][j][3])) r2 += 2;
            else if(b[i][j][0] || b[i][j][1] || b[i][j][2] || b[i][j][3]) r2 += 3;
        }
    }
    printf("%d %d\n", r1, 1ll * r2 * qpow(6, mod - 2) % mod);
    return 0;
}

L Quadruple【简单容斥】

首先不难想到处理出ICPC的前缀和来对付区间询问,但是这样会算多一些,因为区间外的I与区间内的CPC组成的ICPC、区间外的IC和区间内的PC组成的ICPC和区间外的ICP和区间内的C组成的ICPC都会被算进去。于是我们减掉这些即可。CPC的方案数和PC的方案数计算方法也是类似的。

#include<bits/stdc++.h>
#define LL long long
using namespace std;
const int N = 2e6 + 10, mod = 998244353;
char s[N];
int n, q, u[N], v[N];
struct Int 
{
    int a;
    Int(){}
    Int(int _a){a = _a;}
    Int operator + (const Int &b)const {return Int{(a + b.a) % mod};}
    Int operator - (const Int &b)const {return Int{(a - b.a + mod) % mod};}
    Int operator * (const Int &b)const {return Int{1ll * a * b.a % mod};}
}I[N], C[N], P[N], IC[N], CP[N], PC[N], ICP[N], CPC[N], ICPC[N];
int get(int l, int r)
{
    Int sicpc = ICPC[r] - ICPC[l - 1];
    Int spc = PC[r] - PC[l - 1] - P[l - 1] * (C[r] - C[l - 1]);
    Int scpc = CPC[r] - CPC[l - 1] - CP[l - 1] * (C[r] - C[l - 1]) - C[l - 1] * spc;
    sicpc = sicpc - scpc * I[l - 1] - IC[l - 1] * spc - ICP[l - 1] * (C[r] - C[l - 1]);
    return sicpc.a;
}
int main()
{
    scanf("%d%d", &n, &q);
    scanf("%s", s + 1);
    int x, a, b, p, ans = 0;
    scanf("%d%d%d%d", &x, &a, &b, &p);
    for(int i = 1; i <= n; i ++)
    {
        
        ICPC[i] = ICPC[i - 1] + ICP[i - 1] * (s[i] == 'C');
        ICP[i] = ICP[i - 1] + IC[i - 1] * (s[i] == 'P'), CPC[i] = CPC[i - 1] + CP[i - 1] * (s[i] == 'C');
        IC[i] = IC[i - 1] +  I[i - 1] * (s[i] == 'C'), CP[i] = CP[i - 1] + C[i - 1] * (s[i] == 'P'), PC[i] = PC[i - 1] +  P[i - 1] * (s[i] == 'C');
        I[i] = I[i - 1] + (s[i] == 'I'), C[i] = C[i - 1] + (s[i] == 'C'), P[i] = P[i - 1] + (s[i] == 'P');
    }
    for(int i = 1; i <= n; i ++) u[i] = x = (1ll * a * x + b) % p, u[i] = u[i] % n + 1;
    for(int i = 1; i <= n; i ++) v[i] = x = (1ll * a * x + b) % p, v[i] = v[i] % n + 1;
    for(int i = 1; i <= n; i ++) ans = (ans + get(min(u[i], v[i]), max(u[i], v[i]))) % mod;
    printf("%d\n", ans);
    return 0;
}
  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值