【Codeforces】Codeforces Round #535

Problem A

分三种情况:

  • l_1=l_2并且l_1=r_2,由于必定有解,所以r_1必定不会跟l_1相等。所以直接输出r_1l_1即可。
  • 除此之外,如果l_1=l_2,那么输出l_1r_2即可。
  • 否则,直接输出l_1l_2即可。

时间复杂度为O(q)

#include <iostream>

using namespace std;

int main()
{
    int q;
    cin >> q;
    while(q--)
    {
        int l1, r1, l2, r2;
        cin >> l1 >> r1 >> l2 >> r2;
        if((l1 == l2) && (l1 == r2))
        {
            cout << r1 << " " << l1 << endl;
        }
        else if(l1 == l2)
        {
            cout << l1 << " " << r2 << endl;
        }
        else
        {
            cout << l1 << " " << l2 << endl;
        }
    }
    return 0;
}

Problem B

由于必定有解,于是对数组做两次如下操作:

  • 选出数组中最大的数字,记为ans
  • 将数组中所有ans的因数去掉一份(去掉一份你的意思是,假设数组里有两个1,那么只去掉一个)。

数组用map统计每个数的个数来存储,找因数使用根号试除法,则时间复杂度为O(n{\sqrt d}{\log n})

#include <iostream>
#include <map>

using namespace std;

int getAns(map<int, int> &mp)
{
    auto it = mp.end();
    it--;
    int ret = it->first;
    for(int i = 1; i * i <= ret; i++)
    {
        if(ret % i)continue;
        mp[i]--;
        if(mp[i] == 0)
        {
            mp.erase(i);
        }
        if(ret / i != i)
        {
            mp[ret / i]--;
            if(mp[ret / i] == 0)
            {
                mp.erase(ret / i);
            }
        }
    }
    return ret;
}

int main()
{
    int n;
    while(cin >> n)
    {
        map<int, int> mp;
        for(int i = 0; i < n; i++)
        {
            int p;
            cin >> p;
            mp[p]++;
        }
        cout << getAns(mp) << " ";
        cout << getAns(mp) << endl;
    }
    return 0;
}

Problem C

根据题意显然可以得知,符合题目要求的字符串必定是以3为周期重复的。
所以只需要枚举前三个字母,然后构造跟输入的字符串s等长的字符串,再比较其字母不同的个数,最后从这些值中取最小值即可。

时间复杂度为O(n)

#include <iostream>
#include <string>
#include <map>

using namespace std;

const string AlphaList = "RGB";

int getAns(string &c, string &s)
{
    int ret = 0;
    for(int i = 0; i < s.length(); i++)
    {
        if(s[i] != c[i % 3])
        {
            ret++;
        }
    }
    return ret;
}

int main()
{
    int n;
    while(cin >> n)
    {
        string s;
        cin >> s;
        int ans = s.length() + 5;
        string c;
        for(int i = 0; i < 3; i++)
        {
            for(int j = 0; j < 3; j++)
            {
                if(i == j)continue;
                for(int k = 0; k < 3; k++)
                {
                    if(i == k)continue;
                    if(j == k)continue;
                    string pc;
                    pc += AlphaList[i];
                    pc += AlphaList[j];
                    pc += AlphaList[k];
                    int pans = getAns(pc, s);
                    if(ans > pans)
                    {
                        ans = pans;
                        c = pc;
                    }
                }
            }
        }
        cout << ans << endl;
        for(int i = 0; i < s.length(); i++)
        {
            cout << c[i % 3];
        }
        cout << endl;
    }
    return 0;
}

Problem D

dp[i][j]为字符串为原串的子串[0,i]的情况下,且最后一个字符强制改为j的情况下的最小修改数。然后另设一个数组prt用于记录dp过程中的选择,然后直接dp过去就行了。

时间复杂度为O(n)

#include <iostream>
#include <string>
#include <algorithm>

using namespace std;

const string AlphaList = "RGB";

int dp[200005][4];
int prt[200005][4];

void Solve(string &s)
{
    for(int i = 0; i < 3; i++)
    {
        if(s[0] == AlphaList[i])
        {
            dp[0][i] = 0;
        }
        else
        {
            dp[0][i] = 1;
        }
    }
    
    for(int i = 1; i < s.length(); i++)
    {
        for(int j = 0; j < 3; j++)
        {
            dp[i][j] = s.length() + 5;
            for(int k = 0; k < 3; k++)
            {
                if(j == k)continue;
                if(dp[i][j] > dp[i-1][k])
                {
                    dp[i][j] = dp[i-1][k];
                    prt[i][j] = k;
                }
            }
            if(s[i] != AlphaList[j])
            {
                dp[i][j]++;
            }
        }
    }

    int ans = 0;
    if(dp[s.length() - 1][1] < dp[s.length() - 1][ans])
    {
        ans = 1;
    }
    if(dp[s.length() - 1][2] < dp[s.length() - 1][ans])
    {
        ans = 2;
    }

    cout << dp[s.length() - 1][ans] << endl;
    string ansstr;
    for(int i = s.length() - 1; i >= 0; i--)
    {
        ansstr += AlphaList[ans];
        ans = prt[i][ans];
    }
    reverse(ansstr.begin(), ansstr.end());
    cout << ansstr << endl;
}

int main()
{
    int n;
    while(cin >> n)
    {
        string s;
        cin >> s;
        Solve(s);
    }
}

Problem E (E1 + E2)

首先显然需要建立两颗支持区间更新的线段树,分别维护数组的最小值和最大值。
能够想到的一个思路是枚举数组中的每个数,令其尽可能小,除它以外的数尽可能大(只需要只选取所有包含当前数的区间就可以达到这个目的来),然后得到此时的极差,并取所有极差的最大值。
对于每个枚举的值计算极差的时间复杂度是O(m{\log n}),于是总的时间复杂度是O(nm{\log n})。这显然爆炸。
换个思路,考虑到相邻两次枚举的时候选取的区间是有相交的部分的。由此可以考虑使用尺取法来维护所选的区间。
于是我们需要复制两个原来的区间序列,分别以按l升序和按r升序来排序(分别记为数列v_l和数列v_r)。那么每次我们枚举完第i个数之后,我们便需要根据v_r去掉所有右值小于i的区间,再根据v_l加入所有左值等于i+1开头的区间。这样我们就可以从所有只覆盖到i的区间推出所有只覆盖i+1的区间。

使用尺取法之后的总时间复杂度为O(n+m{\log (m+n)})。不知道会不会算错不过如果按这个算法的话貌似比标程要优一点(标程是O(nm))?感觉十分的慌啊总觉得要么就是时间复杂度算错了要么就是有可以叉掉我的case。

#include <cstdio>
#include <cstring>
#include <algorithm>
#include <vector>
#include <set>

using namespace std;

const int INF = 1e8;

struct SegTree
{
    int a[400005];
    int b[400005];
    int lazy[400005];
    
    void init()
    {
        memset(lazy, 0, sizeof(lazy));
        for(int i = 0; i < 400005; i++)
        {
            a[i] = INF;
            b[i] = -1 * INF;
        }
    }

    void SetElem(int pos, int elem, int l, int r, int p)
    {
        if(l == r)
        {
            a[p] = elem;
            b[p] = elem;
            return;
        }
        int mid = (l + r) >> 1;
        if(pos <= mid)
        {
            SetElem(pos, elem, l, mid, p << 1);
        }
        else
        {
            SetElem(pos, elem, mid + 1, r, (p << 1) + 1);
        }
        a[p] = min(a[p << 1], a[(p << 1) + 1]);
        b[p] = max(b[p << 1], b[(p << 1) + 1]);
    }

    void SetSeg(int pl, int pr, int elem, int l, int r, int p)
    {
        if((pl == l) && (pr == r))
        {
            lazy[p] += elem;
            return;
        }
        int mid = (l + r) >> 1;
        if(pr <= mid)
        {
            SetSeg(pl, pr, elem, l, mid, p << 1);
        }
        else if(pl > mid)
        {
            SetSeg(pl, pr, elem, mid + 1, r, (p << 1) + 1);
        }
        else
        {
            SetSeg(pl, mid, elem, l, mid, p << 1);
            SetSeg(mid + 1, pr, elem, mid + 1, r, (p << 1) + 1);
        }

        a[p] = min(a[p << 1] + lazy[p << 1], a[(p << 1) + 1] + lazy[(p << 1) + 1]);
        b[p] = max(b[p << 1] + lazy[p << 1], b[(p << 1) + 1] + lazy[(p << 1) + 1]);
    }

    int GetMin()
    {
        return a[1] + lazy[1];
    }

    int GetMax()
    {
        return b[1] + lazy[1];
    }

    int GetMinPos(int l, int r, int p)
    {
        if(l == r)
        {
            return l;
        }
        int mid = (l + r) >> 1;
        if(a[p] + lazy[p] == a[p << 1] + lazy[p << 1])
        {
            return GetMinPos(l, mid, p << 1);
        }
        else
        {
            return GetMinPos(mid + 1, r, (p << 1) + 1);
        }
    }
}S;

struct node
{
    int l, r;
    int pos;
    node(int _l = 0, int _r = 0, int _pos = 0)
    {
        l = _l;
        r = _r;
        pos = _pos;
    }
};

bool cmp_l(const node &a, const node &b)
{
    if(a.l != b.l)
    {
        return a.l < b.l;
    }
    return a.r < b.r;
}
bool cmp_r(const node &a, const node &b)
{
    if(a.r != b.r)
    {
        return a.r < b.r;
    }
    return a.l < b.l;
}
vector<node> vl, vr;

int main()
{
    int n, m;
    while(~scanf("%d%d", &n, &m))
    {
        S.init();
        vl.clear();
        vr.clear();
        for(int i = 1; i <= n; i++)
        {
            int a;
            scanf("%d", &a);
            S.SetElem(i, a, 1, n, 1);
        }
        for(int i = 0; i < m; i++)
        {
            int l, r;
            scanf("%d%d", &l, &r);
            vl.push_back(node(l, r, i + 1));
            vr.push_back(node(l, r, i + 1));
        }
        sort(vl.begin(), vl.end(), cmp_l);
        sort(vr.begin(), vr.end(), cmp_r);

        int pos_l = 0;
        int pos_r = 0;
        int ans = -1;
        int ans_pos;
        for(int i = 1; i <= n; i++)
        {
            while((pos_r < m) && (vr[pos_r].r < i))
            {
                S.SetSeg(vr[pos_r].l, vr[pos_r].r, 1, 1, n, 1);
                pos_r++;
            }
            while((pos_l < m) && (vl[pos_l].l == i))
            {
                S.SetSeg(vl[pos_l].l, vl[pos_l].r, -1, 1, n, 1);
                pos_l++;
            }
            if(ans < S.GetMax() - S.GetMin())
            {
                ans = S.GetMax() - S.GetMin();
                ans_pos = i;
            }
        }

        while(pos_l < m)
        {
            S.SetSeg(vl[pos_l].l, vl[pos_l].r, -1, 1, n, 1);
            pos_l++;
        }
        while(pos_r < m)
        {
            S.SetSeg(vr[pos_r].l, vr[pos_r].r, 1, 1, n, 1);
            pos_r++;
        }

        vector<int> ans_v;
        for(int i = 0; i < m; i++)
        {
            if((ans_pos <= vl[i].r) && (ans_pos >= vl[i].l))
            {
                ans_v.push_back(vl[i].pos);
            }
        }

        printf("%d\n%d\n", ans, (int)ans_v.size());
        for(int i = 0; i < ans_v.size(); i++)
        {
            if(i)printf(" ");
            printf("%d", ans_v[i]);
        }
        printf("\n");

    }
    return 0;
}

Problem F

一个十分显然的想法是,对图建最小生成树。然后对于剩下的边<u,v,w>而言,只要查询最小生成树上从uv的路径的最大值是否等于w即可。如果是则意味着当前最小生成树中从uv的路径上有一条边的权值跟当前边<u,v,w>相等,即表明可以用边<u,v,w>替换之使其依旧为最小生成树。那么这显然是需要加权值的边,于是答案+1。
对于上述查询的话可以使用树链剖分维护最小生成树来实现,总时间复杂度为O(m({\log m})^2)。但是事实上有更加简单且高效的解法。
我们可以把边按权值归类,并让这些分类按权值从小到大排序。那么对于每一类边而言,其权值就是相等的。那么我们便可以按权值从小到大枚举每一类边。对于每一类边而言,用这些边做Kruskal的过程。如果出现会构成环且不是自环的边,那么表明这条边肯定跟同一类边冲突了,那么答案加一。然后再每次对一类边做完Kruskal之后,按照当前Kruskal的并查集对当前图缩点
思路很简洁,但是缩点这个过程时耗很长(如果每次都去缩点的话,时间复杂度是O(n)的)。所以需要考虑如何对缩点这个过程进行优化。
于是考虑不直接缩点,而是将上一个类Kruskal之后的并查集存下来用于当前类Kruskal过程的查询。于是就可以这么做:维护两个并查集D1和D2,其中D1记录的就是上一个类遍历完之后的并查集。然后从小到大遍历每一类边。对于每一类边而言做两遍Kruskal:

  • 第一遍针对D2,如果出现无法插入D2的边,则先去D1检查一下两个点是否已经并起来了。如果已经并起来了的话说明当前边在现在这个缩完点的图中是一条自环边,直接舍弃。否则则表明该边是一条冲突边,答案加一
  • 第二遍针对D1,加就完事了,仅仅是用于将D2的并查集同步到D1。

分类使用map维护,总的时间复杂度为O(m{\log m})

#include <cstdio>
#include <cstring>
#include <algorithm>
#include <map>
#include <vector>

using namespace std;

const int MAXN = 200005;

struct DSU
{
    int prt[MAXN];
    int n;

    void init(int _n)
    {
        n = _n;
        for(int i = 1; i <= n; i++)
        {
            prt[i] = i;
        }
    }

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

    bool unite(int x, int y)
    {
        x = find(x);
        y = find(y);
        if(x == y)return false;
        if(x > y)swap(x, y);
        prt[y] = x;
        return true;
    }
};

struct Kruskal
{
    map< int, vector< pair<int, int> > > mp;
    int n;

    DSU D1, D2;
    
    void init(int _n)
    {
        n = _n;
        mp.clear();
        D1.init(n);
        D2.init(n);
    }

    void push_back(int _u, int _v, int _w)
    {
        mp[_w].push_back(make_pair(_u, _v));
    }

    int Solve()
    {
        int ret = 0;
        for(auto it = mp.begin(); it != mp.end(); it++)
        {
            vector< pair<int, int> > &v = it->second;
            for(int i = 0; i < v.size(); i++)
            {
                if(!D2.unite(v[i].first, v[i].second))
                {
                    if(D1.find(v[i].first) == D1.find(v[i].second))
                    {
                        continue;
                    }
                    ret++;
                }
            }
            for(int i = 0; i < v.size(); i++)
            {
                D1.unite(v[i].first, v[i].second);
            }
        }
        return ret;
    }
}K;

int main()
{
    int n, m;
    while(~scanf("%d%d", &n, &m))
    {
        K.init(n);
        for(int i = 0; i < m; i++)
        {
            int u, v, w;
            scanf("%d%d%d", &u, &v, &w);
            K.push_back(u, v, w);
        }
        printf("%d\n", K.Solve());
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,539评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,911评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,337评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,723评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,795评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,762评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,742评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,508评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,954评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,247评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,404评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,104评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,736评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,352评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,557评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,371评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,292评论 2 352

推荐阅读更多精彩内容