PROB: cowtour

题目来自 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 算法。这个当然是巨大的效果咯。

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

推荐阅读更多精彩内容

  • 参见贪心算法——最短路径Dijkstra算法参见动态规划 目录 0.最短路径问题0.1 最短路径问题描述 0.1....
    王侦阅读 4,762评论 1 9
  • 第一章 绪论 什么是数据结构? 数据结构的定义:数据结构是相互之间存在一种或多种特定关系的数据元素的集合。 第二章...
    SeanCheney阅读 5,735评论 0 19
  • 课程介绍 先修课:概率统计,程序设计实习,集合论与图论 后续课:算法分析与设计,编译原理,操作系统,数据库概论,人...
    ShellyWhen阅读 2,249评论 0 3
  • Floyd 算法 简介 Floyd 算法又称为插点法,是一种利用动态规划的思想寻找给定的加权图中多源点之间最短路径...
    廖少少阅读 8,263评论 0 1
  • -DFS(Depth First Search):深度优先搜索 访问完一个顶点的所有邻接点之后,会按原路返回,对应...
    Spicy_Crayfish阅读 2,823评论 1 0