• 257. 关押罪犯


    S 城现有两座监狱,一共关押着 N 名罪犯,编号分别为 1∼N。

    他们之间的关系自然也极不和谐。

    很多罪犯之间甚至积怨已久,如果客观条件具备则随时可能爆发冲突。

    我们用“怨气值”(一个正整数值)来表示某两名罪犯之间的仇恨程度,怨气值越大,则这两名罪犯之间的积怨越多。

    如果两名怨气值为 c 的罪犯被关押在同一监狱,他们俩之间会发生摩擦,并造成影响力为 c 的冲突事件。

    每年年末,警察局会将本年内监狱中的所有冲突事件按影响力从大到小排成一个列表,然后上报到 S 城 Z 市长那里。

    公务繁忙的 Z 市长只会去看列表中的第一个事件的影响力,如果影响很坏,他就会考虑撤换警察局长。

    在详细考察了 N 名罪犯间的矛盾关系后,警察局长觉得压力巨大。

    他准备将罪犯们在两座监狱内重新分配,以求产生的冲突事件影响力都较小,从而保住自己的乌纱帽。

    假设只要处于同一监狱内的某两个罪犯间有仇恨,那么他们一定会在每年的某个时候发生摩擦。

    那么,应如何分配罪犯,才能使 Z 市长看到的那个冲突事件的影响力最小?这个最小值是多少?

    输入格式

    第一行为两个正整数 N 和 M,分别表示罪犯的数目以及存在仇恨的罪犯对数。

    接下来的 M 行每行为三个正整数 aj,bj,cj,表示 aj 号和 bj 号罪犯之间存在仇恨,其怨气值为 cj。

    输出格式

    输出共 1 行,为 Z 市长看到的那个冲突事件的影响力。

    如果本年内监狱中未发生任何冲突事件,请输出 0。

    数据范围

    N≤20000,M≤100000

    输入样例:

    4 6
    1 4 2534
    2 3 3512
    1 2 28351
    1 3 6618
    2 4 1805
    3 4 12884
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7

    输出样例:

    3512
    
    • 1

    代码:

    /*
    (二分,染色法判断二分图) O((N+M)logC)
    将罪犯当做点,罪犯之间的仇恨关系当做点与点之间的无向边,边的权重是罪犯之间的仇恨值。
    那么原问题变成:将所有点分成两组,使得各组内边的权重的最大值尽可能小。
    
    我们在 [0,109] 之间枚举最大边权 limit,当 limit 固定之后,剩下的问题就是:
    
    判断能否将所有点分成两组,使得所有权值大于 limit 的边都在组间,而不在组内。也就是判断由所有点以及所有权值大于 limit 的边构成的新图是否是二分图。
    判断二分图可以用染色法,时间复杂度是 O(N+M),其中 N 是点数,M 是边数
    
    为了加速算法,我们来考虑是否可以用二分枚举 limit, 假定最终最大边权的最小值是 Ans:
    
    那么当 limit∈[ans,109] 时,所有边权大于 limit 的边,必然是所有边权大于Ans 的边的子集,因此由此构成的新图也是二分图。
    当 limit∈[0,ans−1] 时,由于ans 是新图可以构成二分图的最小值,因此由大于 limit 的边构成的新图一定不是二分图。
    所以整个区间具有二段性,可以二分出分界点 ans 的值。
    */
    
    #include <bits/stdc++.h>
    using namespace std;
    const int N = 20010, M = 200010;
    
    int n, m;
    int h[N], e[M], w[M], ne[M], idx;
    int color[N];
    
    void add(int a, int b, int c)
    {
        e[idx] = b, w[idx] = c, ne[idx] = h[a], h[a] = idx++;
    }
    bool dfs(int u, int c, int mid)
    {
        color[u] = c;
        for (int i = h[u]; ~i; i = ne[i])
        {
            int j = e[i];
            if (w[i] <= mid)
                continue;
            if (color[j])
            {
                if (color[j] == c)
                    return false;
            }
            else if (!dfs(j, 3 - c, mid))
                return false;
        }
        return true;
    }
    
    bool check(int mid)
    {
        memset(color, 0, sizeof color);
        for (int i = 1; i <= n; i++)
            if (!color[i])
                if (!dfs(i, 1, mid))
                    return false;
        return true;
    }
    
    int main()
    {
        scanf("%d%d", &n, &m);
        memset(h, -1, sizeof h);
    
        while (m--)
        {
            int a, b, c;
            scanf("%d%d%d", &a, &b, &c);
            add(a, b, c), add(b, a, c);
        }
    
        int l = 0, r = 1e9;
        while (l < r)
        {
            int mid = r + l >> 1;
            if (check(mid))
                r = mid;
            else
                l = mid + 1;
        }
    
        printf("%d\n", r);
    
        return 0;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
    • 56
    • 57
    • 58
    • 59
    • 60
    • 61
    • 62
    • 63
    • 64
    • 65
    • 66
    • 67
    • 68
    • 69
    • 70
    • 71
    • 72
    • 73
    • 74
    • 75
    • 76
    • 77
    • 78
    • 79
    • 80
    • 81
    • 82
    • 83
    • 84
  • 相关阅读:
    JVM内存和垃圾回收-04.程序计数器(PC寄存器)
    Mybatis02动态sql和分页
    java装饰器模式
    自动化办公01 smtplib 邮件⾃动发送
    软件测试工作步骤详情
    MySQL使用教程(基础篇03)
    哈希算法(二)哈希算法与一致性哈希算法
    【Go】【反射】反射基本介绍和使用
    linux高级---k8s中的五种控制器
    详解数据库中的索引和视图
  • 原文地址:https://blog.csdn.net/segegse/article/details/125438197