Cogs 6. 线型网络

6. 线型网络

★★☆   输入文件:linec.in   输出文件:linec.out   简单对比
时间限制:1 s   内存限制:256 MB

【问题描述】

有 N(N<=20)台 PC 放在机房内,现在要求由你选定一台 PC,用共N1条网线从这台机器开始一台接一台地依次连接他们,最后接到哪个以及连接的顺序也是由你选定的,为了节省材料,网线都拉直。求最少需要一次性购买多长的网线。(说白了,就是找出 N 的一个排列 P1P2P3..PN 然后 P1>P2>P3>...>PN 找出 |P1P2|+|P2P3|+...+|PN1PN| 长度的最小值)

【输入格式】

第一行 N ,下面 N 行,每行分别为机器的坐标(x,y)(x100<=x,y<=100)

【输出格式】

最小的长度,保留两位小数。

【输入样例】

3
0 0
1 1
1 -1

【输出样例】

2.83
/*
    ans[sta][i]点的集合为sta,且最后一个点是i的最短路 
*/
#include<cmath>
#include<cstdio>
#include<iostream>
using namespace std;
int n,num1[1<<20];
double x[21],y[21],dis[21][21],ans[1<<20][21];
int main(){
    //freopen("Cola.txt","r",stdin);
    freopen("linec.in","r",stdin);freopen("linec.out","w",stdout);
    scanf("%d",&n);
    for(int i=1;i<=n;i++)scanf("%lf%lf",&x[i],&y[i]);
    for(int i=1;i<=n;i++)
        for(int j=1;j<=n;j++){
            if(i==j)continue;
            dis[i][j]=sqrt((x[i]-x[j])*(x[i]-x[j])+(y[i]-y[j])*(y[i]-y[j]));
        }
    for(int i=0;i<=n;i++)num1[1<<i]=i+1;
    for(int i=1;i<(1<<n);i++){
        if(!num1[i]){
            for(int j=i;j;j-=j&-j){
                ans[i][num1[j&-j]]=0x3f3f3f3f;
                for(int k=i-(j&-j);k;k-=(k&-k))
                    if (ans[i][num1[j&-j]]>ans[i-(j&-j)][num1[k&(-k)]]+dis[num1[j&-j]][num1[k&-k]])
                        ans[i][num1[j&-j]]=ans[i-(j&-j)][num1[k&(-k)]]+dis[num1[j&-j]][num1[k&-k]];
            }
        }
    }
    double Ans=0x7fffffff;
    for(int i=1;i<=n;i++)Ans=min(Ans,ans[(1<<n)-1][i]);
    printf("%.2lf",Ans);
}

 

posted @ 2017-10-23 16:42  Echo宝贝儿  阅读(179)  评论(0编辑  收藏  举报