NOIP模拟 17.8.20
NOIP模拟17.8.20
A.阶乘
【题目描述】
亲爱的xyx同学正在研究数学与阶乘的关系,但是他喜欢颓废,于是他就制作了一个和阶乘有关系的数学游戏:
给出两个整数 n,m,令 t = !n,每轮游戏的流程如下
1.如果 m不能整除t ,即 t mod m ≠ 0,跳到第三步;如果能整除,跳到第二步
2.令 t = t/m,xyx的得分+1并返回第一步
3.游戏结束
xyx共进行T轮游戏,他想知道每轮他的得分是多少
【输入描述】
第一行一个整数 T,表示游戏轮数
接下来 T行,每行两个正整数n,m ,含义见题目描述
【输出描述】
共T 行,每行输出一个整数,表示xyx该轮游戏得到的分数
【样例输入】
4
2 3
3 3
23456 5
8735373 10
【样例输出】
0
1
5861
2183837
【限制与约定】
对于30% 的数据, n,m ≤ 20
对于60% 的数据, n,m ≤ 1e6
对于 100%的数据, n,m ≤ 1e9 ,T ≤ 10,1 < m <= 15
【题解】
此题极好
n! = 1*2*3*4*5*...*n
把m分解,设第i个质因数为pi,假设n中有k个pi
n! = 1*2*3..*pi*...*2pi*...*3pi*...*kpi...*n
然后我们让n/=pi,可以得到
n/pi! = 1*2*3*...*pi*...*2pi*...*3pi*...*(k - n/p - 1)pi*...*n/pi
证明比较显然,k*p每次-p就到达下一个p的倍数,于是,n每次-p,(n-p)!就
少一个p。干脆直接n不断/p就好了
最终结果要除以m中pi的数量
取一个min即可
1 #include <iostream> 2 #include <cstdio> 3 #include <cstdlib> 4 #include <cstring> 5 #define min(a, b) ((a) < (b) ? (a) : (b)) 6 7 inline void read(long long &x) 8 { 9 x = 0;char ch = getchar(), c = ch; 10 while(ch < '0' || ch > '9')c = ch, ch = getchar(); 11 while(ch <= '9' && ch >= '0')x = x * 10 + ch - '0', ch = getchar(); 12 if(c == '-')x = -x; 13 } 14 15 const long long INF = 0x3f3f3f3f; 16 17 long long n,m,t,ans; 18 19 int main() 20 { 21 read(t); 22 register long long k,tmp,p; 23 for(;t;--t) 24 { 25 read(n), read(m); 26 ans = INF; 27 for(register int i = 2;i <= m;++ i) 28 { 29 if(m % i == 0) 30 { 31 k = 0, p = n, tmp = 0; 32 while(m % i == 0)m /= i, ++ k; 33 while(p)tmp += p/i, p/= i; 34 ans = min(ans, tmp/k); 35 } 36 } 37 printf("%I64d\n", ans); 38 } 39 return 0; 40 }
B.妹子
【题目描述】
亲爱的xyx同学正在研究数学与妹子的关系,但是他喜欢颓废,于是他就制作了一个和妹子有关系的数学游戏:
有一棵树,树上每个点i 有一个权值ai ,每轮游戏中xyx和妹子会各自选择一个点,然后对于树上所有的点来说,如果它与
xyx所选点的距离更近,那么xyx就会得到等于该点权值的分数;如果它与妹子所选点的距离更近,那么妹子就会得到等
于该点权值的分数;如果距离相等,那么xyx和妹子都不会得到分数。
xyx想知道,每轮游戏中,他和妹子得到的分数分别是多少
两点间的距离定义为该两点最短路径所经过的边的权值之和
【输入描述】
第一行一个整数 n,表示树的顶点数。
接下来n -1行,每行两个整数 u,v,w,表示第i 条边连接点 u,v,边权为w
接下来一行共 n个整数,表示树上每个点 i的权值ai 。
接下来一行一个整数 m,表示游戏轮数
接下来 m行,每行两个整数x,y ,分别表示xyx所选的点和妹子所选的点
【输出描述】
共 m行,每行输出两个整数,表示xyx得到的分数和妹子得到的分数,中间用一个空格隔开。
【样例输入】
3
1 2 1
1 3 1
10 1 1
2
2 3
1 3
【样例输出】
1 1
11 1
【限制与约定】
对于10% 的数据, n,m ≤ 100
对于30%的数据, n,m ≤ 1000
对于另外15% 的数据,第 i条边连接点i,i+1
对于另外15% 的数据,树和询问保证随机生成
对于 100%的数据,n ≤ 100000,m ≤ 50000,1 ≤ x, y,u, v ≤ n,1 ≤ w,ai ≤ 10
【题解】
此题极好+1
考察细节,参见代码
“点的类型可以看成是单调的(属于x->谁都不属于->属于y)
设dis[x]表示x到根的距离,每次询问中dis[x]>dis[y],他们的lca为p
那么点的类型边界一定在p->x的路径上
由于在链上存在单调性,可以二分+倍增找深度差为k的父亲来做
复杂度O(n log^2 n)
也可以倍增+从高到低确定二进制位,这一位是1时如果已经不受x控制,则这一位一定为0;反之这一位为0(贪心思路)
复杂度O(n log n)”
——张浩南
1 #include <iostream> 2 #include <cstdio> 3 #include <cstdlib> 4 #include <cstring> 5 6 inline void read(long long &x) 7 { 8 x = 0;char ch = getchar(),c = ch; 9 while(ch < '0' || ch > '9')c = ch, ch = getchar(); 10 while(ch <= '9' && ch >= '0')x = x * 10 + ch - '0', ch = getchar(); 11 if(c == '-')x = -x; 12 } 13 14 const long long MAXN = 200000 + 10; 15 16 struct Edge 17 { 18 long long u,v,next,w; 19 Edge(long long _u, long long _v, long long _next, long long _w){u = _u;v = _v;next = _next;w = _w;} 20 Edge(){} 21 }edge[MAXN << 1]; 22 23 long long n,m,b[MAXN],value[MAXN],sum[MAXN],path[MAXN],p[25][MAXN],deep[MAXN],fa[MAXN],head[MAXN],cnt; 24 25 inline void insert(long long a, long long b, long long c) 26 { 27 edge[++cnt] = Edge(a,b,head[a],c); 28 head[a] = cnt; 29 } 30 31 void dfs(long long u) 32 { 33 b[u] = 1; 34 sum[u] = value[u]; 35 for(register long long pos = head[u];pos;pos = edge[pos].next) 36 { 37 long long v = edge[pos].v; 38 if(b[v])continue; 39 path[v] = path[u] + edge[pos].w; 40 deep[v] = deep[u] + 1; 41 dfs(v); 42 fa[v] = u; 43 p[0][v] = u; 44 sum[u] += sum[v]; 45 } 46 } 47 48 //p[0]是点 p[1]是路径长度 49 inline void yuchuli() 50 { 51 long long M = 0; 52 while((1 << M) <= n)++ M; 53 -- M; 54 for(register long long i = 1;i <= M;++ i) 55 for(register long long j = 1;j <= n;++ j) 56 p[i][j] = p[i - 1][p[i - 1][j]]; 57 } 58 59 long long lca(long long va, long long &vb) 60 { 61 int flag = 0; 62 if(deep[va] < deep[vb]) 63 { 64 long long tmp = va; 65 va = vb; 66 vb = tmp; 67 flag = 1; 68 } 69 long long M = 1; 70 while((1 << M) + deep[vb] <= deep[va])++ M; 71 -- M; 72 for(register long long i = M;i >= 0;-- i) 73 if((1 << i) + deep[vb] <= deep[va]) 74 va = p[i][va]; 75 if(va == vb)return va; 76 M = 1; 77 while((1 << M) + deep[vb] <= n)++ M; 78 -- M; 79 for(register long long i = M;i >= 0;-- i) 80 if(p[i][va] != p[i][vb]) 81 { 82 va = p[i][va]; 83 vb = p[i][vb]; 84 } 85 if(flag)vb = va; 86 return p[0][va]; 87 } 88 89 int main() 90 { 91 read(n); 92 register long long tmp1, tmp2, tmp3; 93 for(register long long i = 1;i < n;++ i) 94 { 95 read(tmp1),read(tmp2),read(tmp3); 96 insert(tmp1, tmp2, tmp3); 97 insert(tmp2, tmp1, tmp3); 98 } 99 for(register long long i = 1;i <= n;++ i) read(value[i]); 100 deep[1] = 0; 101 dfs(1); 102 yuchuli(); 103 read(m); 104 register long long a,b,ans1,ans2,lenth,sa,sb,c,ok,M = 0; 105 while((1 << M) <= n)++ M; 106 -- M; 107 for(register long long i = 1;i <= m;++ i) 108 { 109 read(a),read(b); 110 if(a == b) 111 { 112 printf("0 0\n"); 113 continue; 114 } 115 ok = 0; 116 if(path[a] < path[b]) 117 { 118 int tmp = a; 119 a = b; 120 b = tmp; 121 ok = 1; 122 } 123 sa = a, sb = b; 124 c = lca(sa, sb); 125 ans1 = ans2 = 0; 126 /* if(c != 0)*/ 127 lenth = path[a] + path[b] - (path[c] << 1); 128 /* else 129 lenth = path[a] - path[b];*/ 130 for(register int i = M;i >= 0;-- i) 131 if(((path[a] - path[p[i][sa]]) * 2) < lenth) 132 sa = p[i][sa]; 133 if(((path[a] - path[p[0][sa]]) << 1) == lenth) 134 { 135 if(p[0][sa] == c) 136 { 137 if(ok) 138 printf("%lld %lld\n", sum[sb], sum[sa]); 139 else 140 printf("%lld %lld\n", sum[sa], sum[sb]); 141 continue; 142 } 143 else 144 { 145 if(ok) 146 printf("%lld %lld\n", sum[1] - sum[fa[sa]], sum[sa]); 147 else 148 printf("%lld %lld\n", sum[sa], sum[1] - sum[fa[sa]]); 149 continue; 150 } 151 } 152 else 153 { 154 if(ok) 155 printf("%lld %lld\n", sum[1] - sum[sa], sum[sa]); 156 else 157 printf("%lld %lld\n", sum[sa], sum[1] - sum[sa]); 158 continue; 159 } 160 } 161 return 0; 162 }
这题我调了很久,最后发现我本以为最不可能错的LCA居然写挂了,其他地方都没错。。。
C.在
【题目描述】
亲爱的xyx同学正在研究数学与存在的关系,但是他喜欢颓废,于是他就制作了一个和存在有关系的数学游戏:
有一个集合,游戏开始时里面有一些数,xyx每次可以选择集合中的两个数(这两个数可以是同一个数)相减,然后把它
们差的绝对值放入集合中(如果集合中已存在该数则不放入),经过任意次这样的操作后,xyx会给出一个数字 k,问它
是否能存在在集合中,如果能,那么游戏成功,反之失败。
(换句话说,xyx想知道这样不断操作下去,是否能凑出数字 k)
xyx有一个数列 a,他每次会给出三个数 l,r,x,即把所有的ai(l ≤ i ≤ r) 放入集合中,并且想要知道当 k = x时游戏是否成功
有多组询问且询问之间相互独立,你可以认为每次询问后xyx就把游戏中的集合清空了
【输入描述】
第一行两个整数n 和 m,分别表示数字集合的大小和询问次数。
第二行共 n个整数,表示数列 a,第 i个数是 ai。
接下来 m行,每行三个整数 l,r,x,含义见题目描述
【输出描述】
共 m行,每行一个字符串,表示对应询问的答案,如果游戏成功输出”win”,反之输出”lose”。(不输出引号)
【样例输入】
7 5
3 6 4 2 7 1 8
1 2 3
3 7 0
3 4 2
3 4 5
2 3 3
【样例输出】
win
win
win
lose
lose
【限制与约定】
对于20% 的数据, ai,x ∈ [0,1]
对于另外20% 的数据,保证 ai 随机生成
对于另外 20%的数据, l = 1
对于100% 的数据, n,m ≤ 100000,0 ≤ x,ai ≤ 10 ,1 ≤ l ≤ r ≤ n
【题解】
此题一般
当时场外试了几组数据,然后就看出来了
其实能不能凑出k,取决于两个条件:
1、数列中存在大于等于k的数
2、数列的gcd为k的因数
很显然,数列每个数都可以写作p*gcd的性质,他们加减,仍是p*gcd
每次写st表都要手推。。。所以注释里保存了我手推的过程,便于大家学习。。。
1 #include <iostream> 2 #include <cstdio> 3 #include <cstdlib> 4 #include <cstring> 5 #define max(a, b) ((a) > (b) ? (a) : (b)) 6 7 inline void read(int &x) 8 { 9 x = 0;char ch = getchar(),c = ch; 10 while(ch < '0' || ch > '9')c = ch, ch = getchar(); 11 while(ch <= '9' && ch >= '0')x = x * 10 + ch - '0', ch = getchar(); 12 if(c == '-')x = -x; 13 } 14 15 const int MAXN = 100000 + 10; 16 17 18 /* 19 data[i][j]表示区间[j, j + 2^i - 1] 20 data[i][j] = gcd(data[i - 1][j], data[i - 1][j + 2^(i - 1)] 21 [j, j + 2^(i - 1) - 1] [j + 2 ^ (i - 1), j + 2^ i - 1] 22 23 [l,r] 24 2 ^ M <= r - l + 1 < 2 ^ (M + 1) 25 data[M][l], data[M][r - 2^M + 1] 26 */ 27 int n,m,num[MAXN],data[2][22][MAXN],lo[MAXN],pow2[30]; 28 29 inline int gcd(int a, int b) 30 { 31 return b == 0 ? a : gcd(b, a % b); 32 } 33 34 inline void yuchuli() 35 { 36 int M = lo[n]; 37 for(register int i = 1;i <= M;++ i) 38 for(register int j = 1;j <= n;++ j) 39 { 40 data[1][i][j] = gcd(data[1][i - 1][j], data[1][i - 1][j + pow2[i - 1]]); 41 data[0][i][j] = max(data[0][i - 1][j], data[0][i - 1][j + pow2[i - 1]]); 42 } 43 } 44 45 int findma(int l, int r) 46 { 47 return max(data[0][lo[r - l + 1]][l], data[0][lo[r - l + 1]][r - pow2[lo[r - l + 1]] + 1]); 48 } 49 50 int findgcd(int l, int r) 51 { 52 return gcd(data[1][lo[r - l + 1]][l], data[1][lo[r - l + 1]][r - pow2[lo[r - l + 1]] + 1]); 53 } 54 55 int main() 56 { 57 read(n);read(m); 58 lo[2] = 1; 59 for(register int i = 3;i <= n;++ i)lo[i] = lo[i >> 1] + 1; 60 pow2[0] = 1; 61 for(register int i = 1;i <= 25;++ i)pow2[i] = pow2[i - 1] << 1; 62 for(register int i = 1;i <= n;++ i)read(num[i]), data[0][0][i] = data[1][0][i] = num[i]; 63 yuchuli(); 64 register int l,r,k; 65 for(register int i = 1;i <= m;++ i) 66 { 67 read(l),read(r),read(k); 68 int ma = findma(l, r); 69 if(ma < k) 70 { 71 printf("lose\n"); 72 continue; 73 } 74 int g = findgcd(l, r); 75 if(g == 0) printf("win\n"); 76 else if(k % g == 0) 77 { 78 printf("win\n"); 79 continue; 80 } 81 else 82 { 83 printf("lose\n"); 84 continue; 85 } 86 } 87 return 0; 88 }