PAT乙级(Basic Level)练习题-NowCoder数列总结

题目描述



NowCoder最近在研究一个数列:

  • F(0) = 7
  • F(1) = 11
  • F(n) = F(n-1) + F(n-2) (n≥2)

    他称之为NowCoder数列。请你帮忙确认一下数列中第n个数是否是3的倍数。


输入描述:

输入包含多组数据。

每组数据包含一个整数n,(0≤n≤1000000)。


输出描述

对应每一组输入有一行输出。

如果F(n)是3的倍数,则输出“Yes”;否则输出“No”。

---

输入例子:

0

1

2

3

4

5

输出例子:

No

No

Yes

No

No

No

---

题目分析

这是一个特殊初始条件的Fibonacci sequence.

1.初始条件n的值比较小,可以直接枚举每个F(n)的值,然后输入n查询:

#include <iostream>
#include <cstdio>
using namespace std;
int main (){
    int n;
    int *a=new int[N];
    a[0]=7;
    a[1]=11;
    for(int i=2;i<N;i++)
        a[i]=a[i-2]%3+a[i-1]%3;
    while(~scanf("%d", &n)){
        cout<<a[n]<<endl;
        if(a[n]%3==0)
            cout<<"Yes"<<endl;
        else
            cout<<"No"<<endl;
    }
    return 0;
}  

2.可以寻找到规律:(牛客网友“我要过pat”提供)

这个真牛批,想不到还有这种规律。那Fibonacci数列是否有相似的规律呢?

#include <iostream>
using namespace std;

int main (){
    int n;
    while(cin>>n){
    if((n-2)%4==0)
        cout<<"Yes"<<endl;
    else
        cout<<"No"<<endl;
    }
    return 0;
}  

3.用矩阵快速幂来求解,这题n的范围比较小(0≤n≤1000000)所以用第1种方法可以不超时求解,但是,倘若n很大,达到(1<=n<=1000,000,000,000,000,000)这个范围,那么显而易见,必定超时。而且根本开不到那么大的数组。学校的算法课,有另外一题类似的,可以思考一下:

借此思路,用矩阵快速幂计算,稍微修改下就可以了。

 #include <iostream>
#include <cstdio>
#include <cstring>
#define N 3
#define maxn 2   

using namespace std;  

struct Matrix{
    long long a[maxn][maxn];
    void init(){    //初始化为单位矩阵
        memset(a, 0, sizeof(a));
        for(int i=0;i<maxn;++i){
            a[i][i] = 1;
        }
    }
};  

//矩阵乘法
Matrix mul(Matrix a, Matrix b){
    Matrix ans;
    for(int i=0;i<maxn;++i){
        for(int j=0;j<maxn;++j){
            ans.a[i][j] = 0;
            for(int k=0;k<maxn;++k){
                ans.a[i][j] += a.a[i][k] * b.a[k][j];
                ans.a[i][j] %= N;
            }
        }
    }
    return ans;
}  

//矩阵快速幂
Matrix qpow(Matrix a, long long n){
    Matrix ans;
    ans.init();
    while(n){
        if(n&1)
            ans = mul(ans, a);
        a = mul(a, a);
        n /= 2;
    }
    return ans;
}  

int main (){
    long long n;  

    while(~scanf("%lld", &n)){
    Matrix a;
    a.a[0][0] = 1;
    a.a[0][1] = 1;
    a.a[1][0] = 1;
    a.a[1][1] = 0;
    long long s0=7,s1=11;
    if (n>1)
        {
        Matrix ans= qpow(a, n-1);
        long long res=ans.a[0][0]*s1+ans.a[0][1]*s0;
//        printf("%lld",res);
        if(res%3==0)
            cout<<"Yes"<<endl;
        else
            cout<<"No"<<endl;
        }
    else
        cout<<"No"<<endl;   

    }
    return 0;
}  

测评网站真的恶心,有时候循环输入用"cin>>"就可以,有时候又超时。。。得改"成~scanf("%lld", &n)"。另外一开始没想到对中间结果对3取模,导致数值过大溢出。用到unsigned long long 都不够。。。

PS:检测一个数被3整除的算法

1.检测一个数能否被3整除----位运算

如果所有的偶数位出现1的次数为 even_count, 奇数位出现1的次数为 odd_count,两者只差如果是3的倍数,那么这个数就是3倍数。

2.不用除法和求模运算,判断一个数能否被3整除

现在给出一个数a,假设它能被3整除,结果是b,即a=3*b,那么从二进制乘法运算判断出,b的最低位与a的最低位一定是相同的,从而得到了b的最低位,将这个位左移1位变成次低位,那么a的次低位以上的比特减去这个位后在次低位上的结果一定是b的次低位。以此类推可以求出b的各个比特,如果最后能完成对b的各位的计算,那么a能够被3整除,否则不能被3整除。

那么算法原理是什么呢?

原文地址:https://www.cnblogs.com/cafe3165/p/9732826.html

时间: 2024-08-29 06:33:34

PAT乙级(Basic Level)练习题-NowCoder数列总结的相关文章

PAT乙级(Basic Level)真题训练

写在前面:PAT冬季赛马上就要开始了!??这一次先报一个乙级冲鸭!我感Jio乙级里面还是有蛮多水题的,也有些题虽然看上去是水题,但是真正用代码实现起来的话会卡你那么一下,比如第5题数素数真的神打脸. 天上不会掉馅饼的,好好学习,努力奋斗才能梦想成真. 1. D进制的A + B(20) 题目描述: 输入两个非负10进制整数A和B(<= 230-1),输出A + B的D(1 <D <= 10)进制数. 输入描述: 输入在一行中依次给出3个整数A,B和D. 输出描述: 输出A + B的D进制数

PAT乙级(Basic Level)真题,福尔摩斯的约会

题目描述 大侦探福尔摩斯接到一张奇怪的字条:“我们约会吧! 3485djDkxh4hhGE 2984akDfkkkkggEdsb s&hgsfdk d&Hyscvnm”.大侦探很快就明白了,字条上奇怪的乱码实际上就是约会的时间“星期四 14:04”,因为前面两字符串中第1对相同的大写英文字母(大小写有区分)是第4个字母'D',代表星期四:第2对相同的字符是'E',那是第5个英文字母,代表一天里的第14个钟头(于是一天的0点到23点由数字0到9.以及大写字母A到N表示):后面两字符串第1对相

PAT(Basic Level)practice1003

我要通过 “答案正确”是自动判题系统给出的最令人欢喜的回复.本题属于PAT的“答案正确”大派送 —— 只要读入的字符串满足下列条件,系统就输出“答案正确”,否则输出“答案错误”. 得到“答案正确”的条件是: 1. 字符串中必须仅有P, A, T这三种字符,不可以包含其它字符:2. 任意形如 xPATx 的字符串都可以获得“答案正确”,其中 x 或者是空字符串,或者是仅由字母 A 组成的字符串:3. 如果 aPbTc 是正确的,那么 aPbATca 也是正确的,其中 a, b, c 均或者是空字符

PAT(Basic Level)practice1001

写在前面的话:其实自己学习计算机已经接近五年了,从本科一直到了研究生,然而只能说自己浪费了大把的时光,对c语言也是最近才开始真正的上手学习,可想而知大学四年都是如何水过来的,现在想起来,内心其实是后悔的,后悔自己浪费了时间,后悔自己没能珍惜那么好的资源.也担心,担心自己没办法弥补.但是是谁说过的来着,学习一件事情,最好的时间是十年前,其次现在.既然我有了改变自己的勇气,就希望自己可以勇敢走下去.从PAT最基础的开始练习,因为是新手,写的代码可能很丑,希望有一天当自己回头看,可以看到自己的努力与进

PTA(Basic Level)1030.完美数列

给定一个正整数数列,和正整数 p,设这个数列中的最大值是 M,最小值是 m,如果 M≤*m**p*,则称这个数列是完美数列. 现在给定参数 p 和一些正整数,请你从中选择尽可能多的数构成一个完美数列. 输入格式: 输入第一行给出两个正整数 N 和 p,其中 N(≤105)是输入的正整数的个数,p(≤109)是给定的参数.第二行给出 N 个正整数,每个数不超过 109. 输出格式: 在一行中输出最多可以选择多少个数可以用它们组成一个完美数列. 输入样例: 10 8 2 3 20 4 5 1 6 7

PAT(Basic Level)practice1002

写出这个数 读入一个自然数n,计算其各位数字之和,用汉语拼音写出和的每一位数字. 输入格式:每个测试输入包含1个测试用例,即给出自然数n的值.这里保证n小于10100. 输出格式:在一行内输出n的各位数字之和的每一位,拼音数字间有1 空格,但一行中最后一个拼音数字后没有空格. 输入样例: 1234567890987654321123456789 输出样例: yi san wu 这道题首先要明白n的范围已经超过了普通的类型,所以肯定要利用字符数组来进行处理.其次在求得和之后,输出有两种方法,如果是

PAT(Basic Level)practice1004

读入n名学生的姓名.学号.成绩,分别输出成绩最高和成绩最低学生的姓名和学号. 输入格式:每个测试输入包含1个测试用例,格式为 第1行:正整数n 第2行:第1个学生的姓名 学号 成绩 第3行:第2个学生的姓名 学号 成绩 ... ... ... 第n+1行:第n个学生的姓名 学号 成绩 其中姓名和学号均为不超过10个字符的字符串,成绩为0到100之间的一个整数,这里保证在一组测试用例中没有两个学生的成绩是相同的. 输出格式:对每个测试用例输出2行,第1行是成绩最高学生的姓名和学号,第2行是成绩最低

乙级(Basic Level) 1006

题目描述 正整数A的"DA(为1位整数)部分"定义为由A中所有DA组成的新整数PA.例如:给定A = 3862767,DA = 6,则A的"6部分"PA是66,因为A中有2个6. 现给定A.DA.B.DB,请编写程序计算PA + PB. 输入描述: 输入在一行中依次给出A.DA.B.DB,中间以空格分隔,其中0 < A, B < 1010. 输出描述: 在一行中输出PA + PB的值. 输入例子: 3862767 6 13530293 3 输出例子: 3

乙级(Basic Level) 1007

题目描述 本题要求计算A/B,其中A是不超过1000位的正整数,B是1位正整数.你需要输出商数Q和余数R,使得A = B * Q + R成立. 输入描述: 输入在1行中依次给出A和B,中间以1空格分隔. 输出描述: 在1行中依次输出Q和R,中间以1空格分隔. 输入例子: 123456789050987654321 7 输出例子: 17636684150141093474 3 解法: Python:a = input().split() a = [int(i) for i in a] print(