2021"MINIEYE杯"中超(1)补题

2021"MINIEYE杯"中超(1)

1.1001 Mod, Or and Everything

考点:二进制

思路:分析两个例子:

n = 8, 那么有8 mod 8 == 0,8 mod 7 == 1,8 mod 6 == 2,8 mod 5 == 3,8 mod 4 == 0

之后再往下mod的话,所有值都是已经出现过的了,也就是说如果样例是8的话,那么答案那就是0|1|2|3的结果。

n = 9, 如上例分析,最后结果应该为0|1|2|3|4的结果。

综上我们可以知道如果输入为n,那么答案应该是0|1|...|(n - 1) / 2。接着想一下|运算,即有1出1,而且因为|运算的各个值是从小到大来的,所以只需考虑最大值的位数即可,如n=9时,最大|运算的数是4即(100),那么前两位都可以取到1,因为有4说明有3,2,1也参与了运算。

结论:只需考虑参加或运算的数中的最大值,统计max的位数,结果就是1 << 位数 - 1

不开long long见神仙。

#include <iostream>

using namespace std;

typedef long long LL;

int t;
LL x;

LL cal(LL u)//统计位数
{
    int cnt = 0;
    while (u)
    {
        cnt ++ ;
        u >>= 1;
    }
    
    return cnt;
}

int main()
{
    cin >> t;

    while (t -- )
    {
        cin >> x;
        cout << (1ll << cal((x - 1) / 2)) - 1 << endl;
    }
    
    return 0;
}
2.1005 Minimum spanning tree

看数据范围,不可能是prim / kruskal,就想着找规律。

边权为lcm(a, b)即a * b / gcd(a, b)。点从2开始,每个质数最佳情况是连2,边权为2 * prime,合数只需连接自己的任意一个质因子即可,边权值与合数相等。所以ans = 质数 * 2 + 合数

提交的时候记得选G++, 选C++会tle....

做法:线性筛

#include <iostream>
#include <cstring>
#include <algorithm>

using namespace std;

const int N = 10000000;

typedef long long LL;

int t;
bool st[N + 5];
int primes[N + 5], cnt;
LL n;

int main()
{
    for (int i = 2; i <= N; i ++ )//线性筛
    {
        if (!st[i]) primes[cnt ++ ] = i;
        for (int j = 0; primes[j] <= N / i; j ++ )
        {
            st[primes[j] * i] = true;
            if (i % primes[j] == 0) break;
        }
    }
    
    cin >> t;
    while (t -- )
    {
        cin >> n;
        
        LL ans = 0;
        for (LL i = 3; i <= n; i ++ )
            if (!st[i]) ans += i * 2;
            else ans += i;
        
        cout << ans << endl;
    }
    
    return 0;
}
3.1006 Xor sum

看到区间异或,就想到异或前缀和 + trie树,但是用的不熟练,比赛没做出来。

思路:把题面转化成求s[l - 1] ^ s[r] >= k的最短距离,我们枚举右边终点,左边s[0] ~ s[i - 1]都是已经插入到trie树上的,之所以s[0] = 0也在树中是因为可能存在一种情况是整个序列异或在一起,所以我们要插入0来保证这种情况取得到。对于k,我们可以看它的二进制表示,从高位开始,假如某一位k的二进制是1,那么此时异或结果一定要是1才可以,如果k本位是0,那么如果此时异或结果可以是1,那么就说明有一个合法方案,取此方案。

要用pos来储存每个点在序列中的位置。每次初始化放在插入时,否则会超时。

code
#include <bits/stdc++.h>

using namespace std;

const int N = 3200010, M = 1e5 + 10;

int t;
int son[N][2], idx;//trie树
int pos[N];//每个结点对应的序列下标
int n, k;
int s[M];//异或前缀和

void solve()
{
    scanf("%d%d", &n, &k);
    for (int i = 1; i <= n; i ++ )
    {
        scanf("%d", &s[i]);
        s[i] ^= s[i - 1];
    }
    
    int l = -1, r = n;//存答案
    int idx = 1;//trie起点
    son[1][0] = son[1][1] = 0;//初始化
    pos[1] = -1;
    
    for (int i = 0; i <= n; i ++ )//依次询问s[i]并插入s[i]
    {
        int p = 1;
        int ans = -1;
        for (int j = 31; j >= 0; j -- )
        {
            int u = (s[i] >> j) & 1;
            if ((k >> j) & 1)//k本位为1
                p = son[p][u ^ 1];//只能走这里,如果没路,就break掉
            else 
            {
                if (son[p][u ^ 1])//如果异或结果是1
                    ans = max(ans, pos[son[p][u ^ 1]]);//记录靠右的左边界
                p = son[p][u];//另一个分支记录过答案了,不需要进入那个分支
            }
            
            if (!p) break;
        }
        
        if (p) ans = max(ans, pos[p]);
        if (ans >= 0 && (i - ans < r - l)) r = i, l = ans;//如果记录到了新方案,就取更优
        
        p = 1;//插入操作
        for (int j = 31; j >= 0; j -- )
        {
            int u = (s[i] >> j) & 1;
            if (!son[p][u])//如果没有这个分支
            {
                son[p][u] = ++ idx;//插入字典树
                son[idx][0] = son[idx][1] = 0;
                pos[idx] = -1;
            }
            p = son[p][u];
            pos[p] = max(pos[p], i); 
        }
    }
    
    if (l >= 0) printf("%d %d\n", l + 1, r);
    else printf("-1\n");
}

int main()
{
    scanf("%d", &t);
    while (t -- ) solve();
    
    return 0;
}
4.1008 Maximal submatrix
前置知识:直方图中最大的矩形(知识点涉及:单调栈)

2021"MINIEYE杯"中超(1)补题

各矩形高分别为:2 1 4 5 1 3 3,那么我们从h[1]开始,假设矩形高为h[i],向左右扩展直至扩展不了,蓝色框住的部分即为从h[1]扩展的情况,绿色部分为h[2]扩展的情况,依此类推。依次扩展结束后,我们将所得矩形面积取一个max即为答案。

换句话说,对于每个h[i],我们要找到它左边第一个比它低的位置,找到右边第一个比它低的位置,然后所得到的这个范围是L,那么此次得到的矩形面积即为L * h[i]。那么每次求左边(右边)第一个比它低的位置我们可以用单调栈来维护。假设i < j,且h[i] >= h[j],那么h[k] (k > j)寻找左边第一个比它小的高度时一定访问不到h[i],因为h[j]一定是比它更优的,所以我们可以用栈来维护序列,将大于等于h[i]的栈顶弹出。

题目大意:输入n个矩形高度,求最大矩形面积。

code
#include <iostream>
#include <cstring>
#include <algorithm>

using namespace std;

const int N = 100010; 

int n;
int h[N], stk[N], le[N], ri[N];

void get_len(int k[])
{
    int tt = 0;
    for (int i = 1; i <= n; i ++ )
    {
        while (tt > 0 && h[stk[tt]] >= h[i]) tt -- ;
        k[i] = stk[tt];
        stk[++ tt] = i;
    }
}

int main()
{
    while (cin >> n, n)
    {
        for (int i = 1; i <= n; i ++ ) cin >> h[i];
        
        get_len(le);
        reverse(h + 1, h + n + 1);
        get_len(ri);
        
        //这里注意h已经翻转了,所以le[]对应的要翻转一下!
        long long ans = 0;
        for (int i = 1, j = n; i <= n; i ++ , j -- )
            ans = max(ans, 1ll * h[i] * (n - ri[i] - le[j]));
        
        cout << ans << endl;
    }
    
    return 0;
} 
本题思路:

本题让我们求不下降子序列构成的矩形的最大面积,我们可以统计出h[i, j]与h[i - 1, j]的大小关系,然后对于每一排跑一遍上面求正方图中最大矩形面积的代码然后取max即可。

code
#include <iostream>
#include <cstring>
#include <algorithm>

using namespace std;

const int N = 2e3 + 10;

int t;
int h[N][N];
int hh[N][N];
int n, m;
int stk[N], l[N], r[N];

void get_len(int q[], int ro)
{
    int tt = 0;
    for (int i = 1; i <= m; i ++ )
    {
        while (tt > 0 && hh[ro][stk[tt]] >= hh[ro][i]) tt -- ;
        q[i] = stk[tt];
        stk[++ tt] = i;
    }
}

int main()
{
    scanf("%d", &t);
    while (t -- )
    {
        scanf("%d%d", &n, &m);
        for (int i = 1; i <= n; i ++ )
            for (int j = 1; j <= m; j ++ )
                scanf("%d", &h[i][j]);
                
        for (int i = 1; i <= n; i ++ )//转化为正方图问题
            for (int j = 1; j <= m; j ++ )
                if (h[i][j] >= h[i - 1][j]) hh[i][j] = hh[i - 1][j] + 1;
                else hh[i][j] = 1;
        
        //for (int i = 1; i <= n; i ++ )
        //   for (int j = 1; j <= m; j ++ ) printf("%d%c", hh[i][j], j == m ? '\n': ' ');
        
        int ans = 0;
        for (int i = 1; i <= n; i ++ )//对每一行求一次最大面积
        {
            get_len(l, i);
            reverse(hh[i] + 1, hh[i] + m + 1);
            get_len(r, i);
            
            for (int j = 1, k = m; j <= m; j ++ , k -- )
            {
                ans = max(ans, hh[i][j] * (m - l[k] - r[j]));
                //cout << hh[i][j] * (m - l[k] - r[j]) << ' ';
            }
        }
        
        printf("%d\n", ans);
    }
    
    return 0;
}
5.1009 KD-Graph

思路:要求将点分成k组,然后组内存在len <= D的边,组间不存在len <= D的边。我们可以删除大于D的边,然后统计连通块个数即可,换个思路我们可以连接小于等于D的边,然后统计连通块个数,如果权值为w的所有边都连接完了此时连通块数目是k的话那么输出答案就是w。如果对于某个权值w,没连接它时连通块大于k个,连完之后小于k个,那么没有答案,输出-1。

code
#include <bits/stdc++.h>

using namespace std;

const int N = 1e6 + 10;

int t;
int p[N];
int n, m, k;
struct node
{
    int l, r, w;
    
    bool operator< (const node& A)const
    {
        return w < A.w;
    }
}seg[N * 5];

int find(int x)
{
    if (p[x] != x) p[x] = find(p[x]);
    
    return p[x];
}

void solve()
{
    scanf("%d%d%d", &n, &m, &k);
    for (int i = 1; i <= m; i ++ ) scanf("%d%d%d", &seg[i].l, &seg[i].r, &seg[i].w);
    
    for (int i = 1; i <= n; i ++ ) p[i] = i;
    
    sort(seg + 1, seg + m + 1);
    //for (int i = 1; i <= m; i ++ ) cout << seg[i].l << ' ' << seg[i].r << ' ' << seg[i].w << endl;
    
    int cnt = n, ans = 0;
    for (int i = 1; i <= m; i ++ )
    {
        if (seg[i].w != seg[i - 1].w)//与前面的不同权重,有答案则跳出
        {
            if (cnt == k)
            {
                printf("%d\n", ans);
                return;
            }
        }
        
        if (find(seg[i].l) == find(seg[i].r)) continue;//已经在一个连通块中
        
        cnt -- ;//否则连通块数量 --         
        p[find(seg[i].l)] = find(seg[i].r);
        ans = seg[i].w;
    }
    
    printf("%d\n", cnt == k ? ans : -1);
}

int main()
{
    scanf("%d", &t);
    while (t -- ) solve();
    
    return 0;
}
上一篇:P5341 [TJOI2019]甲苯先生和大中锋的字符串


下一篇:Treap