【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)}
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,则对方面临必胜态。
综上......这是个水题......
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:前缀和,这是博主推荐的方法
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:线段树,要是非想这么做博主当然也不拦着...
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 = =!
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了。。。
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 }