码迷,mamicode.com
首页 > 其他好文 > 详细

bzoj 2058+2059+2060 Usaco2010 Nov

时间:2016-08-16 14:43:07      阅读:148      评论:0      收藏:0      [点我收藏+]

标签:

三道金组比较容易的题目。。

2058

首先交换次数就是逆序对数,因为只能交换相邻的两数

先对原序列找逆序对数

用树状数组nlogn求出

 

然后O(n)依次求出其循环序列的逆序对数

比如 3 5 4 2 1

循环之后相对应的位置变成

2 4 3 1 5

表示第一个数到位置2,第二个数到位置4,第三个数到位置1 etc.

那么依次减一的那些数的逆序对数是不变的。只有1变成5的那个增加或减少了逆序对数

由于5是序列里最大的数,设其所在位置为i,因此增加了n-i对,减少了i-1对

这样总复杂度就是nlogn的

记得开Long long

 1 #include<stdio.h>
 2 #include<string.h>
 3 #include<algorithm>
 4 #define LL long long
 5 using namespace std;
 6 struct node{
 7     int v,id;
 8 }a[100010];
 9 LL tot,ans,p[100010];
10 int n;
11 
12 bool cmp(node a, node b){
13     return a.v<b.v;
14 }
15 
16 void add(int x){
17     while (x<=n){
18         p[x]++;
19         x+=x&-x; 
20     }
21 }
22 
23 LL sum(int x){
24     LL ret=0;
25     while (x){
26         ret+=p[x];
27         x-=x&-x;
28     }return ret;
29 }
30 
31 int main(){
32     scanf("%d", &n);
33     for (int i=1; i<=n; i++) scanf("%d", &a[i].v), a[i].id=i,p[i]=0;
34     tot=0;
35     for (int i=1; i<=n; i++){
36         add(a[i].v);
37         tot+=(LL)i-sum(a[i].v);
38     }
39     sort(a+1,a+1+n,cmp);
40     ans=tot;
41     for (int i=1; i<=n; i++){
42         int del=n-a[i].id-(a[i].id-1);
43         tot+=(LL)del;
44         ans=min(ans,tot);
45     }
46     printf("%lld\n", ans);
47     return 0;
48 }

2059

单调队列DP,降复杂度为O(NK)

分n阶段进行单调队列的优化dp

设dis=a[i].x-a[i-1].x

f[i][k]=cost[i]*k+min{f[i-1][j]-cost[i]*j+dis*j*j};

显然f[i][k]只与f[i-1][k]有关所以省略一维,并且先插入队列(此时为f[i-1][k]的值),然后再更新为f[i][k]

 1 #include<stdio.h>
 2 #include<string.h>
 3 #include<algorithm>
 4 #define LL long long
 5 using namespace std;
 6 const int maxn = 800010;
 7 struct node{
 8     int x,f;
 9     LL c;
10 }a[maxn];
11 struct que{
12     int v;
13     LL val;
14 }q[maxn*2];
15 int K,end,n;
16 LL f[maxn];
17 
18 bool cmp(node a, node b){
19     return a.x<b.x;
20 }
21 
22 int main(){
23     scanf("%d%d%d", &K, &end, &n);
24     for (int i=1; i<=n; i++) scanf("%d%d%lld", &a[i].x, &a[i].f, &a[i].c);
25     sort(a+1,a+1+n,cmp);
26     memset(f,0x3f3f3f3f,sizeof(f));
27     f[0]=0;
28     for (int i=1; i<=n; i++){
29         int head=0, tail=0;
30         LL dis=a[i].x-a[i-1].x;
31         q[tail].v=0,q[tail].val=0,tail++;
32         for (int j=1; j<=K; j++){
33             LL now=f[j]-(LL)j*a[i].c+dis*(LL)j*(LL)j;
34             while (head<tail && q[tail-1].val>=now) tail--;
35             q[tail].v=j, q[tail].val=now, tail++;
36             while (head<tail && q[head].v+a[i].f<j) head++;
37             f[j]=q[head].val+(LL)j*a[i].c;
38         }
39     }
40     printf("%lld\n", f[K]+(LL)K*(LL)K*(LL)(end-a[n].x));
41     return 0;
42 }

2060

树形dp的入门水题。。

f[u]表示选u的最大值

g[u]表示不选u的最大值

 1 #include<stdio.h>
 2 #include<string.h>
 3 #include<algorithm>
 4 using namespace std;
 5 const int maxn = 100010;
 6 struct node{
 7     int to,next;
 8 }e[maxn];
 9 int g[maxn],f[maxn],n,head[maxn],tot,u,v;
10 
11 void insert(int u, int v){
12     e[++tot].to=v; e[tot].next=head[u]; head[u]=tot;
13 }
14 
15 void dfs(int x, int fa){
16     g[x]=0; f[x]=1;
17     for (int i=head[x]; i!=-1; i=e[i].next){
18         int v=e[i].to;
19         if (v==fa) continue;
20         dfs(v,x);
21         g[x]+=max(g[v],f[v]);
22         f[x]+=g[v];
23     }
24 }
25 
26 int main(){
27     scanf("%d", &n);
28     tot=1; memset(head,-1,sizeof(head));
29     for (int i=1; i<n; i++){
30         scanf("%d%d", &u, &v);
31         insert(u,v); insert(v,u);
32     }
33     dfs(1,0);
34     printf("%d\n", max(g[1],f[1]));
35     return 0;
36 }

 

bzoj 2058+2059+2060 Usaco2010 Nov

标签:

原文地址:http://www.cnblogs.com/mzl120918/p/5776190.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!