美文网首页
PROB: cowtour

PROB: cowtour

作者: SylviaShen | 来源:发表于2017-09-07 20:32 被阅读0次

题目来自 USACO
题目翻译见 NOCOW

超时的暴搜

这道题一看也是很清楚了,求一个连通部分的直径,那就肯定是 floyd 算法。根据给的邻接矩阵,可以算出图上的路径。

后来才看懂这个题目给的可能是好几个连通的子集组成的,而我要在任意两个连通域里面加任意一条边。所以我又做了一下 dfs,对各个区域染色。这样,任取 i 和 j,判断他俩不是一个连通集里的,就加一条边做 floyd。

我试了一下,floyd 这样直接算不连通的图好像是没有问题的,算出来各自内部的最短路径,不同连通域的就还是 +Inf。

其中还遇到了一些不熟悉的东西:

  • double 类型的最大值,在 float.h 中有常量 DBL_MAX;
  • 对于输入,这次给的是一串 1 和 0,我没有用 int,而是读成 %c ,再处理成 bool;
  • sqrt 在 math.h 里;
  • 输出 double 要用 %lf,前面同样可以 .6 限制小数位数。

尽管代码看起来不丑,可是到 Test 5 就超时了。

/*
ID:
LANG: C++
TASK: cowtour
*/

#include <cstdio>
#include <cstdlib>
#include <cmath>
#include <cfloat>
#include <cstring>

const int maxN = 150; 
int N;
bool adj[maxN][maxN]; //邻接矩阵
double graph[maxN][maxN]; //图中距离
double dist[maxN][maxN]; //当前最短路径
int loc[maxN][2]; //坐标
int color[maxN]; //分出不同的连通域

double diameter(int x, int y, double d){
    //在图中新增一条边
    memcpy(dist, graph, sizeof(graph));
    dist[x][y] = d;
    dist[y][x] = d;

    //floyd算法
    for(int k = 0; k < N; k ++){
        for(int i = 0; i < N; i ++){
            for(int j = 0; j < N; j ++){
                if(dist[i][k] + dist[k][j] < dist[i][j])
                    dist[i][j] = dist[i][k] + dist[k][j];
            }
        }
    }

    //找diameter
    double dia = 0;
    for(int i = 0; i < N; i ++){
        for(int j = 0; j < N; j ++){
            if(dist[i][j] > dia) dia = dist[i][j];
        }
    }
    return dia;
}

void dfs(int idx, int clr){
    if(color[idx]) return;
    color[idx] = clr;
    for(int i = 0; i < N; i ++){
        if(adj[idx][i]) dfs(i, clr);
    }
}

int main(){
    FILE *fin = fopen("cowtour.in", "r");
    FILE *fout = fopen("cowtour.out", "w");

    //输入
    fscanf(fin, "%d", &N);
    for(int i = 0; i < N; i ++){
        fscanf(fin, "%d %d\n", &(loc[i][0]), &(loc[i][1]));
    }
    char temp;
    for(int i = 0; i < N; i ++){
        for(int j = 0; j < N; j ++){
            fscanf(fin, "%c", &temp);
            adj[i][j] = temp - 48;
        }
        fscanf(fin, "%c", &temp);
    }

    //确定连通域
    int clr = 1;
    for(int i = 0; i < N; i ++){
        if(!color[i]) dfs(i, clr ++);
    }

    //建立图
    for(int i = 0; i < N; i ++){
        for(int j = 0; j < N; j ++){
            if(adj[i][j]){
                graph[i][j] = sqrt(pow((loc[i][0] - loc[j][0]), 2) + pow((loc[i][1] - loc[j][1]), 2)); //勾股
            }else{
                graph[i][j] = DBL_MAX;
            }
        }
        graph[i][i] = 0;
    }    

    //不同连通域两个点连起来,再算图的直径
    double minDia = DBL_MAX, dia;
    for(int i = 0; i < N; i ++){ 
        for(int j = 0; j < N; j ++){
            if(color[j] == color [i] || adj[i][j]) continue; 
            dia = diameter(i, j, sqrt(pow((loc[i][0] - loc[j][0]), 2) + pow((loc[i][1] - loc[j][1]), 2)));
            if(dia < minDia) minDia = dia;
        }
    }
    fprintf(fout, "%.6lf\n", minDia);

    return 0;
}

改思路

这个思路是我在吃完吉祥馄饨回来的路上想到的,后来看 nocow 上大家好像都是这么做的。

首先我在想,添加一条边,究竟造成了什么影响?能不能不去暴力重新算呢。这个新的连通域的直径,可能是:

  • 含有这条边的某一路径,那么,原连通域 1 中的每个点都可以有最短路径到这条边的顶点 i,再通过这条边,从顶点 j 有最短路径到原连通域 2 的每个点。那这其中的最大距离就是(maxDist[i] + maxDist[j] + 边(i, j),maxDist 是 i, j 在原来连通域中到某个点的最短路径的最大值。
  • 原来的连通域有条最短路太长了,那新连通域可能还是这个直径。

因为要把新添的边枚举完,所以每个点的 maxDist 都要做,此外,还需要计算每个连通域的直径。这些可以在算该连通域的 floyd 算法后一起算掉。

由于这些点都是混的,我也只能在每一步循环里判断是不是属于某连通域了。代码拖得很长。

数据太弱

这样, Test 5 一闪而过,特别开心,可是后面的 test 又出错了。这时候数据量已经到最大了,很醉,我只能对着代码瞪眼,后来竟发现了一个非常致命的错误,取连通域直径的时候应该给 color 我却给了顶点,可这样竟然跑通了一大半的例子。

还有据 nocow 说,虽然题目说不少于两个连通域,这里给的全是两个连通域的,理解错题意都没问题。还有写错大于小于号的,都跑到了 test 7,也是真服。

精度

本来以为 float 就够了,结果遇到了啼笑皆非的问题:

 > Run 7: Execution error: Your program did not produce an answer
        that was judged as correct. The program stopped at 0.014 seconds;
        it used 4292 KB of memory. At character number 9, your answer says
        '0' while the correct answer says '2'. 

        Here are the respective outputs:
        ----- our output ---------
        39796.392691
        ---- your output ---------
        39796.390625
        --------------------------

这……欺负我第一次用浮点数嘛……

我就全改成了 double,然后顺利 AC 了。

AC代码:

/*
ID: 
LANG: C++
TASK: cowtour
*/

#include <cstdio>
#include <cstdlib>
#include <cmath>
#include <cfloat>
#include <cstring>
#define triangle(x, y) sqrt(pow((loc[x][0] - loc[y][0]), 2) + pow((loc[x][1] - loc[y][1]), 2))
#define max(x, y) ((x) > (y)? (x): (y))

const int maxN = 150; 
int N;
bool adj[maxN][maxN]; //邻接矩阵
double dist[maxN][maxN]; //某个连通域内的最短路径
int loc[maxN][2]; //坐标
int color[maxN]; //每个点所在的连通域
double maxDist[maxN]; //每个点在该连通域据其余点的最长距离
double diameters[maxN];//颜色为i - 1的连通域的直径

void diameter(int nclr){
    //仅针对连通域 nclr 的 floyd 算法
    for(int k = 0; k < N; k ++){
        if(color[k] != nclr) continue;
        for(int i = 0; i < N; i ++){
            if(color[i] != nclr) continue;
            for(int j = 0; j < N; j ++){
                if(color[j] != nclr) continue;
                if(dist[i][k] + dist[k][j] < dist[i][j])
                    dist[i][j] = dist[i][k] + dist[k][j];
            }
        }
    }

    //存maxDist, 找diameter
    double dia = 0, maxd;
    for(int i = 0; i < N; i ++){
        if(color[i] != nclr) continue;
        maxd = 0;
        for(int j = 0; j < N; j ++){
            if(color[j] != nclr) continue;
            if(dist[i][j] > maxd) maxd = dist[i][j];
        }
        maxDist[i] = maxd;
        if(maxd > dia) dia = maxd;
    }
    diameters[nclr - 1] = dia;
}

void dfs(int idx, int nclr){
    if(color[idx]) return;
    color[idx] = nclr;
    for(int i = 0; i < N; i ++){
        if(adj[idx][i]) dfs(i, nclr);
    }
}

int main(){
    FILE *fin = fopen("cowtour.in", "r");
    FILE *fout = fopen("cowtour.out", "w");

    //输入
    fscanf(fin, "%d", &N);
    for(int i = 0; i < N; i ++){
        fscanf(fin, "%d %d\n", &(loc[i][0]), &(loc[i][1]));
    }
    char temp;
    for(int i = 0; i < N; i ++){
        for(int j = 0; j < N; j ++){
            fscanf(fin, "%c", &temp);
            adj[i][j] = temp - 48;
        }
        fscanf(fin, "%c", &temp);
    }

    //确定连通域
    int nclr = 1;
    for(int i = 0; i < N; i ++){
        if(!color[i]) dfs(i, nclr ++);
    }

    //建立图
    for(int i = 0; i < N; i ++){
        for(int j = 0; j < N; j ++){
            if(adj[i][j]){
                dist[i][j] = triangle(i, j); //勾股
            }else{
                dist[i][j] = DBL_MAX;
            }
        }
        dist[i][i] = 0;
    }    

    //对各个连通域算 floyd 
    for(int i = 1; i < nclr; i ++){
        diameter(i);
    }

    //不同连通域两个点连起来
    double minDia = DBL_MAX, dia;
    for(int i = 1; i < N; i ++){ 
        for(int j = 0; j < N; j ++){ //i和j连接组成新的连通域
            if(color[j] == color[i]) continue; 
            dia = max(diameters[color[i] - 1], diameters[color[j] - 1]); //可能是原连通域本身的直径
            dia = max(dia, (maxDist[i] + maxDist[j] + triangle(i, j))); //可能是含有这条新边的
            if(dia < minDia) minDia = dia;
        }
    }
    fprintf(fout, "%.6f\n", minDia);
    
    return 0;
}

性能

瞬间飞一般的效果:

Compiling...
Compile: OK

Executing...
   Test 1: TEST OK [0.000 secs, 4380 KB]
   Test 2: TEST OK [0.000 secs, 4380 KB]
   Test 3: TEST OK [0.000 secs, 4380 KB]
   Test 4: TEST OK [0.000 secs, 4380 KB]
   Test 5: TEST OK [0.000 secs, 4380 KB]
   Test 6: TEST OK [0.000 secs, 4380 KB]
   Test 7: TEST OK [0.014 secs, 4380 KB]
   Test 8: TEST OK [0.000 secs, 4380 KB]
   Test 9: TEST OK [0.000 secs, 4380 KB]

All tests OK.
Your program ('cowtour') produced all correct answers!  This is your
submission #9 for this problem.  Congratulations!

回顾一下,按照原来的写法,floyd 是实打实的 N ^ 3,那么150 ^ 3 = 3375000。然后,不同连通域任意两点连起来就要算一次,虽然不是 N ^ 2 次,也是这个级别的了。乖乖,150 ^ 5,逆天了,怪不得我等了两分钟。

修改之后,不同连通域连接起来就不用再算了。只是前面对于每个连通域算过一次 floyd 算法。这个当然是巨大的效果咯。

相关文章

  • PROB: cowtour

    题目来自 USACO题目翻译见 NOCOW 超时的暴搜 这道题一看也是很清楚了,求一个连通部分的直径,那就肯定是 ...

  • numpy篇

    Numpy 2018-05-21 numpy.prob:numpy.prob(a, axis=None, dtyp...

  • 负二项分布

    dnbinom(x, size, prob):返回发生x次失败事件的概率rnbinom(n, size, prob...

  • PROB: money

    题目来自 USACO题目翻译见 NOCOW 思路 用给定的货币系统,求某值有多少种组合。 几乎就是个背包问题了。只...

  • PROB: concom

    题目来自 USACO题目翻译见 NOCOW 最初的思路 看到这道题我是很懵的,就是让我自己手算,我也不知道该怎么算...

  • PROB: nocows

    题目来自 USACO题目翻译见 nocow dp 经点拨才有思路。二叉树这么漂亮的递归结构,想想也是很容易用前面已...

  • PROB: ttwo

    题目来自 USACO题目翻译见 NOCOW 思路 题目很软萌,看懂规则模拟就好了。可能是我代码能力下降太快了,写了...

  • PROB:zerosum

    题目来自 USACO题目翻译见 nocow 题目很简单,自己都说了让我们暴搜。 运算符最多 8 个,每个 3 种,...

  • 韦小绿

    很多很多 grov prob am

  • linear algebra week3 matrices

    Matrices, vectors, and solving simultaneous equation prob...

网友评论

      本文标题:PROB: cowtour

      本文链接:https://www.haomeiwen.com/subject/izhwjxtx.html