bzoj1237 [SCOI2008]配对

Description

你有n 个整数Ai和n 个整数Bi。你需要把它们配对,即每个Ai恰好对应一 个Bp[i]。要求所有配对的整数差的绝对值之和尽量小,但不允许两个相同的数配 对。例如A={5,6,8},B={5,7,8},则最优配对方案是5配8, 6配5, 8配7,配对整数 的差的绝对值分别为2, 2, 1,和为5。注意,5配5,6配7,8配8是不允许的,因 为相同的数不许配对。

Input

第一行为一个正整数n,接下来是n 行,每行两个整数Ai和Bi,保证所有 Ai各不相同,Bi也各不相同。

Output

输出一个整数,即配对整数的差的绝对值之和的最小值。如果无法配对,输 出-1。

Sample Input

3
3 65
45 10
60 25

Sample Output

32

HINT

1 <= n <= 10^5,Ai和Bi均为1到10^6之间的整数。

 

正解:$dp$。

因为每个数只出现一次,所以我们可以发现一个很显然的事,就是$a[i]$只会和$b[i-1],b[i],b[i+1]$匹配。

然后我们就可以$dp$了。设$f[i]$表示匹配了前$i$位的最小值。

于是$f[i]=max(f[i-1]+val,f[i-2]+val,f[i-3]+val)$,$val$为对应权值,当然还要判断一下合法性。

 

 1 #include <bits/stdc++.h>
 2 #define il inline
 3 #define RG register
 4 #define ll long long
 5 #define inf (1LL<<60)
 6 #define N (100010)
 7 
 8 using namespace std;
 9 
10 int a[N],b[N],p[3],n;
11 ll f[N];
12 
13 il int gi(){
14   RG int x=0,q=1; RG char ch=getchar();
15   while ((ch<'0' || ch>'9') && ch!='-') ch=getchar();
16   if (ch=='-') q=-1,ch=getchar();
17   while (ch>='0' && ch<='9') x=x*10+ch-48,ch=getchar();
18   return q*x;
19 }
20 
21 int main(){
22 #ifndef ONLINE_JUDGE
23   freopen("match.in","r",stdin);
24   freopen("match.out","w",stdout);
25 #endif
26   n=gi(); for (RG int i=1;i<=n;++i) a[i]=gi(),b[i]=gi();
27   sort(a+1,a+n+1),sort(b+1,b+n+1);
28   if (n==1){
29     if (a[1]==b[1]) puts("-1");
30     else cout<<abs(a[1]-b[1]); return 0;
31   }
32   for (RG int i=1;i<=n;++i){
33     f[i]=inf; if (a[i]!=b[i]) f[i]=min(f[i],f[i-1]+abs(a[i]-b[i]));
34     if (i>=2){
35       if (a[i]!=b[i] && a[i-1]!=b[i-1])
36     f[i]=min(f[i],f[i-2]+abs(a[i]-b[i])+abs(a[i-1]-b[i-1]));
37       else f[i]=min(f[i],f[i-2]+abs(a[i]-b[i-1])+abs(a[i-1]-b[i]));
38     }
39     if (i>=3){
40       p[0]=0,p[1]=1,p[2]=2;
41       do{
42     if (a[i]!=b[i-p[0]] && a[i-1]!=b[i-p[1]] && a[i-2]!=b[i-p[2]])
43       f[i]=min(f[i],f[i-3]+abs(a[i]-b[i-p[0]])+abs(a[i-1]-b[i-p[1]])+abs(a[i-2]-b[i-p[2]]));
44       }while (next_permutation(p,p+3));
45     }
46   }
47   cout<<f[n]; return 0;
48 }

 

posted @ 2017-10-24 22:27  wfj_2048  阅读(202)  评论(0编辑  收藏  举报