zoukankan      html  css  js  c++  java
  • Catcats Contest 系列

    NOI 赛前抱 3 个月的佛脚(ku

    完成度:5/17

    Catcats Contest #1

    字符串 (count)

    神必转化,壬都傻了

    假设一个串有 (i)(1)(n-i)(0),则定义 (0) 的权值是 (i)(1) 的权值是 (i-n),一个字符串的权值是其所有字符权值之和。

    所有长为 (l) 的循环子串的权值的平均值为 (0),且两个相同长度的字符串相差不超过 (1) 当且仅当权值相差不超过 (n),且相同长度的字符串权值(mod n) 的值都相同。

    所以,一个字符串是好的当且仅当所有子串的权值 (in(-n,n)),也就是说 (exist kin[0,n)) 使得所有前缀和 (in(k-n,k])

    从大到小枚举 (k),此时字符串是唯一的,维护出这个字符串然后顺便计算出不符合询问串限制的位置数量,如果为 (0) 说明满足条件。

    时间复杂度 (O(n^2)),而且还可以得到答案也是 (le n^2) 的。注意不要算重了。

    #include<bits/stdc++.h>
    #define PB emplace_back
    using namespace std;
    const int N = 1030;
    int T, n, ans, pre[N], cnt; bool num[N]; char str[N];
    vector<int> vec[N];
    void solve(){
        scanf("%s", str+1); n = strlen(str+1); ans = 0;
        for(int i = 0;i <= n;++ i){
            cnt = pre[0] = 0; vec[0].PB(0);
            for(int j = 1;j <= n;++ j){
                if(pre[j-1] + i < n){
                    pre[j] = pre[j-1] + i;
                    num[j] = false;
                    cnt += str[j] == '1';
                } else {
                    pre[j] = pre[j-1] + i - n;
                    num[j] = true;
                    cnt += str[j] == '0';
                } if(pre[j] >= 0) vec[pre[j]].PB(j);
            } for(int j = n-1;;-- j){
                ans += !vec[j].empty() && !cnt;
                if(!j) break;
                for(int k : vec[j]){
                    cnt += (str[k] == '0') + (str[k+1] == '1') - (str[k] == '1') - (str[k+1] == '0');
                    swap(num[k], num[k+1]);
                }
            } for(int j = 0;j < n;++ j) vec[j].resize(0);
        } printf("%d
    ", ans);
    } int main(){scanf("%d", &T); while(T --) solve();}
    

    Catcats Contest #2

    己酸集合 (geo)

    每个询问分别做,对左右边界跑 two-pointer。时间复杂度 (O(nq))

    构造题 (buhui)

    IOI2006 的加强版,有点东西(/kel

    首先发现一个性质:如果凸包上 (S,T) 不是连续段,则一定无解。构造证明其他情况都有解。

    主要思路是把凸包范围做一个三角剖分,然后每个三角形内部分别做。

    不知为何考虑递归构造,设 ( exttt{Solve(A,B,C)}) 表示 (A,B,C) 是逆时针方向,其中 (A)(B,C) 分别属于不同集合,(BC) 有连边,你需要将 (Delta ABC) 内部的所有与 (A) 所属集合相同的点与 (A) 连通,与 (B) 所属集合相同的点与 (B)(C) 中一个连通。

    若内部点全都与 (B,C) 所属集合相同,就可以随便分治了,因为全部相同所以只要保证是一棵生成树即可。

    否则任意选择一个与 (A) 所属集合相同的点 (X),与 (A) 连边,将该三角形划分为三个小三角形,变成了子问题 ( exttt{Solve(X,B,C),Solve(C,A,X),Solve(B,X,A)}),递归做。

    至于三角剖分,分类讨论一下:

    • 若凸包上的点所属集合全相同,那么就在内部找一个不同的,以其为中心做三角剖分。
    • 若凸包上的点所属集合不同,则一定是一段 (S) 还有一段 (T),两边可以分别做,具体实现见代码(雾

    在随机数据下的时间复杂度大概是 (O(nlog n)),最坏是 (O(n^2))

    #include<bits/stdc++.h>
    #define PB emplace_back
    #define fi first
    #define se second
    using namespace std;
    typedef long long LL;
    typedef vector<int> VI;
    typedef pair<int, int> pii;
    const int N = 3003;
    template<typename T>
    void read(T &x){
        int ch = getchar(); x = 0;
        for(;ch < '0' || ch > '9';ch = getchar());
        for(;ch >= '0' && ch <= '9';ch = getchar()) x = x * 10 + ch - '0';
    } int n, m;
    struct Point {
        int x, y;
        Point(int _x = 0, int _y = 0): x(_x), y(_y){}
        bool operator < (const Point &o) const {if(x != o.x) return x < o.x; return y < o.y;}
        LL operator * (const Point &o) const {return (LL)x * o.y - (LL)y * o.x;}
        Point operator - (const Point &o) const {return Point(x - o.x, y - o.y);}
        Point operator + (const Point &o) const {return Point(x + o.x, y + o.y);}
    } p[N];
    vector<pii> ans[2];
    inline bool col(int x){return x > n;}
    void adde(int x, int y){
        assert(col(x) == col(y));
        if(col(x)) ans[1].PB(x - n, y - n);
        else ans[0].PB(x, y);
    }
    bool intri(int a, int b, int c, int d){ // d in angle bac ?
        Point v1 = p[b] - p[a], v2 = p[c] - p[a], v3 = p[d] - p[a];
        if(v1 * v2 < 0) swap(v1, v2);
        return v1 * v3 > 0 && v3 * v2 > 0;
    }
    void work(int a, int b, int c, VI now){
        // printf("a = %d, b = %d, c = %d
    ", a, b, c);
        if(now.empty()) return; VI tmp;
        for(int i : now) if(col(i) == col(a)){
            adde(a, i); tmp.resize(0);
            for(int j : now) if(intri(i, a, b, j)) tmp.PB(j);
            work(b, i, a, tmp); tmp.resize(0);
            for(int j : now) if(intri(i, b, c, j)) tmp.PB(j);
            work(i, b, c, tmp); tmp.resize(0);
            for(int j : now) if(intri(i, c, a, j)) tmp.PB(j);
            work(c, a, i, tmp); return;
        } static LL dis[N];
        for(int i : now) dis[i] = labs((p[c] - p[b]) * (p[i] - p[b]));
        sort(now.begin(), now.end(), [&](int u, int v){return dis[u] < dis[v];});
        adde(b, now[0]); for(int i = 1;i < now.size();++ i) adde(now[i-1], now[i]);
    }
    int main(){
        read(n); read(m);
        for(int i = 1;i <= n+m;++ i){read(p[i].x); read(p[i].y);}
        VI id(n+m), hul; iota(id.begin(), id.end(), 1);
        sort(id.begin(), id.end(), [&](int a, int b){return p[a] < p[b];});
        for(int i : id){
            while(hul.size() > 1 && (p[i] - p[hul.back()]) * (p[i] - p[hul[hul.size()-2]]) > 0) hul.pop_back();
            hul.PB(i);
        } id.pop_back(); reverse(id.begin(), id.end()); int tmp = hul.size();
        for(int i : id){
            while(hul.size() > tmp && (p[i] - p[hul.back()]) * (p[i] - p[hul[hul.size()-2]]) > 0) hul.pop_back();
            hul.PB(i);
        } hul.pop_back();
        int pos = 1; for(;pos < hul.size() && col(hul[0]) == col(hul[pos]);++ pos);
        // printf("pos = %d
    ", pos);
        if(pos == hul.size()){
            for(int i = 1;i < hul.size();++ i) adde(hul[i-1], hul[i]);
            hul.PB(hul[0]);
            for(int i = 1;i <= n+m;++ i) if(col(i) != col(hul[0])){
                for(int j = 1;j < hul.size();++ j){
                    int a = hul[j-1], b = hul[j]; VI tmp;
                    for(int k = 1;k <= n+m;++ k) if(intri(i, a, b, k)) tmp.PB(k);
                    work(i, a, b, tmp);
                } break;
            }
        } else {
            rotate(hul.begin(), hul.begin() + pos, hul.end());
            // for(int i : hul) printf("%d ", i); putchar('
    ');
            for(pos = 1;pos < hul.size() && col(hul[0]) == col(hul[pos]);++ pos);
            for(int _ = pos;_ < hul.size();++ _) if(col(hul[0]) == col(hul[_])){puts("Impossible"); return 0;}
            for(int i = 1;i < pos;++ i){
                int a = hul[pos], b = hul[i-1], c = hul[i]; VI tmp;
                for(int k = 1;k <= n+m;++ k) if(intri(a, b, c, k)) tmp.PB(k);
                adde(b, c); work(a, b, c, tmp);
            } for(int i = pos+1;i < hul.size();++ i){
                int a = hul[0], b = hul[i-1], c = hul[i]; VI tmp;
                for(int k = 1;k <= n+m;++ k) if(intri(a, b, c, k)) tmp.PB(k);
                adde(b, c); work(a, b, c, tmp);
            }
        } assert(ans[0].size() == n-1); assert(ans[1].size() == m-1);
        for(int _ = 0;_ < 2;++ _) for(pii &u : ans[_]) printf("%d %d
    ", u.fi, u.se);
    }
    

    最长环 (route)

    人类智慧,肝败吓疯(

    #include<bits/stdc++.h>
    using namespace std;
    const int N = 303, B = 666, NIF = -1000000007;
    template<typename T>
    void read(T &x){
        int ch = getchar(); x = 0;
        for(;ch < '0' || ch > '9';ch = getchar());
        for(;ch >= '0' && ch <= '9';ch = getchar()) x = x * 10 + ch - '0';
    } template<typename T>
    bool chmax(T &a, const T &b){if(a < b) return a = b, 1; return 0;}
    mt19937 rng(chrono::steady_clock::now().time_since_epoch().count());
    int n, m, k, x[N], y[N], z[N], a[N][N], d[2][N][N], w[N][N][3], ans; bool f[N], o;
    void ins(int u, int v, int x){
        int p = 0;
        for(;p < 3;++ p){
            int t = w[u][v][p];
            if(a[u][t] + a[v][t] < a[u][x] + a[v][x]) break;
        } if(p < 3){
            for(int i = 2;i > p;-- i) w[u][v][i] = w[u][v][i-1];
            w[u][v][p] = x;
        }
    }
    void calc(int u, int v, int x, int y){
        int p = 0;
        while(w[u][v][p] == x || w[u][v][p] == y) ++ p;
        int z = w[u][v][p];
        chmax(d[o][x][y], a[x][u] + a[u][z] + a[z][v] + a[v][y]);
    }
    void solve(int k){
        for(int i = 0;i <= n;++ i)
            for(int j = 0;j <= n;++ j) d[o][i][j] = NIF;
        switch(k){
        case 1: for(int i = 1;i <= n;++ i) if(f[i] == o) d[o][i][i] = 0; break;
        case 2:
            for(int i = 1;i <= m;++ i) if(f[x[i]] == o && f[y[i]] == o)
                d[o][x[i]][y[i]] = d[o][y[i]][x[i]] = z[i]; break;
        case 3:
            for(int i = 1;i <= m;++ i) if(f[x[i]] == o && f[y[i]] == o)
                for(int j = 1;j <= m;++ j) if(i != j && f[x[j]] == o && f[y[j]] == o){
                    if(x[i] == x[j]) chmax(d[o][y[i]][y[j]], z[i] + z[j]);
                    else if(y[i] == x[j]) chmax(d[o][x[i]][y[j]], z[i] + z[j]);
                    else if(x[i] == y[j]) chmax(d[o][y[i]][x[j]], z[i] + z[j]);
                    else if(y[i] == y[j]) chmax(d[o][x[i]][x[j]], z[i] + z[j]);
                } break;
        case 4:
            for(int i = 1;i <= m;++ i) if(f[x[i]] == o && f[y[i]] == o)
                for(int j = 1;j <= m;++ j) if(i != j && f[x[j]] == o && f[y[j]] == o && x[i] != x[j] && x[i] != y[j] && y[i] != x[j] && y[i] != y[j]){
                    chmax(d[o][x[i]][x[j]], z[i] + z[j] + a[y[i]][y[j]]);
                    chmax(d[o][x[i]][y[j]], z[i] + z[j] + a[y[i]][x[j]]);
                    chmax(d[o][y[i]][x[j]], z[i] + z[j] + a[x[i]][y[j]]);
                    chmax(d[o][y[i]][y[j]], z[i] + z[j] + a[x[i]][x[j]]);
                } break;
        case 5:
            for(int i = 1;i <= n;++ i) if(f[i] == o)
                for(int j = 1;j <= n;++ j) if(f[j] == o){
                    w[i][j][0] = w[i][j][1] = w[i][j][2] = 0;
                }
            for(int i = 1;i <= m;++ i) if(f[x[i]] == o && f[y[i]] == o)
                for(int j = 1;j <= m;++ j) if(i != j && f[x[j]] == o && f[y[j]] == o){
                    if(x[i] == x[j]) ins(y[i], y[j], x[i]);
                    else if(y[i] == x[j]) ins(x[i], y[j], y[i]);
                    else if(x[i] == y[j]) ins(y[i], x[j], x[i]);
                    else if(y[i] == y[j]) ins(x[i], x[j], y[i]);
                }
            for(int i = 1;i <= m;++ i) if(f[x[i]] == o && f[y[i]] == o)
                for(int j = 1;j <= m;++ j) if(i != j && f[x[j]] == o && f[y[j]] == o && x[i] != x[j] && x[i] != y[j] && y[i] != x[j] && y[i] != y[j]){
                    calc(x[i], x[j], y[i], y[j]);
                    calc(y[i], x[j], x[i], y[j]);
                    calc(x[i], y[j], y[i], x[j]);
                    calc(y[i], y[j], x[i], x[j]);
                }
        }
    }
    int main(){
        read(n); read(m); read(k);
        for(int i = 0;i <= n;++ i)
            for(int j = 0;j <= n;++ j) a[i][j] = NIF;
        for(int i = 1;i <= m;++ i){
            read(x[i]); read(y[i]); read(z[i]);
            a[x[i]][y[i]] = a[y[i]][x[i]] = z[i];
        } for(int _ = 0;_ < B;++ _){
            for(int i = 1;i <= n;++ i) f[i] = rng() & 1;
            if(k == 6){o = 0; solve(2); o = 1; solve(4);}
            else {o = 0; solve(k>>1); o = 1; solve(k+1>>1);}
            for(int i = 1;i <= m;++ i) if(f[x[i]] && !f[y[i]]) swap(x[i], y[i]);
            for(int i = 1;i <= m;++ i) if(f[x[i]] != f[y[i]])
                for(int j = 1;j <= m;++ j) if(i != j && f[x[j]] != f[y[j]])
                    chmax(ans, d[0][x[i]][x[j]] + d[1][y[i]][y[j]] + z[i] + z[j]);
        } if(ans > 0) printf("%d
    ", ans); else puts("impossible");
    }
    

    Catcats Contest #3

    计数题 (count)

    壬被玩傻了

    首先 (n,m) 都为奇数时答案是 (1)

    其次 (n) 为偶数,(m) 为奇数时答案是 ((n/2+1)^{(m-1)/2}),这事因为 (1) 所在列已经被确定,每个要填的列都有 (n/2+1) 种独立的方式。

    然后就是 (n,m) 都为偶数的情况。若将整个矩阵划分为 (2 imes 2) 的 block,则每块都恰有一个 (1),且每行、每列填的 (1) 的位置形如 (010101|101010),我们按列做 dp,将这个分界线的位置记录下来,并且将第 (i) 个 block 内的 (1) 是否靠右记录下来。要做的就是避免下面这种情况:

    .x..	..x.
    ...x	x...
    x...	...x
    ..x.	.x..
    

    转移是一个 SOS 形式,时间复杂度 (O(n^3m2^n)),其中这里的 (n,m) 是原来的一半。

    #include<bits/stdc++.h>
    using namespace std;
    typedef long long LL;
    int n, m, lim, mod, dp[2][13][4096], a[4096], ans; bool cr;
    int ksm(int a, int b){
        int res = 1;
        for(;b;b >>= 1, a = (LL)a * a % mod)
            if(b & 1) res = (LL)res * a % mod;
        return res;
    } void qmo(int &x){x += x >> 31 & mod;}
    void FMT(int *a){
        for(int mid = 1;mid < lim;mid <<= 1)
            for(int i = 0;i < lim;i += mid<<1)
                for(int j = i;j < i+mid;++ j)
                    qmo(a[j] += a[j|mid] - mod);
    }
    int main(){
        scanf("%d%d%d", &n, &m, &mod);
        if((n & 1) && (m & 1)){puts("1"); return 0;}
        if(n & 1) swap(n, m); if(m & 1){printf("%d
    ", ksm((n>>1)+1, m>>1)); return 0;}
        n>>=1; m>>=1; lim = 1<<n;
        for(int i = 0;i <= n;++ i)
            for(int j = 0;j < lim;++ j)
                dp[0][i][j] = 1;
        for(int _ = 1;_ < m;++ _, cr ^= 1){
            memset(dp[!cr], 0, sizeof dp[!cr]);
            for(int i = 0;i <= n;++ i){
                memcpy(a, dp[cr][i], lim<<2); FMT(a);
                for(int j = 0;j <= n;++ j)
                    for(int S = 0;S < lim;++ S){
                        int T = S;
                        for(int k = i+1;k < j;++ k)
                            if((S>>k-1&1) && !(S>>k&1)) T |= 1<<k;
                        for(int k = j;k < i-1;++ k)
                            if(!(S>>k&1) && (S>>k+1&1)) T |= 1<<k;
                        qmo(dp[!cr][j][S] += a[T] - mod);
                    }
            }
        }
        for(int i = 0;i <= n;++ i)
            for(int j = 0;j < lim;++ j)
                qmo(ans += dp[cr][i][j] - mod);
        printf("%d
    ", ans);
    }
    

    Catcats Contest #4

    平方过百万 (million)

    有一个我不会的转化:对于第 (2) 种操作,在 ((u_p,v_p)) 之间加一条权值为操作编号的边,则编号为 (T) 的询问 ( exttt{1 u}) 即为求从 (u) 开始走权值递增且 (le T) 的边,能到达的节点个数。

    可以使用点分治,假设当前处理的分治中心是 (r),对每个操作 (2),求出从它到达分治中心的最早时间,和分治中心能到达它所出发的最晚时间。这大概是一个二维偏序,用排序 + 树状数组即可,时间复杂度 (O((n+m)log n))有点烦,自闭了

    Catcats Contest #5

    构造题 (buhui)

    stm 域论,等拿到数学女孩第 5 册了就学(

    有限域相关基础:( ext{GF}(p^n)) 可以表示为系数模 (p),整体模 (n) 次本原多项式 (F) 的意义下定义四则运算,由不超过 (n-1) 次的多项式构成的代数结构。

    可以证明这东西是域,也可以证明它存在且唯一。其特征为 (p),所以 ((x+y)^p=x^p+y^p)

    那至于这题怎么做:取最小的 (w) 使得 (2^w>|S|),计算出 ( ext{GF}(2^w)),构造 (a_i=icdot 2^w+f(i)),其中 (f(i))( ext{GF}(2^w)) 意义下的立方。

    至于正确性,考虑反证法:假设存在 ((i_1,j_1) e (i_2,j_2)) 使得 (a_{i_1}oplus a_{j_1}=a_{i_2}oplus a_{j_2}),则 (i_1,j_1,i_2,j_2) 必须两两不同,且在 ( ext{GF}(2^w)) 意义下 (i_1+j_1=i_2+j_2 e 0),且 (i_1^3+j_1^3=i_2^3+j_2^3),得到 (i_1^2+i_1j_1+j_1^2=i_2^2+i_2j_2+j_2^2),所以 (i_1j_1=i_2j_2)

    (s=i_1+j_1,p=i_1j_1),得到 (x(s-x)=p) 有至少 (4) 个根,而其只有 (2) 次,矛盾。

    因此 (a_ioplus a_j) 两两不同,得证。本原多项式可以事先预处理,具体实现上 mod 2 运算可以使用异或运算。

    这个构造的背景在于一个经典问题:如何在 ([0,p)^2) 中构造 (p) 个整点使得不存在三点共线,其中 (p) 是质数。一种构造是取 (x_i=i,y_i=i^2mod p),可以得到 (frac{y_j-y_i}{x_j-x_i}=frac{y_k-y_j}{x_k-x_j}Rightarrow x_i+x_j=x_j+x_kRightarrow x_i=x_k)

    #include<bits/stdc++.h>
    using namespace std;
    const int _[] = {2,7,11,19,37,67,131,283,515,1033,2053,4105};
    int n, mod, lim;
    int red(int x){
        int p = mod; while(p <= x) p <<= 1;
        for(;p >= mod;p >>= 1) if((x ^ p) < x) x ^= p;
        return x;
    } int mul(int x, int y){
        int r = 0;
        for(;x;x >>= 1, y <<= 1) if(x & 1) r ^= y;
        return red(r);
    }
    int main(){
        scanf("%d", &n); mod = _[n-1]; lim = 1<<n;
        for(int i = 0;i < lim;++ i) printf("%d ", i<<n|mul(mul(i,i),i));
    }
    
  • 相关阅读:
    mysql 7.5.8 服务无法启动 服务没有报告任何错误
    Ubuntu 16.04 php卸载
    函数式编程(3)-匿名函数
    函数式编程(2)-返回函数
    函数式编程(1)-高阶变成(3)-sorted
    函数式编程(1)-高阶变成(2)-filter
    函数式编程(1)-高阶变成(1)-map/reduce
    高级特性(4)-生成器
    高级特性(3)-列表生成式
    高级特性(2)-迭代
  • 原文地址:https://www.cnblogs.com/AThousandMoons/p/14678047.html
Copyright © 2011-2022 走看看