Cogs 6. 线型网络
6. 线型网络
★★☆ 输入文件:linec.in
输出文件:linec.out
简单对比
时间限制:1 s 内存限制:256 MB
【问题描述】
有 N(N<=20)台 PC 放在机房内,现在要求由你选定一台 PC,用共N−1条网线从这台机器开始一台接一台地依次连接他们,最后接到哪个以及连接的顺序也是由你选定的,为了节省材料,网线都拉直。求最少需要一次性购买多长的网线。(说白了,就是找出 N 的一个排列 P1P2P3..PN 然后 P1−>P2−>P3−>...−>PN 找出 |P1P2|+|P2P3|+...+|PN−1PN| 长度的最小值)
【输入格式】
第一行 N ,下面 N 行,每行分别为机器的坐标(x,y)(x为实数−100<=x,y<=100)
【输出格式】
最小的长度,保留两位小数。
【输入样例】
3
0 0
1 1
1 -1
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); }