Codeforces Beta Round #8 C. Looking for Order 状压dp

题目链接:

http://codeforces.com/problemset/problem/8/C

C. Looking for Order

time limit per test:4 seconds
memory limit per test:512 megabytes
#### 问题描述 > Girl Lena likes it when everything is in order, and looks for order everywhere. Once she was getting ready for the University and noticed that the room was in a mess — all the objects from her handbag were thrown about the room. Of course, she wanted to put them back into her handbag. The problem is that the girl cannot carry more than two objects at a time, and cannot move the handbag. Also, if he has taken an object, she cannot put it anywhere except her handbag — her inherent sense of order does not let her do so. > > You are given the coordinates of the handbag and the coordinates of the objects in some Сartesian coordinate system. It is known that the girl covers the distance between any two objects in the time equal to the squared length of the segment between the points of the objects. It is also known that initially the coordinates of the girl and the handbag are the same. You are asked to find such an order of actions, that the girl can put all the objects back into her handbag in a minimum time period.

输入

The first line of the input file contains the handbag's coordinates xs, ys. The second line contains number n (1 ≤ n ≤ 24) — the amount of objects the girl has. The following n lines contain the objects' coordinates. All the coordinates do not exceed 100 in absolute value. All the given positions are different. All the numbers are integer.

输出

In the first line output the only number — the minimum time the girl needs to put the objects into her handbag.

In the second line output the possible optimum way for Lena. Each object in the input is described by its index number (from 1 to n), the handbag's point is described by number 0. The path should start and end in the handbag's point. If there are several optimal paths, print any of them.

样例输入

1 1
3
4 3
3 4
0 0

样例输出

32
0 1 2 0 3 0

题意

给你垃圾桶的位置和垃圾的位置,你每次能从垃圾桶出发捡一到两个垃圾然后回到垃圾桶,问如何规划使得捡垃圾所花时间最短(时间是以两点距离平方为基准)

题解

状压dp,有点像最优顶点配对问题的算法来处理,时间复杂度为n*2^n。

代码

#include<map>
#include<set>
#include<cmath>
#include<queue>
#include<stack>
#include<ctime>
#include<vector>
#include<cstdio>
#include<string>
#include<bitset>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
#include<functional>
using namespace std;
#define X first
#define Y second
#define mkp make_pair
#define lson (o<<1)
#define rson ((o<<1)|1)
#define mid (l+(r-l)/2)
#define sz() size()
#define pb(v) push_back(v)
#define all(o) (o).begin(),(o).end()
#define clr(a,v) memset(a,v,sizeof(a))
#define bug(a) cout<<#a<<" = "<<a<<endl
#define rep(i,a,b) for(int i=a;i<(b);i++)
#define scf scanf
#define prf printf

typedef long long  LL;
typedef vector<int> VI;
typedef pair<int,int> PII;
typedef vector<pair<int,int> > VPII;

const int INF=0x3f3f3f3f;
const LL INFL=0x3f3f3f3f3f3f3f3fLL;
const double eps=1e-8;
const double PI = acos(-1.0);

//start----------------------------------------------------------------------

const int maxn=24;

int dp[1<<maxn],n;
int pre[1<<maxn];
struct Node{
    int a,b;
    bool type;
}nds[1<<maxn];

PII pt[maxn];

int dis(int i,int j){
    int a=pt[i].X-pt[j].X;
    int b=pt[i].Y-pt[j].Y;
    return a*a+b*b;
}

int main() {
    scf("%d%d",&pt[0].X,&pt[0].Y);
    scf("%d",&n);
    for(int i=1;i<=n;i++){
        scf("%d%d",&pt[i].X,&pt[i].Y);
    }

    clr(dp,0x7f);
    dp[0]=0;
    clr(pre,-1);
    for(int stat=1;stat<(1<<n);stat++){
        for(int i=0;i<n;i++){
            if(stat&(1<<i)){
                    ///第一个单独不匹配
                if(dp[stat]>dp[stat^(1<<i)]+2*dis(0,i+1)){
                    dp[stat]=dp[stat^(1<<i)]+2*dis(0,i+1);
                    pre[stat]=stat^(1<<i);
                    nds[stat].a=i+1;
                    nds[stat].type=0;
                }
                dp[stat]=min(dp[stat],dp[stat^(1<<i)]+2*dis(0,i+1));

                    ///第一个和后面的某一个匹配
                for(int j=0;j<n;j++){
                    if(j!=i&&stat&(1<<j)){
                        int tmp=dp[stat^(1<<i)^(1<<j)]+dis(0,i+1)+dis(i+1,j+1)+dis(0,j+1);
                        if(dp[stat]>tmp){
                            dp[stat]=tmp;
                            pre[stat]=stat^(1<<i)^(1<<j);
                            nds[stat].a=i+1; nds[stat].b=j+1;
                            nds[stat].type=1;
                        }
                    }
                }
                //这个减枝非常关键!和最优顶点配对的做法一样
                break;
            }
        }
    }

    VI ans;
    ans.pb(0);
    int p=(1<<n)-1;
    while(p!=0){
        if(nds[p].type==0){
            ans.pb(nds[p].a);
        }else{
            ans.pb(nds[p].a);
            ans.pb(nds[p].b);
        }
        ans.pb(0);
        p=pre[p];
    }
    reverse(all(ans));
    prf("%d
",dp[(1<<n)-1]);

    rep(i,0,ans.sz()-1) prf("%d ",ans[i]);
    prf("%d
",ans[ans.sz()-1]);

    return 0;
}

//end-----------------------------------------------------------------------
原文地址:https://www.cnblogs.com/fenice/p/5846852.html