多校联赛第三场(部分题解)

1001. Magician (hdu5316)

这个题目用到线段树来合并区间,题目有句话的意思A beautiful subsequence is a subsequence that all the adjacent pairs of elves in the sequence have a different parity of position说的是相邻的两个元素必须有不同的奇偶位置,所有一个序列有四种可能的形式,奇...奇, 奇...偶,偶...奇, 偶...偶,这四种形式只是端点的奇偶性,所以线段树的每个节点就得有四个值,分别维护这四个值就行了,举个例子来说,根节点的偶偶的最大值取决于 左孩子的偶偶 + 右孩子的奇偶, 左孩子的偶奇 + 右孩子的偶偶, 左孩子的偶偶,右孩子的偶偶, 所以根节点的偶偶就等于他们当中的最大值,同理,其他的三个也是这样,剩下的另外一种操作就是更新点了。

#include<iostream>
#include <stdio.h>
using namespace std;
typedef long long LL;
const int maxn = 100010;
const LL inf = 1LL << 61;
LL d[maxn];
struct Data{
    LL s00, s01, s10, s11;//s00表示偶偶, s01表示偶奇, s10表示奇偶,s11表示奇奇
    void init()
    {
        s00 = s01 = s10 = s11 = -inf;//初始化一定要为负无穷,因为可能有负数
    }
    LL max_ele()
    {
        return max(max(s00, s01), max(s10, s11));//找出他们当中最大的
    }
};
struct tree{
    Data data;
};
tree sum[maxn * 4];//线段树
void checkmax(LL &a, const LL b)
{
    if (a < b)
        a = b;
}
Data operator + (const Data &a, const Data &b)
{
    Data ret;
    ret.s11 = max(a.s11 + b.s01, a.s10 + b.s11); checkmax(ret.s11, a.s11); checkmax(ret.s11, b.s11);
    ret.s10 = max(a.s11 + b.s00, a.s10 + b.s10); checkmax(ret.s10, a.s10); checkmax(ret.s10, b.s10);
    ret.s00 = max(a.s00 + b.s10, a.s01 + b.s00); checkmax(ret.s00, a.s00); checkmax(ret.s00, b.s00);
    ret.s01 = max(a.s00 + b.s11, a.s01 + b.s01); checkmax(ret.s01, a.s01); checkmax(ret.s01, b.s01);
    return ret;
}
void pushup(int rt)
{
    sum[rt].data = sum[rt<<1].data + sum[rt<<1|1].data;
}
void build(int L, int R, int rt)
{
    sum[rt].data.init();
    if (L == R)
    {
        if (L & 1)
            sum[rt].data.s11 = d[L];
        else
            sum[rt].data.s00 = d[L];
        return;
    }
    int mid = (L + R) / 2;
    build(L, mid, rt<<1);
    build(mid + 1, R, rt<<1|1);
    pushup(rt);
}
void update(int pos, LL value, int L, int R, int rt)
{
    if (L == pos && R == pos)
    {
        if (pos & 1)
            sum[rt].data.s11 = value;
        else
            sum[rt].data.s00 = value;
        return;
    }
    int mid = (L + R) / 2;
    if (pos <= mid)
        update(pos, value, L, mid, rt<<1);
    else
        update(pos, value, mid + 1, R, rt<<1|1);
    pushup(rt);
}
Data query(int l, int r, int L, int R, int rt)
{
    if (l <= L && r >= R)
    {
        return sum[rt].data;
    }
    int mid = (L + R) >> 1;
    Data lson, rson;
    lson.init(); rson.init();
    if (l <= mid)
        lson = query(l, r, L, mid, rt<<1);
    if (r > mid)
        rson = query(l, r, mid + 1, R, rt<<1|1);
    return lson + rson;
}
int main()
{
    int T, n, m;
    scanf("%d", &T);
    while (T--)
    {

        scanf("%d %d", &n ,&m);
        for (int i = 1; i <= n; i++)
            scanf("%I64d", &d[i]);
        build(1, n, 1);
        int op, a, b;
        for (int i = 0; i < m; i++)
        {
            scanf("%d %d %d", &op, &a, &b);
            if (op)
                update(a, (LL)b, 1, n, 1);
            else
            {
                Data ans = query(a, b, 1, n, 1);
                printf("%I64d\n", ans.max_ele());
            }
        }
    }
    return 0;
}

1002.RGCDQ (hdu5317)

这道题F(x)表示x当中素因子的个数,由于x最大1000000,所以F(x)最多不超过8个,这个可以拿出前几个素数乘一下试试,2*3*5*7*11*13*17*19=9699690>1000000所以可以先预处理出来每个数的F(x),然后求出他的前缀和,用O(1)的复杂度去查询,求F(x)的时候注意不要除着求,要直接像筛素数一样去求。具体见代码

#include <cstdio>
#include <string.h>
const int maxn = 1000010;
int prime[maxn];//标记筛选的素数
int f[maxn];//F(x)
int S[maxn][9];//前缀和,因为一共最多可能有8个,所以开个8的数组足够了,0的位置为空
int pr[100000];//保存素数
void init()
{
    memset(prime, true, sizeof(prime));
    prime[1] = prime[0] = false;
    for (int i = 2; i * i <= 1000000; i++)//素数筛选
    {
        if (prime[i])
            for (int j = i + i; j <= 1000000; j += i)
                prime[j] = false;
    }
    int tot = 0;
    for (int i = 2; i <= 1000000; i++)
        if (prime[i])
            pr[tot++] = i;//将筛选出来的素数放到pr中
    for (int i = 0; i < tot; i++)//求F(x),这里很关键,不能放到素数筛选的过程当中去,因为有大素数
    {
        for (int j = pr[i]; j <= 1000000; j += pr[i])//所有包含pr[i]这个素因子的都要加1
            f[j]++;
    }
    for (int i = 2; i <= 1000000; i++)
    {
        for (int j = 1; j <= 8; j++)
        {
            S[i][j] = (j == f[i]) ? S[i - 1][j] + 1 : S[i - 1][j];//求出前缀和
        }
    }
}

int main()
{
    init();
    int T, L, R;
    scanf("%d",  &T);
    while (T--)
    {
        scanf("%d %d", &L, &R);
        int ans = 1;
        for (int i = 8; i > 0; i--)
        {

            if (S[R][i] - S[L - 1][i] > 1)//找出这个区间最大的来
            {
                ans = i;
                break;
            }
        }
        printf("%d\n", ans);
    }
    return 0;
}

1003.The Goddess Of The Moon (hdu5318)

给出n条链子,让你选择m个,问一共方案有多少种。因为m比较大(1e9),所以用矩阵快速幂来解。构造一个矩阵表示任意两个之间是否可以链接,如果可以链接的话就是1,不能链接的话就是0,这个题有点模糊。大致知道一点意思,但是细推的话,状态转移方程不太清楚。。。看题解上写S[i][j]表示前i个,以j结尾的方案数,估计意思就是当链接i个的时候,最后以j结尾的方案数,他就应该等于前i - 1个所有能链接j,并且j链接在后面的个数,对于每一个j,对应一列,所以,就是一个矩阵,而S[i][j]可以由S[i - 1][]来推,那么就构成了矩阵连乘的条件,所以就能用矩阵快速幂了。这个题还是不是完全透彻的理解。如果哪位大牛路过,恳请指点。

#include<iostream>
#include <cstdio>
#include <cstring>
#include <vector>
#include <algorithm>
using namespace std;
typedef long long LL;
const int maxn = 52;
const int mod = 1000000007;
int sz;
struct Mat{
    int mat[maxn][maxn];
    void init()
    {
        memset(mat, 0, sizeof(mat));
    }
};
Mat operator * (const Mat &a, const Mat &b)//矩阵乘法
{
    Mat ret;
    ret.init();
    for (int i = 0; i < sz; i++)
    {
        for (int j = 0; j < sz; j++)
        {
            ret.mat[i][j] = 0;
            for (int k = 0; k < sz; k++)
            {
                ret.mat[i][j] += 1LL * a.mat[i][k] * b.mat[k][j] % mod;
                ret.mat[i][j] %= mod;
            }

        }
    }
    return ret;
}
Mat operator ^ (Mat a, int b)//矩阵快速幂
{
    Mat ans;
    for (int i = 0; i < sz; i++)
        for (int j = 0; j < sz; j++)
            ans.mat[i][j] = (i == j ? 1 : 0);
    while (b)
    {
        if (b & 1)
            ans = ans * a;
        a = a * a;
        b >>= 1;
    }
    return ans;
}
bool connect(int a, int b)//判断a链和b链是否能连接
{
    int dig1[maxn], dig2[maxn];
    int tot1 = 0, tot2 = 0;
    for (; a; a /= 10)
        dig1[tot1++] = a % 10;
    for (; b; b /= 10)
        dig2[tot2++] = b % 10;
    int ans = 1;
    for (int len = 1; len <= min(tot1, tot2); len++)
    {
        bool flag = false;
        for (int i = len - 1; i >= 0; i--)
        {
            if (dig1[i] != dig2[tot2 - (len - i)])
            {
                flag = true;
                break;
            }
        }
        if (!flag)
            ans = len;
    }
    if (ans > 1)
        return true;
    return false;
}
int solve(vector <int> num, int n)
{
    vector<int> tmp = num;
    sort(tmp.begin(), tmp.end());
    tmp.erase(unique(tmp.begin(), tmp.end()), tmp.end());//去重
    int m = (int)tmp.size();
    Mat A, B;
    sz = m;
    for (int i = 0; i < m; i++)
    {
        for (int j = 0; j < m; j++)
            B.mat[i][j] = connect(tmp[i], tmp[j]);//构造转移矩阵
    }
    A.init();
    for (int i = 0; i < m; i++)
        A.mat[0][i] = 1; //初始第一行全为1
    B = B ^ (n - 1);
    A = A * B;
    int ans = 0;
    for (int i = 0; i < m; i++)
    {
        ans += A.mat[0][i];
        ans %= mod;
    }
    return ans;
}
int main()
{
    int T, n, m;
    scanf("%d", &T);
    while (T--)
    {
        scanf("%d%d", &n, &m);
        vector<int> v(n);
        int tmp;
        for (int i = 0; i < n; i++)
        {
            scanf("%d", &v[i]);
        }
        printf("%d\n", solve(v, m));
    }
    return 0;
}

1004.Painter (hdu5319)

这个题属于模拟题,红色只能正向斜着刷(\),蓝色只能反向斜着刷(/),被正向反向刷了两次的点是绿色,给出最终状态求出最少经过多少步能到这种状态。直接枚举每一个点就行。

#include <cstdio>
#include <cstring>
const int maxn = 60;
char a[maxn][maxn];
int mp[maxn][maxn];
int m;
int ans;
int n;
const int dir[2][2] = {1, 1, 1, -1};
void work(int x, int y, int color)
{
    mp[x][y] -= color;
    while (true)
    {
        x += dir[color - 1][0];
        y += dir[color - 1][1];
        if (x >= 0 && y >= 0 && x < n && y < m && (mp[x][y] == color || mp[x][y] == 3))
            mp[x][y] -= color;
        else
            break;
    }
}
int main()
{
    int T;
    scanf("%d", &T);
    while (T--)
    {
        scanf("%d", &n);
        for (int i = 0; i < n; i++)
        {
            scanf("%s", a[i]);
            m = strlen(a[i]);
                for (int j = 0; j < m; j++)
                {
                    if (a[i][j] == ‘R‘)//转化成数字便于计算
                        mp[i][j] = 1;
                    else if (a[i][j] == ‘B‘)
                        mp[i][j] = 2;
                    else if (a[i][j] == ‘G‘)
                        mp[i][j] = 3;
                    else
                        mp[i][j] = 0;
                }
        }
        ans = 0;
        for (int i = 0; i < n; i++)
        {
            for (int j = 0; j < m; j++)
            {
                if (mp[i][j] == 0) continue;
                if (mp[i][j] == 1)
                    work(i, j, 1);
                else if (mp[i][j] == 2)
                    work(i, j, 2);
                else
                {
                    work(i, j, 1);
                    work(i, j, 2);
                    ans++;
                }
                ans++;
            }
        }
        printf("%d\n", ans);

    }
    return 0;
}

1008.Solve this interesting problem (hdu5323)

此题给出线段树的区间定义,给出一个节点的左区间端点和右区间端点,求最小n,其中n为根节点的右端点。

从当前节点(L, R)往他的父节点走的话有四种路线,分别是:当他为左孩子的时候父节点的区间端点(L, R + (R - L + 1))或者(L, R + (R - L + 1) - 1),当他为右孩子的时候父节点的区间为(L - (R - L + 1), R)或者(L - (R - L + 1) - 1, R),因为线段树节点的左右孩子区间长度要不 相等,要不 左孩子比右孩子大1, 所以根据这个结论所以就可以得出上面四种方式,其中(R - L + 1)为区间长度,题目中还给了一个关键的就是L / (R - L + 1) <= 2015,这个就说明 L <= 2015 * len, 其中len为区间长度,线段树中len越小,也就是左右端点靠的越近,他的位置越深,就比如叶子节点的len为0,所以他在最下面,这就说明了一个问题就是当这个树取最深的顶点的时候,那么len取1,所以L<=2015, 并且从2015开始往上找到0的时候最多22层,当是左孩子的时候11层,右孩子11层。所以搜索这四种路径。

#include<iostream>
#include <cstdio>

using namespace std;
typedef long long LL;
const LL inf = 1LL << 60;
LL n;
void search(LL L, LL R)
{
    if (L < 0 || L > R)
        return;
    if (L == 0)
    {
        n = min(n, R);
        return ;
    }
    int len = R - L + 1;
    if (L < len)//剪枝,因为如果l小于区间长度的话,那么他的父节点的左端点一定为负值,所以不符合
        return;
    search(L, R + len);//左孩子
    if (len != 1)
        search(L, R + len - 1);//左孩子
    search(L - len, R);//右孩子
    search(L - len - 1, R);//右孩子
}
LL work(int L, int R)
{
    n = inf;
    search(L, R);
    return n == inf ? -1 : n;
}
int main()
{

    int L, R;
    while (~scanf("%d %d", &L, &R))
    {
        printf("%lld\n", work(L, R));
    }
    return 0;
}

1011. Work (hdu5326)

官方签到题。直接dfs即可

#include <cstdio>
#include <cstring>
#include <vector>
using namespace std;
const int maxn = 110;
vector<int> mp[maxn];
int in[maxn];
int n, k;
int siz[maxn];
int dfs(int u)
{
    siz[u] = 1;
    int len = mp[u].size();
    for (int i = 0; i < len; i++)
    {
        siz[u] += dfs(mp[u][i]);
    }
    return siz[u];
}
int main()
{
    while (~scanf("%d%d", &n, &k))
    {
        memset(siz, 0, sizeof(siz));
        memset(in, 0, sizeof(in));
        for (int i = 0; i <= n; i++)
            mp[i].clear();
        int a, b;
        for (int i = 1; i < n; i++)
        {
            scanf("%d%d", &a, &b);
            mp[a].push_back(b);
            in[b]++;
        }
        int root = 0;
        for (int i = 1; i <= n; i++)
            if (!in[i])
                root = i;
        dfs(root);
        int ans = 0;
        for (int i = 1; i <= n; i++)
            if (siz[i] - 1 == k)
                ans++;
        printf("%d\n", ans);
    }
    return 0;
}

其它题目未补完。。。

时间: 2024-10-25 22:22:42

多校联赛第三场(部分题解)的相关文章

【多校赛第三场】Redraw Beautiful Drawings【网络流】【谜のWA】

参考题解:http://blog.csdn.net/qian99/article/details/38276887 #include <iostream> #include <cstdio> #include <cstdlib> #include <cmath> #include <cstring> #include <queue> #include <vector> #include <algorithm>

中南多校对抗赛 第三场 B

B:Arithmetic Progressions 题意: 给你一个长度为n的序列,问你这个序列中长度最长的等差数列长度为多少 题解: 方法一:将数组从小到大排序,n方扫,枚举出公差d,然后二分找有多少个满足等差序列的值即可,emmm虽然说不知道为什么这种最差复杂度为n^3*logn的暴力能过,可能是csuoj太快了吧. 代码: #include <bits/stdc++.h> #define eps 1e-8 #define INF 0x3f3f3f3f #define PI acos(-1

HDU 5762 Teacher Bo (鸽笼原理) 2016杭电多校联合第三场

题目:传送门. 题意:平面上有n个点,问是否存在四个点 (A,B,C,D)(A<B,C<D,A≠CorB≠D)使得AB的横纵坐标差的绝对值的和等于CD的横纵坐标差的绝对值的和,n<10^5,点的坐标值m<10^5. 题解:表面上这道题复杂度是O(n^2)会超时的,而实际上这些坐标差绝对值的和最大是2*10^5,所以复杂度不是O(n^2),而是O(min(n^2,m)),这就是著名的鸽笼原理. #include <iostream> #include <cstdio

hdu5737(2016多校联赛第2场D)

题意:给2组数据a和b数组,每次有2种操作:(+,l,r,x)把a数组第l个到第r个元素全置为x,(?,l,r)查询[l,r]之间哪些位置满足a[i]>=b[i](i>=l && i<=r)并把这些位置的数量统计 一直想很久,没想到什么有效的方案,直到看到题解才明白过来,原来线段树套平衡树还有这种情况:里面其实不是平衡树,只是有序表. 然后这题就转换为区间查找数对应排名 由于此题不用对2个数组都修改,其中1个b树可作为固定的线段树套有序表以节省时间,另外1个表a树则单纯使

HDU 5752 Sqrt Bo (思维题) 2016杭电多校联合第三场

题目:传送门. 题意:一个很大的数n,最多开5次根号,问开几次根号可以得到1,如果5次还不能得到1就输出TAT. 题解:打表题,x1=1,x2=(x1+1)*(x1+1)-1,以此类推.x5是不超过long long的,判断输出即可. #include <iostream> #include <cstdio> #include <cstring> #include <cmath> using namespace std; typedef long long

HDU 4893 Wow! Such Sequence!(2014年多校联合 第三场 G)(线段树)

磨了一天的线段树,不能说完全搞清楚,只能说有一个大概的了解,靠着模板才把这道题A了,只能说太弱~~! 题意: 初始时有一字符串,全为0. 三种操作: 1 k d - add  把d加到第k个数上去2 l r - query sum 计算l到r所有数的和3 l r - change to nearest Fibonacci 把l到r的数修改为距离它最近的斐波那契数 节点附件三个值: s1:由lazy控制的区间的正确的和. s2:区间内与所有数相近的fib数之和,随着单点更新而更新. col:laz

中南多校对抗赛 第三场 E

E:Eulerian Flight Tour 题意: 给你一张无向图,要你给这个图加边使得其形成一个欧拉回路 题解: 首先使得所有节点的度都为偶数,然后将这个图联通起来 对于度为奇数的点,将将他和他的父节点连接起来 连接完后如果这个图是联通的,那么就直接输出结果 如果这个图有多个联通块: 分类讨论: 如果有2个联通块,两个联通块对着连成一个环 如果有多个联通块,用一个环将这几个联通块连起来即可 代码: # E:Eulerian Flight Tour ## 题意: 给你一张无向图,要你给这个图加

hdu 5326 Work(杭电多校赛第三场)

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=5326 Work Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submission(s): 583    Accepted Submission(s): 392 Problem Description It’s an interesting experience to

2018 HDU多校第三场赛后补题

2018 HDU多校第三场赛后补题 从易到难来写吧,其中题意有些直接摘了Claris的,数据范围是就不标了. 如果需要可以去hdu题库里找.题号是6319 - 6331. L. Visual Cube 题意: 在画布上画一个三维立方体. 题解: 模拟即可. 代码: #include <bits/stdc++.h> using namespace std; int a, b, c, R, C; char g[505][505]; int main () { int T; cin >>