【Codeforces #169 Div2】Solutions

  转载请注明出处:http://www.cnblogs.com/Delostik/archive/2013/02/25/2932004.html

  做了仨题还掉分简直没天理了TAT...不过第四题没写对真是可惜,看来是变不紫了...

【A.Lunch Rush】

  http://www.codeforces.com/contest/276/problem/A

  题目大意:求ans=max{f[i]-max(0,t[i]-k)}

View Code
 1 #include <iostream>
 2 using namespace std;
 3 
 4 int n,k,ans=-2147483640,a,b;
 5 
 6 int main(){
 7     cin>>n>>k;
 8     for(int i=1;i<=n;i++){
 9         cin>>a>>b;
10         ans=max(ans,a-max(0,(b-k)));
11     }
12     cout<<ans<<endl;
13     return 0;
14 }

【B.Little Girl and Game】

  http://www.codeforces.com/contest/276/problem/B

  题目大意:给定字符串s,两人轮流操作,每次从中拿走一个字符。当某一方在拿走之前可将字符串重组成回文串即获胜。问是否先手必胜。

  构成回文串的条件是:最多存在一个字母出现了奇数次,这样出现偶数次的字母就可以对称放在两侧。设当前出现奇数次的字母有n个:

  当n=0、1时为先手必胜态;

  n=2时先手必败;

  n=3时,先手可以通过拿走一个出现奇数次的字母,使对方出现必败态;

  n=4时,先手可以拿偶数次的字母使n'=5,但对方也可以拿这个字母使先手继续面对n=4;但先手如果拿奇数次字母使n'=3,则对方面临必胜态。

  综上......这是个水题......

View Code
 1 #include <iostream>
 2 #include <string>
 3 using namespace std;
 4 
 5 string s;
 6 int cnt,a[30];
 7 
 8 int main(){
 9     cin>>s;
10     int len=s.length();
11     for(int i=0;i<len;i++)
12         a[s[i]-'a']++;
13     for(int i=0;i<26;i++)
14         if(a[i]&1) cnt++;
15     if(!cnt || cnt&1) cout<<"First"<<endl;
16     else cout<<"Second"<<endl;
17     return 0;
18 }


【C.Little Girl and Maximum Sum】

  http://www.codeforces.com/contest/276/problem/C

  题目大意:有若干对区间和的查询,问如何重组数组使查询结果的和最大。

  显然查询次数最多的点让他权值最大就好了(排序不等式),关键是怎么统计每个点的查询次数...

  方法1:前缀和,这是博主推荐的方法

View Code
 1 #include <iostream>
 2 #include <algorithm>
 3 using namespace std;
 4 
 5 int n,q,l,r;
 6 long long a[200010],t[200010],ans;
 7 
 8 int main(){
 9     cin>>n>>q;
10     for(int i=0;i<n;i++)
11         cin>>a[i];
12     sort(a,a+n);
13     for(int i=0;i<q;i++){
14         cin>>l>>r;
15         t[l-1]++,t[r]--;
16     }
17     for(int i=1;i<n;i++)
18         t[i]+=t[i-1];
19     sort(t,t+n);
20     for(int i=0;i<n;i++)
21         ans+=a[i]*t[i];
22     cout<<ans;
23     return 0;
24 }

  方法2:线段树,要是非想这么做博主当然也不拦着...

View Code
 1 #include <iostream>
 2 #include <cstdio>
 3 #include <cstdlib>
 4 #include <cstring>
 5 #include <algorithm>
 6 #define lch(x) x<<1
 7 #define rch(x) (x<<1)+1
 8 using namespace std;
 9 
10 int n,q,l,r,c,a[200010],b[200010];
11 char sym[2];
12 long long ans;
13 
14 struct T_node{
15     int sum,add;
16     int l,r,size;
17 }tree[800010];
18 
19 void build(int v,int l,int r){
20     tree[v].l=l,tree[v].r=r;
21     if(l==r) tree[v].size=1;
22     else{
23         build(lch(v),l,(l+r)>>1);
24         build(rch(v),((l+r)>>1)+1,r);
25         tree[v].sum=tree[lch(v)].sum+tree[rch(v)].sum;
26         tree[v].size=tree[lch(v)].size+tree[rch(v)].size;
27     }
28 }
29 
30 void pushdown(int v){
31     if(tree[v].l!=tree[v].r){
32         tree[lch(v)].add+=tree[v].add,tree[lch(v)].sum+=tree[v].add*tree[lch(v)].size;
33         tree[rch(v)].add+=tree[v].add,tree[rch(v)].sum+=tree[v].add*tree[rch(v)].size;
34     }
35     tree[v].add=0;
36 }
37 
38 void modify(int v,int l,int r){
39     pushdown(v);
40     if(l==tree[v].l && r==tree[v].r) tree[v].add+=1,tree[v].sum+=tree[v].size;
41     else{
42         int mid=(tree[v].l+tree[v].r)>>1;
43         if(r<=mid) modify(lch(v),l,r);
44         else if(l>mid) modify(rch(v),l,r);
45         else{
46             modify(lch(v),l,mid);
47             modify(rch(v),mid+1,r);
48         }
49         tree[v].sum=tree[lch(v)].sum+tree[rch(v)].sum;
50     }
51 }
52         
53 int query(int v,int x){
54     pushdown(v);
55     if(x==tree[v].l && x==tree[v].r) return tree[v].sum;
56     int mid=(tree[v].l+tree[v].r)>>1;
57     if(x<=mid) return query(lch(v),x);
58     else if(x>mid) return query(rch(v),x);
59 }
60 
61 int main(){
62     cin>>n>>q;
63     for(int i=1;i<=n;i++)
64         cin>>a[i];
65     sort(a+1,a+1+n);
66     build(1,1,n);
67     for(int t=1;t<=q;t++){
68         cin>>l>>r;
69         modify(1,l,r);
70     }
71     for(int i=1;i<=n;i++)
72         b[i]=query(1,i);
73     sort(b+1,b+1+n);
74     for(int i=1;i<=n;i++){
75         ans+=(long long)a[i]*(long long)b[i];
76     }
77     cout<<ans<<endl;
78     return 0;
79 }


【D.Little Girl and Maximum XOR】

  http://www.codeforces.com/contest/276/problem/D

  题目大意:在区间[l,r]中找到ans=max{a^b},l≤a,b≤r。

  先找打l和r最高的不同的那一位,设a≥b,那么a一定是1XXXXXXX,b一定是0XXXXXXX,在区间[l,r]内最优解便是a=1000000,b=01111111  = =!

View Code
 1 #include <iostream>
 2 using namespace std;
 3 
 4 long long l,r,i;
 5 
 6 int main(){
 7     cin>>l>>r;
 8     for(i=63;i>=0;i--)
 9         if((l^r)&(1LL<<i)) break;
10     cout<<(1LL<<(i+1))-1;
11     return 0;
12 }


【E.Little Girl and Problem on Trees】

  http://www.codeforces.com/contest/276/problem/E

  题目大意:给定一棵树,操作1:将以节点v为中心,与v距离不超过d的节点权值都+w;操作2,查询某一节点权值。

  题目中有一句话说每个节点的度不超过2,也就意味着,去掉根节点后,树将变成若干条链。对每一条链分别进行维护,使用树状数组或线段树。

  但是可能出现需要更新很多条链的情况,这样一来每次操作就完全退化成O(n),所以需要一个额外的树状数组,维护相同深度的节点的增加值。

  感谢wyl8899讲解~另外下面代码没A掉,大概是指针比较慢的缘故,改成数组模拟就能A了。。。

View Code
 1 #include <cstdio>
 2 #include <vector>
 3 #define mm 200010
 4 #define mn 100010
 5 #define lowbit(x) ((x)&(-x))
 6 using namespace std;
 7 
 8 struct EDGE{
 9     int pnt;
10     EDGE *pre;
11     EDGE(){}
12     EDGE(int _pnt,EDGE *_pre):pnt(_pnt),pre(_pre){}
13 }Edge[mm],*SP=Edge,*edge[mn];
14 
15 inline void addedge(int a,int b){
16     edge[a]=new(++SP)EDGE(b,edge[a]);
17 }
18 
19 int n,q,x,y,v,w,d,tot,opt,root,bng[mn],pos[mn],len[mn],fa[mn],son[mn];
20 vector<int> tree[mn];
21 
22 void add(vector<int> &c,int x,int key){
23     for(;x<c.size();x+=lowbit(x)) c[x]+=key;
24 }
25 
26 int get(vector<int> c,int x){
27     int res=0;
28     for(;x>0;x-=lowbit(x)) res+=c[x];
29     return res;
30 }
31 
32 void dfs(int x,int father){
33     fa[x]=father;
34     for(EDGE *j=edge[x];j;j=j->pre)
35         if(j->pnt!=father){
36             son[x]=j->pnt;
37             dfs(j->pnt,x);
38         }
39 }
40 
41 void build_tree(){
42     for(EDGE *j=edge[1];j;j=j->pre,tot++){
43         dfs(j->pnt,1);
44         for(int i=j->pnt;i;i=son[i])
45             bng[i]=tot,pos[i]=++len[tot];
46         tree[tot].resize(len[tot]+1);
47     }
48     tree[tot].resize(n+1);
49 }
50 
51 int main(){
52     scanf("%d%d",&n,&q);
53     for(int i=1;i<n;i++){
54         scanf("%d%d",&x,&y);
55         addedge(x,y),addedge(y,x);
56     }
57     build_tree();
58     while(q--){    
59         scanf("%d",&opt);
60         if(opt){
61             scanf("%d",&v);
62             if(v==1) printf("%d\n",root);
63             else printf("%d\n",get(tree[bng[v]],pos[v])+get(tree[tot],pos[v]));
64         }else{
65             scanf("%d%d%d",&v,&w,&d);
66             if(v==1){
67                 root+=w;
68                 add(tree[tot],1,w);
69                 add(tree[tot],d+1,-w);
70             }else{
71                 add(tree[bng[v]],max(pos[v]-d,1),w);    //向上
72                 add(tree[bng[v]],pos[v]+d+1,-w);        //向下
73                 if(d>=pos[v]){
74                     root+=w;
75                     if(d>pos[v]){
76                         add(tree[tot],1,w);add(tree[tot],d-pos[v]+1,-w);
77                         add(tree[bng[v]],1,-w);add(tree[bng[v]],d-pos[v]+1,w);
78                     }
79                 }
80             }
81         }
82     }
83     return 0;
84 }

  

原文地址:https://www.cnblogs.com/Delostik/p/2932004.html