北京集训:20180320

爆零蒟蒻欢乐多......

T1:


怎么又是构造题啊......
算了扔了不看了......
事实证明这种选择似乎是正确的......
我们直接来看正解:


考虑为什么我们把不同子树的叶子节点连接起来是正确的。
考虑我们割掉一个点,如果他是叶子节点,则原图显然还是联通的。
而如果他不是叶子节点,则他的子树可以通过那些叶子节点连的边通过根节点的其他子树与根节点互连。
所以这个东西显然是正确的,而题解取的max,是因为我们每个叶子节点都必须要连边,
而对于每一个点,分开他形成的叶子节点的子树们都需要互连。
考虑怎么构造,我们先找到重心,并dfs出以他为根的各个子树中的叶子节点。
然后我们钦定一个节点作为hub,其余节点两两配对,如果无法配对则与hub连接。
而为了避免同一个子树内的节点不得不相连(显然这样不优),我们需要把这些子树按照大小排序,连边的时候按照一个从大到小启发式的顺序连接。
为什么,因为重心的每一个子树中的叶子节点个数一定不大于(fulleaf+1)/2。(否则我们可以调整重心位置是吧)
正解代码:

  1 #include<iostream>
  2 #include<cstdio>
  3 #include<cstring>
  4 #include<algorithm>
  5 #include<vector>
  6 #include<queue>
  7 #define debug cout
  8 using namespace std;
  9 const int maxn=5e5+1e2;
 10 
 11 int s[maxn],t[maxn<<1],nxt[maxn<<1],deg[maxn],cnt;
 12 int mx[maxn],sizleaf[maxn],full;
 13 vector<int> leafs[maxn];
 14 priority_queue<pair<unsigned,int> > pq;
 15 vector<pair<int,int> > ans;
 16 int n;
 17 
 18 inline void addedge(int from,int to) {
 19     ++deg[to];
 20     t[++cnt] = to , nxt[cnt] = s[from] , s[from] = cnt;
 21 }
 22 inline void pre(int pos,int fa) {
 23     mx[pos] = sizleaf[pos] = ( deg[pos] == 1 );
 24     for(int at=s[pos];at;at=nxt[at]) if( t[at] != fa ) {
 25         pre(t[at],pos) , sizleaf[pos] += sizleaf[t[at]] , 
 26         mx[pos] = max(mx[pos],sizleaf[t[at]]);
 27     }
 28 }
 29 inline void dfs(int pos,int fa) {
 30     if( deg[pos] == 1 ) return leafs[full].push_back(pos);
 31     for(int at=s[pos];at;at=nxt[at]) if( t[at] != fa ) dfs(t[at],pos);
 32 }
 33 
 34 inline bool cmp(const vector<int> &a,const vector<int> &b) {
 35     return a.size() > b.size();
 36 }
 37 inline void repush(int p) {
 38     leafs[p].pop_back();
 39     if( leafs[p].size() ) pq.push(make_pair(leafs[p].size(),p));
 40 }
 41 inline void solve_case() {
 42     pre(1,-1);
 43     const int fulleaf = sizleaf[1];
 44     int cent = 1;
 45     for(int i=1;i<=n;i++) {
 46         mx[i] = max( mx[i] , fulleaf - sizleaf[i] );
 47         if( mx[i] < mx[cent] ) cent = i;
 48     }
 49     for(int at=s[cent];at;at=nxt[at]) {
 50         ++full;
 51         dfs(t[at],cent);
 52     }
 53     sort(leafs+1,leafs+1+full,cmp);
 54     pq.push(make_pair(leafs[1].size(),1));
 55     const int tp = leafs[1].back();
 56     for(int i=2;i<=full;i++)
 57         if( leafs[i].size() ) {
 58             if( pq.empty() ) {
 59                 ans.push_back(make_pair(leafs[i].back(),tp));
 60             } else {
 61                 const int j = pq.top().second; pq.pop();
 62                 ans.push_back(make_pair(leafs[i].back(),leafs[j].back()));
 63                 repush(j);
 64             }
 65             repush(i);
 66         }
 67     while( pq.size() ) {
 68         const int i = pq.top().second; pq.pop();
 69         if( !pq.size() ) {
 70             ans.push_back(make_pair(leafs[i].back(),tp));
 71         } else {
 72             const int j = pq.top().second; pq.pop();
 73             ans.push_back(make_pair(leafs[i].back(),leafs[j].back()));
 74             repush(j);
 75         }
 76         repush(i);
 77     }
 78     printf("%u\n",(unsigned)ans.size());
 79     for(unsigned i=0;i<ans.size();i++) printf("%d %d\n",ans[i].first,ans[i].second);
 80 }
 81 
 82 inline void reset() {
 83     #define mem(x) memset(x,0,sizeof(*x)*(n+1))
 84     mem(s) , mem(deg) , mem(mx) , mem(sizleaf);
 85     cnt = full = 0 , ans.clear();
 86     for(int i=0;i<=n;i++) leafs[i].clear();
 87     while( pq.size() ) pq.pop();
 88 }
 89 
 90 int main() {
 91     static int T;
 92     scanf("%d",&T);
 93     while(T--) {
 94         scanf("%d",&n) , reset();
 95         for(int i=1,a,b;i<n;i++) {
 96             scanf("%d%d",&a,&b) ,
 97             addedge(a,b) , addedge(b,a);
 98         }
 99         solve_case();
100     }
101     return 0;
102 }
View Code


T2:


看起来很可做的样子啊。
这种期望一看就是先算出总和再除总方案数......
考虑我们怎么计算,20分可以爆搜......
然后看到40分的部分分,考虑n^2怎么做。
我们枚举最大一组的值,然后求让最大的一组的值正好为我们枚举的值的方案数。
然后发现我不会做,但是我们可以求出最大一组<=某个值的方案数,然后差分一下就能得到我们想要的东西。
方法就是我们从后向前遍历这些数字,考虑大于n/2的数字是怎么匹配的。
那么他能匹配的数字的限制就是当前值v-他本身的大小i,当然还要减去比他大的数字用掉的数字数量used(显然他们的方案一定是这个数的方案的子集)。
这样发现你会WA,因为我们可能会计算到自己对自己或者自己对比自己大的数的匹配......
好的,我们让前面的限制与 n-大于等于自己的数字的个数 取min就好了......
于是你get到了40分(我在考场上也就能想到这些了QAQ)。
其实正解已经很接近了......
仔细分析刚才的过程,是不是有一些数他的匹配方案的数量相同?还有一些数可以随意配对?
我们考虑按照限定的最大值v分组,显然>v/2的数字一定要与<v/2的数字配对。
这样的话,这些数的配对方案数就是(n-v/2)^(v/2),注意这里的/2是下取整。
为什么这是对的?考虑后面的数每递减1,前面可选的区间递增1,而更大的数又用掉了一个,所以可用的数的数量不变。
考虑剩下的数怎么配对,假设剩下的数为x个,显然x一定是一个偶数(你告诉我 n-(v/2)*2 怎么能是奇数?),且他们可以两两配对。
那么方案数为x!/((x/2)!*2^x)。
为什么?我们先枚举全排列,钦定相邻的两个数配对,那么我们每个组内无序,所有组间无序,我们除去他们的排列即可。
考场40分代码:

 1 #include<cstdio>
 2 #include<algorithm>
 3 typedef long long int lli;
 4 const int maxn=5e3+1e2;
 5 const int mod=1e9+7;
 6 
 7 lli ways[maxn<<1],sum,full;
 8 int n;
 9 
10 inline lli getinv(lli base) {
11     lli ret = 1;
12     int tim = mod - 2;
13     while( tim ) {
14         if( tim & 1 ) ret = ret * base % mod;
15         if( tim >>= 1 ) base = base * base % mod;
16     }
17     return ret;
18 }
19 
20 inline lli calc(int x) {
21     lli ret = 1;
22     int used = 0;
23     for(int i=n;i>(n>>1);i--) {
24         ret = ret * ( std::min(n-used-1,x-i) - used ) % mod;
25         ++used;
26     }
27     return ret;
28 }
29 
30 int main() {
31     scanf("%d",&n);
32     for(int i=n+1;i<n+n;i++) ways[i] = calc(i);
33     for(int i=n+1;i<n+n;i++) {
34         ( sum += ( ways[i] - ways[i-1] + mod ) % mod * i % mod ) %= mod ,
35         ( full += ( ways[i] - ways[i-1] + mod ) % mod ) %= mod;
36     }
37     printf("%lld\n",sum*getinv(full)%mod);
38     return 0;
39 }
View Code

正解代码:

 1 #include<iostream>
 2 #include<cstdio>
 3 #include<cstring>
 4 #include<algorithm>
 5 #define debug cout
 6 typedef long long int lli;
 7 using namespace std;
 8 const int maxn=5e5+1e2,lim=5e5;
 9 const int mod=1e9+7;
10 
11 lli sum,full;
12 lli fac[maxn],inv[maxn],in[maxn<<1];
13 inline lli fastpow(lli base,int tim) {
14     lli ret = 1;
15     while( tim ) {
16         if( tim & 1 ) ret = ret * base % mod;
17         if( tim >>= 1 ) base = base * base % mod;
18     }
19     return ret;
20 }
21 
22 inline void pre() {
23     *fac = 1;
24     for(int i=1;i<=lim;i++) fac[i] = fac[i-1] * i  % mod;
25     inv[lim] = fastpow(fac[lim],mod-2);
26     for(int i=lim;i;i--) inv[i-1] = inv[i] * i % mod;
27 }
28 inline lli f(lli x) {
29     return fac[x] * inv[x>>1] % mod * fastpow(inv[2],x>>1) % mod;
30 }
31 inline lli g(int n,int v) {
32     int l = n - ( v >> 1 );
33     return fastpow(v-n,l) * f(n-(l<<1)) % mod;
34 }
35 
36 int main() {
37     static int n;
38     scanf("%d",&n) , pre();
39     for(int i=n+1;i<n<<1;i++) in[i] = g(n,i);
40     for(int i=n+1;i<n<<1;i++) {
41         const lli siz = ( in[i] - in[i-1] + mod ) % mod;
42         ( sum += siz * i % mod ) %= mod ,
43         ( full += siz ) %= mod;
44     }
45     printf("%lld\n",sum*fastpow(full,mod-2)%mod);
46     return 0;
47 }
View Code


T3:


显然有一个O(n^2)的暴力区间DP......
然后前缀和in[i]+=in[i-1]写成了in[i]+=in[i+1],爆零了......
正解是一个复杂度玄学的东西:


考虑我们每次的拓展过程,遍历整个可行的区间,如果可以拓展则拓展,显然这种贪心拓展是不会让答案变得更劣的。
然后发现我们无法贪心拓展了,这样我们就必须钦定一边的边界移动。
首先判定一定无解的情况,然后当前的情况就可能有解。
然后我们进行移动,而如果移动后变得非法,则直接返回无解。
为什么么这是正确的?显然l--和r++的两个条件不可能同时满足,否则可以贪心拓展。
(上面那句话为我口胡,其实正确性我也不是很明白......)
考场爆零代码(修改前缀和后有30分):

 1 #include<cstdio>
 2 #include<cstring>
 3 typedef long long int lli;
 4 const int maxn=1e3+1e2;
 5 
 6 lli in[maxn];
 7 bool f[maxn][maxn];
 8 
 9 int main() {
10     static int T,n,k;
11     scanf("%d",&T);
12     while(T--) {
13         scanf("%d%d",&n,&k) , memset(f,0,sizeof(f)) , f[k][k] = 1;
14         for(int i=1;i<=n;i++) scanf("%lld",in+i) , in[i] += in[i+1];
15         for(int len=2;len<=n;len++)
16             for(int st=1,ed;(ed=st+len-1)<=n;st++)
17                 if( in[ed] - in[st] <= 0 )
18                     f[st][ed] |= f[st+1][ed] , f[st][ed] |= f[st][ed-1];
19         puts(f[1][n]?"Yes":"No");
20     }
21     return 0;
22 }
View Code

正解代码:

 1 #include<cstdio>
 2 typedef long long int lli;
 3 const int maxn=1e5+1e2;
 4 
 5 lli in[maxn];
 6 lli prvmi[maxn],prvmx[maxn],sufmi[maxn],sufmx[maxn];
 7 int n,k;
 8 
 9 inline void pre(lli* dst,int st,int delta,lli (calc)(lli,lli)) {
10     dst[st] = in[st];
11     for(int i=st+delta;1<=i&&i<=n;i+=delta)
12         dst[i] = calc(in[i],dst[i-delta]);
13 }
14 inline lli _min(lli a,lli b) {
15     return a < b ? a : b;
16 }
17 inline lli _max(lli a,lli b) {
18     return a < b ? b : a;
19 }
20 
21 inline bool calc() {
22     for(int i=1;i<=n;i++) in[i] += in[i-1];
23     pre(prvmi,1,1,_min) , pre(prvmx,1,1,_max) ,
24     pre(sufmi,n,-1,_min) , pre(sufmx,n,-1,_max);
25     
26     int l = k , r = k;
27     while( l != 1 || r != n ) {
28         bool failed = 0;
29         while( !failed ) {
30             failed = 1;
31             for(int i=l-1;i&&in[i]>=in[r];i--)
32                 if( in[i] >= in[l] ) l = i , failed = 0;
33             for(int i=r+1;i<=n&&in[i]<=in[l];i++)
34                 if( in[i] <= in[r] ) r = i , failed = 0;
35         }
36         if( prvmx[l] < sufmi[r] || in[l] < in[r] ) return 0; // Not fixed
37         if( l == 1 && r == n ) return 1;
38         if( l == 1 ) ++r; // Do some operation which may be illeagle .
39         else if( r == n ) --l;
40         else {
41             if( prvmx[l-1] < sufmx[r+1] && prvmi[l-1] < sufmi[r+1] ) return 0; // No way to extend .
42             if( prvmx[l-1] >= sufmx[r+1] ) --l;
43             else ++r;
44         }
45         if( prvmx[l] < sufmi[r] || in[l] < in[r] ) return 0;
46     }
47     return 1; // It may be (1,n] after extended .
48 }
49 
50 int main() {
51     static int T;
52     scanf("%d",&T);
53     while(T--) {
54         scanf("%d%d",&n,&k);
55         for(int i=1;i<=n;i++) scanf("%lld",in+i);
56         puts(calc()?"Yes":"No");
57     }
58     return 0;
59 }
View Code


解放什么的,真的存在吗?

https://music.163.com/#/song?id=473403621

 

posted @ 2018-03-20 21:12  Cmd2001  阅读(157)  评论(0编辑  收藏  举报