题目描述
房间里放着 n 块奶酪。一只小老鼠要把它们都吃掉,问至少要跑多少距离?老鼠一开始在 (0,0) 点处。
输入格式
第一行有一个整数,表示奶酪的数量 n。
第 2 到第 (n+1) 行,每行两个实数,第 (i+1) 行的实数分别表示第 i 块奶酪的横纵坐标
输出格式
输出一行一个实数,表示要跑的最少距离,保留 2 位小数。
输入输出样例
输入
4
1 1
1 -1
-1 1
-1 -1
输出
7.41
//暴力DFS
#include <bits/stdc++.h>
using namespace std;
typedef long long LL;
const int N=25;
int n;
double a[N],b[N];//第i个点的横纵坐标
bool vis[N];
double ans=1e9;
double getsum(int i,int j){
return sqrt((a[i]-a[j])*(a[i]-a[j])+(b[i]-b[j])*(b[i]-b[j]));
}
void dfs(int step,double sum,int cur){
if(sum>=ans||step>n)return;//最优性剪枝
if(step==n){
ans=min(ans,sum);
return;
}
for(int i=1;i<=n;i++){//枚举所有点
if(!vis[i]){
vis[i]=true;
dfs(step+1,sum+getsum(cur,i),i);
vis[i]=false;
}
}
}
int main()
{
cin>>n;
for(int i=1;i<=n;i++){
cin>>a[i]>>b[i];
}
dfs(0,0,0);
printf("%.2f",round(ans*100)/100);
return 0;
}
#include <bits/stdc++.h>
using namespace std;
typedef long long LL;
const int N=25;
int n;
double a[N],b[N],dp[70000][17];//第i个点的横纵坐标
bool vis[N];
double ans=1e9;
double getsum(int i,int j){
return sqrt((a[i]-a[j])*(a[i]-a[j])+(b[i]-b[j])*(b[i]-b[j]));
}
void dfs(int step,double sum,int cur,int b){//b为状态
if(sum>=ans||step>n)return;//最优性剪枝
if(step==n){
ans=min(ans,sum);
return;
}
for(int i=1;i<=n;i++){//枚举所有点
if(!vis[i]){
int t=b+(1<<(i-1));//加号优先级更大
if(dp[t][i]!=0&&sum+getsum(cur,i)>=dp[t][i])continue;
dp[t][i]=sum+getsum(cur,i); //把每个状态记录下来
vis[i]=true;
dfs(step+1,sum+getsum(cur,i),i,t);
vis[i]=false;
}
}
}
int main()
{
cin>>n;
for(int i=1;i<=n;i++){
cin>>a[i]>>b[i];
}
dfs(0,0,0,0);
printf("%.2f",round(ans*100)/100);
return 0;
}