zoukankan      html  css  js  c++  java
  • ARC089E GraphXY 构造

    传送门


    在Luogu上评了”NOI“之后评级变成了”普及+/提高“……我觉得我可能要退群了

    考虑构造一个这样的图:

    其中上半部分是从(S)开始的一条长(100)、每条边权都为(x)的链(长度超过(100)显然是没有意义的),下半部分是以(T)结束的一条长(100)、每条边权都为(y)的链。在这两条链中间,有(101 imes 101)条边(上图中只画出了部分),边权为一个矩阵(f_{i,j})。如果要经过(f_{i,j})对应的边,那么需要从(S)开始沿着上半部分的链走(i)步,然后经过(f_{i,j}),最后沿着下半部分的链走(j)步到达(T),总边权就是(xi+yj+f_{i,j})

    在这个图中,有(d_{x,y} = minlimits_{a,b geq 0} xa + yb + f_{a,b}),松弛条件得(forall a , b geq 0 , d_{x,y} leq xa + yb + f_{a,b}),即(f_{a,b} geq d_{x,y} - xa - yb)

    这样可以对于每个(f_{a,b})得到(A imes B)个不等式,可得(f_{a,b})的解集。而(f_{a,b})一定需要取到其中的最小值(如果不取到最小值,可能存在(x,y)满足(minlimits_{a,b geq 0} xa + yb + f_{a,b} > d_{x,y})),所以可以得到(f_{a,b})的确切值。然后再代入(d_{x,y})的式子检验以下图是否合法即可。

    时间复杂度(O(N^3))

    #include<iostream>
    #include<cstdio>
    #include<cstdlib>
    #include<ctime>
    #include<cctype>
    #include<algorithm>
    #include<cstring>
    #include<iomanip>
    #include<queue>
    #include<map>
    #include<set>
    #include<bitset>
    #include<vector>
    #include<stack>
    #include<cmath>
    #include<random>
    //This code is written by Itst
    using namespace std;
    
    inline int read(){
        int a = 0;
        char c = getchar();
        bool f = 0;
        while(!isdigit(c) && c != EOF)
            c = getchar();
        if(c == EOF)
            exit(0);
        while(isdigit(c)){
        a = a * 10 + c - 48;
            c = getchar();
        }
        return f ? -a : a;
    }
    
    int d[12][12] , f[107][107];
    
    signed main(){
        int A = read() , B = read();
        for(int i = 1 ; i <= A ; ++i)
        for(int j = 1 ; j <= B ; ++j)
            d[i][j] = read();
        for(int i = 0 ; i <= 100 ; ++i)
        for(int j = 0 ; j <= 100 ; ++j)
            for(int p = 1 ; p <= A ; ++p)
            for(int q = 1 ; q <= B ; ++q)
                f[i][j] = max(f[i][j] , d[p][q] - i * p - j * q);
        for(int i = 1 ; i <= A ; ++i)
        for(int j = 1 ; j <= B ; ++j){
            int minN = 1e9;
            for(int p = 0 ; p <= 100 ; ++p)
            for(int q = 0 ; q <= 100 ; ++q)
                minN = min(minN , f[p][q] + p * i + q * j);
            if(minN != d[i][j]){
            puts("Impossible");
            return 0;
            }
        }
        puts("Possible");
        puts("202 10401");
        for(int i = 1 ; i <= 100 ; ++i)
        printf("%d %d X
    " , i , i + 1);
        for(int i = 102 ; i < 202 ; ++i)
        printf("%d %d Y
    " , i , i + 1);
        for(int i = 0 ; i <= 100 ; ++i)
        for(int j = 0 ; j <= 100 ; ++j)
            printf("%d %d %d
    " , 1 + i , 202 - j , f[i][j]);
        puts("1 202");
        return 0;
    }
    
  • 相关阅读:
    java ssh整合报错:java.lang.NoSuchMethodError: antlr.collections.AST.getLine()I
    Ubuntu16.04安装搜狗输入法后有黑边问题的解决方法
    socket
    vim编辑器的使用
    linux用户和群组
    bash shell
    [LightOJ 1128]Greatest Parent
    [Luogu P4180][BJWC 2010]严格次小生成树
    函数、方法区别
    有关_meta内容(持续更新)
  • 原文地址:https://www.cnblogs.com/Itst/p/10587121.html
Copyright © 2011-2022 走看看