2017.8.29 达哥的随单题...

给出n个正整数a1,a2…an和一个质数mod.一个变量x初始为1.进行m次操作.每次在n个数中随机选一个ai,然后x=x*ai%mod.问m次操作之后x的取值的期望.

答案一定可以表示成a/b的精确分数形式.a和b可能很大,所以只需要输出a*(b^(10^9+5))模10^9+7的结果.

n<=10^5  m<=10^9  mod<=10^3  1<=ai<mod

solution

考试时明白了原根是什么,只可惜我就是脑子抽风,不会dp...

80分:

定义 f[i][j] 表示乘到第i次,x=j的概率 (当然概率是%1e9意义下的概率)

由于m很大,而状态转移可以写成矩阵乘的形式,所以可以用矩阵乘法优化到O(mod^3*logm)

100分:

O(n^2)求出mod的原根(原根rt:rt的1次方,2次方,...,mod-1次方可以取到1~mod-1的所有数)

所有的ai都可以用rt^k表示

所以这时 f[i][j] 表示乘到第i次,rt^j的概率,然后矩阵乘把状态矩阵输出,发现是循环矩阵

之后就优化到了O(n^2*logm)

  1 #include <cstdio>
  2 #include <cstring>
  3 #include <iostream>
  4 #define mem(a,b) memset(a,b,sizeof(a))
  5 #define ll long long
  6 using namespace std;
  7 const int MOD=1000000007;
  8 const int N=100066;
  9  
 10 int n,m,mod;
 11 int a2[N],num[1066],dui[1066];
 12 int rt,mod_1;
 13 ll a[1066][1066],f[1066],temp[1066][1066],gai;
 14 bool flag[1066];
 15  
 16 ll mi(ll a,int ci,int c)
 17 {
 18   ll ans=1;
 19   while(ci)
 20   {
 21     if(ci&1)
 22       ans=ans*a%c;
 23     a=a*a%c;
 24     ci>>=1;
 25   }
 26   return ans;
 27 }
 28  
 29 int get_rt()
 30 {
 31   int judge,temp;
 32   for(int i=1;i<mod;++i)
 33   {
 34     judge=0;temp=1;
 35     mem(flag,0);
 36     for(int j=1;j<mod;++j)
 37     {
 38       temp=temp*i%mod;
 39       if(flag[temp])
 40       {
 41         judge=1;
 42         break;
 43       }
 44       flag[temp]=1;
 45     }
 46     if(!judge)
 47       return i;
 48   }
 49 }
 50  
 51 void get_a2_dui()
 52 {
 53   int temp=1;
 54   for(int i=1;i<mod;++i)
 55   {
 56     temp=temp*rt%mod;
 57     dui[temp]=i;
 58   }
 59 }
 60  
 61 void get_num()
 62 {
 63   for(int i=1;i<=n;++i)
 64     ++num[dui[a2[i]]];
 65 }
 66  
 67 void build_a()
 68 {
 69   for(int i=0;i<mod;++i)
 70     for(int j=1;j<mod;++j)
 71       a[i][(i+j)%mod_1]=(a[i][(i+j)%mod_1]+gai*num[j]%MOD)%MOD;
 72 }
 73  
 74 void cheng()
 75 {
 76   f[0]=1;
 77   while(m)//横着是<mod-1,纵<mod
 78   {
 79     if(m&1)
 80     {
 81       for(int i=0;i<=mod;++i)
 82       {
 83         temp[0][i]=0;
 84         for(int k=0;k<=mod;++k)
 85           temp[0][i]=(temp[0][i]+f[k]*a[k][i]%MOD)%MOD;
 86       }
 87       for(int i=0;i<=mod;++i)
 88         f[i]=temp[0][i];
 89     }
 90     for(int i=0;i<=mod;++i)
 91     {
 92       temp[0][i]=0;
 93         for(int k=0;k<=mod;++k)
 94           temp[0][i]=(temp[0][i]+a[0][k]*a[k][i]%MOD)%MOD;
 95     }
 96      
 97     for(int i=0;i<=mod;++i)
 98       a[0][i]=temp[0][i];
 99      
100     for(int i=1;i<=mod;++i)
101     {
102       a[i][0]=a[i-1][mod_1-1];
103       for(int j=1;j<mod_1;++j)
104         a[i][j]=a[i-1][j-1];
105     }
106     /*for(int i=0;i<mod;++i)
107     {
108       for(int j=0;j<mod_1;++j)
109         printf("%lld ",a[i][j]);
110       printf("\n");
111     }
112     printf("\n");*/
113     m>>=1;
114   }
115 }
116  
117 int main(){
118      
119   //freopen("rand4.in","r",stdin);
120   //freopen("2.txt","w",stdout);
121    
122   scanf("%d%d%d",&n,&m,&mod);
123   mod_1=mod-1;
124   for(int i=1;i<=n;++i)
125     scanf("%d",&a2[i]);
126   gai=mi(n,MOD-2,MOD);
127   rt=get_rt();
128   get_a2_dui();
129   get_num();
130   build_a();
131   cheng();
132   ll ans=0,temp=1;
133   ans=(ans+temp*f[0]%MOD)%MOD;
134   for(int i=1;i<=mod;++i)
135   {
136     temp=temp*rt%mod;
137     ans=(ans+temp*f[i]%MOD)%MOD;
138   }
139   cout<<ans;
140 }
code

给一个n个点的树,有n-1条边,每一个点有一个权值,定义a[i]为i这个点的权值,定义dis(i,j)为i到j的树上距离,dis(i,i)=0

定义b[i]=∑a[j]*dis(i,j)

有两种情况:

1.给定a[i],求出b[i]

2.给定b[i],求出a[i]

solution

对于树上的一对fa和son,我们发现b[fa]和b[son]的差别只是由于他俩之间的边做出贡献

定义 sum为整棵树的取值之和 pre[i]=sum-val[i](以i为根的子树权值和) suf[i]=val[i]

而 pre[i]和suf[i] 都是可以通过一遍O(n)的dfs求出

那么得到n-1个关系 b[fa]-b[son]=-pre[fa]+suf[son]

第一种情况:

由b[fa]可以推到b[son] 即

b[son]=b[fa]+pre[fa]-suf[fa]

所以先dfs一遍求出b[root]

再递推即可

第二种情况:

b[fa]-b[son]=-pre[fa]+suf[son]+ sum=pre[fa]+suf[son]

                  ↓

b[fa]-b[son]=2*suf[son]-sum

把n-1个关系相加得

temp=2*(∑suf[k](1<=k<=n,k!=root))-(n-1)*sum的值

得到的n-1个关系 只是a[i]之间的关系,与它们的具体取值无关

而 b[root]=∑suf[k](1<=k<=n,k!=root)

(temp+b[root]*2)/(n-1)=sum

求出来sum再回代即可求出suf[i],最后差分求出a[i]

  1 #include <cstdio>
  2 #include <iostream>
  3 #include <cstring>
  4 #define mem(a,b) memset(a,b,sizeof(a))
  5 #define ll long long
  6 using namespace std;
  7 const int N=200066;
  8 int read()
  9 {
 10   int ans=0,flag=1;char q=getchar();
 11   while(q<'0'||q>'9'){if(q=='-')flag=-1;q=getchar();}
 12   while(q>='0'&&q<='9'){ans=ans*10+q-'0';q=getchar();}
 13   return ans*flag;
 14 }
 15 struct son
 16 {
 17     int v,next;
 18 };
 19 son a1[N*4];
 20 int first[N*4],e;
 21 void addbian(int u,int v)
 22 {
 23   a1[e].v=v;
 24   a1[e].next=first[u];
 25   first[u]=e++;
 26 }
 27  
 28 int T;
 29 int kk,n;
 30 int u,o;
 31 int a[N],b[N];
 32  
 33 int fa[N];
 34  
 35 int val[N],dep[N];
 36 void dfs0(int x)
 37 {
 38   val[x]=a[x];
 39   for(int i=first[x];i!=-1;i=a1[i].next)
 40   {
 41     int temp=a1[i].v;
 42     if(temp==fa[x])continue;
 43     dep[temp]=dep[x]+1;
 44     fa[temp]=x;
 45     dfs0(temp);
 46     val[x]+=val[temp];
 47   }
 48 }
 49  
 50 void dfs00(int x)
 51 {
 52   for(int i=first[x];i!=-1;i=a1[i].next)
 53   {
 54     int temp=a1[i].v;
 55     if(fa[x]==temp)continue;
 56     b[temp]=b[x]+val[1]-val[temp]-val[temp];
 57     dfs00(temp);
 58   }
 59 }
 60  
 61 void work0()
 62 {
 63   for(int i=1;i<=n;++i)
 64     a[i]=read();
 65   mem(b,0);mem(fa,0);mem(val,0);
 66   mem(dep,0);
 67    
 68   dfs0(1);
 69   for(int i=2;i<=n;++i)
 70     b[1]+=dep[i]*a[i];
 71   dfs00(1);
 72   for(int i=1;i<=n;++i)
 73     printf("%d ",b[i]);
 74   printf("\n");
 75 }
 76  
 77 void dfs1(int x)
 78 {
 79   for(int i=first[x];i!=-1;i=a1[i].next)
 80   {
 81     int temp=a1[i].v;
 82     if(temp==fa[x])continue;
 83     fa[temp]=x;
 84     a[temp]=b[x]-b[temp];
 85     dfs1(temp);
 86   }
 87 }
 88  
 89 void dfs11(int x)
 90 {
 91   for(int i=first[x];i!=-1;i=a1[i].next)
 92   {
 93     int temp=a1[i].v;
 94     if(temp==fa[x])continue;
 95     a[x]-=a[temp];
 96     dfs11(temp);
 97   }
 98 }
 99  
100 void work1()
101 {
102   for(int i=1;i<=n;++i)
103     b[i]=read();
104   mem(a,0);mem(fa,0);
105    
106   dfs1(1);
107   ll SUM=0;
108   for(int i=2;i<=n;++i)
109     SUM+=a[i];
110   SUM-=(2*b[1]);
111   SUM=-SUM;
112   SUM/=(n-1);
113   for(int i=2;i<=n;++i)
114     a[i]=((a[i]+SUM)>>1);
115   a[1]=SUM;
116   dfs11(1);
117   //printf("SUM=%d\n",SUM);
118   for(int i=1;i<=n;++i)
119     printf("%d ",a[i]);
120   printf("\n");
121 }
122  
123 int main(){
124    
125   //freopen("single9.in","r",stdin);
126   //freopen("2.txt","w",stdout);
127    
128   T=read();
129   while(T--)
130   {
131     mem(a1,0);mem(first,-1);e=0;
132     n=read();
133     for(int i=1;i<n;++i)
134     {
135       u=read();o=read();
136       addbian(u,o);
137       addbian(o,u);
138     }
139     kk=read();
140     if(kk==1)work1();
141     else work0();
142   }
143 }
code

你在平面直角坐标系上.

你一开始位于(0,0).

每次可以在上/下/左/右四个方向中选一个走一步.

即:从(x,y)走到(x,y+1),(x,y-1),(x-1,y),(x+1,y)四个位置中的其中一个.

允许你走的步数已经确定为n.现在你想走n步之后回到(0,0).但这太简单了.你希望知道有多少种不同的方案能够使你在n步之后回到(0,0).当且仅当两种方案至少有一步走的方向不同,这两种方案被认为是不同的.

答案可能很大所以只需要输出答案对10^9+7取模后的结果.(10^9+7=1000000007,1和7之间有8个0)

这还是太简单了,所以你给能够到达的格点加上了一些限制.一共有三种限制,加上没有限制的情况,一共有四种情况,用0,1,2,3标号:

0.没有任何限制,可以到达坐标系上所有的点,即能到达的点集为{(x,y)|x,y为整数}

1.只允许到达x轴非负半轴上的点.即能到达的点集为{(x,y)|x为非负数,y=0}

2.只允许到达坐标轴上的点.即能到达的点集为{(x,y)|x=0或y=0}

3.只允许到达x轴非负半轴上的点,y轴非负半轴上的点以及第1象限的点.即能到达的点集为{(x,y)|x>=0,y>=0}

typ=2,n<=1000  typ=3,n<=100000   typ=1,n<=100000  typ=0,n<=100000

solution

注意一下走的步数必须是偶数


tpy=0

ans=∑( C(n,i)*C(i,i/2)*C(n-i,(n-i)/2) )

解释:

C(n,i)是从n步里选出i步水平走

C(i,i/2)是从i步里选出i/2步向右走

C(n-i,(n-i)/2)是从n-i步里选出(n-i)/2步向上走


tpy=1

ans=Catalan[n/2]

解释:

向左、右走 就相当于 左、右括号


tpy=2

定义 f[i] 为走i步回到原点的方案数

f[i]=∑( f[k]*Catalan[(i-k)/2-1]*4 )

ans=f[n]

解释:

Catalan[(i-k)/2-1]意为走i-k步第一次回到原点的方案数 = (0,1)向上走再回来的方案数


tpy=3

ans=∑( C(n,i)*Catalan[i]*Catalan[ (n-i)/2 ] )

 1 #include<cstdio>
 2 #include<cstring>
 3 #include<iostream>
 4 #define ll long long
 5 #define C(n,m) (jie[(n)]*jieni[(m)]%mod*jieni[(n)-(m)]%mod)
 6 #define can(n) (C(((n)<<1),(n))*ni[(n)+1]%mod)
 7 using namespace std;
 8 const int mod=1000000007;
 9 const int N=100066;
10  
11 int n,kk;
12  
13 ll jie[N],jieni[N],ni[N];
14 void chu()
15 {
16   ni[1]=1;
17   for(int i=2;i<N;++i)
18     ni[i]=(ll)(mod-mod/i)*ni[mod%i]%mod;
19    
20   jie[0]=jieni[0]=jie[1]=jieni[1]=1;
21   for(int i=2;i<N;++i)
22   {
23     jie[i]=jie[i-1]*(ll)i%mod;
24     jieni[i]=jieni[i-1]*ni[i]%mod;
25   }
26 }
27  
28 void work0()
29 {
30   ll ans=0,temp;
31   for(int i=0;i<=n;i+=2)
32   {
33     temp=1;
34     temp=temp*C(n,i)%mod*C(i,(i>>1))%mod*C((n-i),((n-i)>>1))%mod;
35     ans=(ans+temp)%mod;
36   }
37   cout<<ans;
38 }
39  
40 void work1()
41 {
42   cout<<can((n>>1));
43 }
44  
45 ll f[N];
46 void work2()
47 {
48   f[0]=1;
49   for(int i=2;i<=n;i+=2)
50     for(int j=2;j<=i;j+=2)
51       f[i]=(f[i]+f[i-j]*can((j>>1)-1)%mod*4%mod)%mod;
52   cout<<f[n];
53 }
54  
55 void work3()
56 {
57   ll ans=0,temp;
58   for(int i=0;i<=n;i+=2)
59   {
60     temp=1;
61     temp=temp*C(n,i)%mod*can((i>>1))%mod*can(((n-i)>>1))%mod;
62     ans=(ans+temp)%mod;
63   }
64   cout<<ans;
65 }
66  
67 int main(){
68    
69   //freopen("problem.in","r",stdin);
70   //freopen("problem.out","w",stdout);
71    
72   chu();
73    
74   scanf("%d%d",&n,&kk);
75    
76   if(kk==0)work0();
77   else if(kk==1)work1();
78   else if(kk==2)work2();
79   else work3();
80    
81   return 0;   
82 }
code

总结

总的来说这套题还是不难的

我考的不好主要是刷题太少,老是没有感觉

posted @ 2017-08-29 17:18  A_LEAF  阅读(389)  评论(0编辑  收藏  举报