2018年全国多校算法寒假训练营练习比赛(第三场)

A 不凡的夫夫

题目描述 

夫夫有一天对一个数有多少位数感兴趣,但是他又不想跟凡夫俗子一样,
所以他想知道给一个整数n,求n!的在8进制下的位数是多少位。

输入描述:

第一行是一个整数t(0<t<=1000000)(表示t组数据)
接下来t行,每一行有一个整数n(0<=n<=10000000)

输出描述:

输出n!在8进制下的位数。
示例1

输入

3
4
2
5

输出

2
1
3

思路: 斯特林公式。

斯特林公式(Stirling's approximation)是一条用来取n的阶乘近似值的数学公式。一般来说,当n很大的时候,n阶乘的计算量十分大,所以斯特林公式十分好用,而且,即使在n很小的时候,斯特林公式的取值已经十分准确。求N!的位数:
lnN!=NlnN-N+0.5*ln(2*N*pi)  !要想求有多少位,将他换成以10为底便可。利用换底公式得  lnN!/ln10=log10N!
把式子取整形加1就是位数!

代码:

#include<map>
#include<vector>
#include<queue>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
#include<cmath>
#define maxn 100010
using namespace std;
typedef long long ll;
#define E  2.71828182845  
#define PI 3.1415926  
int main(){
    ll n,t;
    scanf("%lld",&t);
    while(t--)
    {
        ll ans=1;
        scanf("%lld",&n);
        if(n>3)
        {
            ans=(log10(sqrt((long double)2.0 *PI*n))+(n*(log10((long double)n)-log10((long double)E))))/log10(8)+1;
        }
        printf("%lld\n",ans);
    }
    return 0;
}


B 一个小问题

题目描述 

uu遇到了一个小问题,可是他不想答。你能替他解决这个问题吗?
问题:给你k对a和r是否存在一个正整数x使每队a和r都满足:x mod a=r,求最小正解x或无解。


输入描述:

第一行是正整数k(k<=100000)
接下来k行,每行有俩个正整数a,r(100000>a>r>=0)

输出描述:

在每个测试用例输出非负整数m,占一行。
如果有多个可能的值,输出最小的值。
如果没有可能的值,则输出-1。
示例1

输入

2
8 7
11 9

输出

31

思路:中国剩余定理。模板题

代码:

#include<map>
#include<vector>
#include<queue>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
#define maxn 100010
using namespace std;
typedef long long ll;
void gcd(ll a,ll b,ll &d,ll &x,ll &y)
{
    if(!b){
        d=a;
        x=1;
        y=0;}
    else
    {
        gcd(b,a%b,d,y,x);
        y-=x*(a/b);
    }
}

ll c(ll n,ll a[],ll b[])
{
    ll m1=a[0];
    ll r1=b[0];
    ll flag=0;
    ll d;
    for(ll i=1;i<n;i++)
    {
        ll m2=a[i];
        ll r2=b[i];
        if(flag)
         continue;
         ll x,y;
        gcd(m1,m2,d,x,y);
        ll c=r2-r1;
        if(c%d)
        {
            flag=1;
            continue;
        }
        ll t=m2/d;
        x=(c/d*x%t+t)%t;
        r1=m1*x+r1;
        m1=m1*m2/d;
    }
    if(flag) 
	return -1;
    if(n==1&&r1==0) 
	return m1;
    return r1;
}
ll aa[550000],bb[550000];

int main()
{
    ll k,i;
    cin>>k;
    for(i=0;i<k;i++)
    
    cin>>aa[i]>>bb[i];
    
    cout<<c(k,aa,bb)<<endl;
}

D 小牛vs小客

题目描述 

小牛和小客玩石子游戏,他们用n个石子围成一圈,小牛和小客分别从其中取石子,谁先取完谁胜,每次可以从一圈中取一个或者相邻两个,每次都是小牛先取,请输出胜利者的名字(小牛获胜输出XiaoNiu,小客获胜输出XiaoKe)(1 2 3 4 取走 2 13 不算相邻)


输入描述:

输入包括多组测试数据
每组测试数据一个n(1≤n≤1e9)

输出描述:

每组用一行输出胜利者的名字(小牛获胜输出XiaoNiu,小客获胜输出XiaoKe)
示例1

输入

2
3

输出

XiaoNiu
XiaoKe

思路:博弈题目,取石子(七)问题。


假设石子数等于5,如果先者先取一个,那么后者拿走两个,将剩下的两个石子分成两堆,后者赢。如果先者先取二个,那么后者取一个使剩下的两个石子分成两堆,后者赢。

假设石子数等于6,如果先者先取一个,那么后者拿走一个,将剩下的石子分成两段,每段两个,如果先者再拿两个,那么后者赢,如果先者再拿一个,那么后者再取另一堆中的一个,这样剩下的两个石子被分成两堆, 后者赢。         如果先者先取两个,那么后者也取两个使剩下的两个石子分成两堆,后者赢。

所以当先者取走后,后者取走一个或者两个,将剩下的石子分成对称的两段,以此类推,那么如果石子数大于2后者一定赢。

代码:
#include <stdio.h>  
  
int main (void)  
{  
    int n;  
    while (scanf("%d", &n) != EOF)  
    {  
        if(n > 2)  
            printf("XiaoKe\n");  
        else  
            printf("XiaoNiu\n");  
    }  
    return 0;  
}  

E 进击吧!阶乘

题目描述 

给定一个整数 N0≤N≤10000),求取 N的阶乘

输入描述:

多个测试数据,每个测试数据输入一个数N

输出描述:

每组用一行输出N的阶乘
示例1

输入

1
2
3

输出

1
2
6

思路:Java大数阶乘模板题

代码:

import java.math.BigInteger;
import java.util.Scanner;
public class Main{
    public static void main(String[] args) {     
    Scanner inputScanner=new Scanner(System.in);
       while(inputScanner.hasNext())
       {
           int n=inputScanner.nextInt();
           BigInteger m;
           m=BigInteger.valueOf(1);//将m定义成大数的1
           for(int i=1;i<=n;i++)
           {
               m=m.multiply(BigInteger.valueOf(i));//大数乘法
           }
           System.out.println(m);
       }  
    }
}
F 小牛再战

题目描述 

共有N堆石子,已知每堆中石子的数量,两个人轮流取石子,每次只能选择N堆石子中的一堆取一定数量的石子(最少取一个),取过子之后,还可以将该堆石子中剩余的石子随意选取几个放到其它的任意一堆或几堆上。等哪个人无法取子时就表示此人输掉了游戏。注意:一堆石子没有子之后,就不能再往此处放石子了。

假设每次都是小牛先取石子,并且游戏双方都绝对聪明,现在给你石子的堆数、每堆石子的数量,请判断出小牛能否获胜。

输入描述:

可能有多组测试数据(测试数据组数不超过1000)
每组测试数据的第一行是一个整数,表示N(1<=N<=10)
第二行是N个整数分别表示该堆石子中石子的数量。(每堆石子数目不超过100)
当输入的N为0时,表示输入结束

输出描述:

对于每组测试数据,输出Win表示小牛可以获胜,输出Lose表示小牛必然会败。
示例1

输入

3
2 1 3
2
1 1
0

输出

Win
Lose

备注:

提示:
例如:如果最开始有4堆石子,石子个数分别为3 1 4 2,而小牛想决定要先拿走第三堆石子中的两个石子(石子堆状态变为3 1 2 2),然后他可以使石子堆达到的状态有以下几种:
3 1 2 2(不再移动石子)
4 1 1 2(移动到第一堆一个)
3 2 1 2(移动到第二堆一个)
3 1 1 3(移动到第四堆一个)
5 1 0 2(全部移动到第一堆)
3 3 0 2(全部移动到第二堆)
3 1 0 4(全部移动到最后)

思路:博弈问题。取石子问题(三)

代码:

#include<map>
#include<vector>
#include<queue>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
#include<cmath>
#define maxn 100010
using namespace std;
int n;
int s;
int main()
{
	int ans[110],i,num;
	while(scanf("%d",&n)&&n)
	{
		memset(ans,0,sizeof(ans));
		for(i=0;i<n;++i)
		{
			scanf("%d",&num);
			ans[num]++;
		}
		s=0;
		for(i=0;i<101;++i)
		{
			if(ans[i]&1)
			{
				s=1;
				break;
			}
		}
		if(s)
		   printf("Win\n");
		else
		   printf("Lose\n");
		
	}
	return 0;
} 
G 大水题

题目描述 

给出一个数n,求1到n中,有多少个数不是2 5 11 13的倍数。 

输入描述:

本题有多组输入
每行一个数n,1<=n<=10^18.

输出描述:

每行输出输出不是2 5 11 13的倍数的数共有多少。
示例1

输入

15

输出

4

说明

1 3 7 9

思路:容斥原理:  要计算几个集合并集的大小,我们要先将所有单个集合的大小计算出来,然后减去所有两个集合相交的部分,再加回所有三个集合相交的部分,再减去所有四个集合相交的部分,依此类推,一直计算到所有集合相交的部分。

    那么的面积就是集合ABC各自面积之和减去  的面积,再加上的面积。


         由此,我们也可以解决n个集合求并的问题。

代码:

#include<map>
#include<vector>
#include<queue>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
#include<cmath>
#define maxn 100010
using namespace std;
typedef long long ll;
#define PI acos(-1.0)
int main()
{
    ll n;
    ios::sync_with_stdio(false);
    while(scanf("%lld",&n)!=EOF)
    {
        ll cnt;
        cnt=n-(n/2)-(n/5)-(n/11)-(n/13);
        cnt=cnt+(n/10)+(n/22)+(n/26)+(n/55)+(n/65)+(n/143);
        cnt=cnt-(n/110)-(n/130)-(n/715)-(n/286);
        cnt=cnt+(n/1430);
        cout<<cnt<<endl;
    }
    return 0;
}



















  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值