zoukankan      html  css  js  c++  java
  • 2017年蓝桥杯大题总结

    第七题:

    标题:日期问题
     
    小明正在整理一批历史文献。这些历史文献中出现了很多日期。小明知道这些日期都在1960年1月1日至2059年12月31日。令小明头疼的是,这些日期采用的格式非常不统一,有采用年/月/日的,有采用月/日/年的,还有采用日/月/年的。更加麻烦的是,年份也都省略了前两位,使得文献上的一个日期,存在很多可能的日期与其对应。  
     
    比如02/03/04,可能是2002年03月04日、2004年02月03日或2004年03月02日。  
     
    给出一个文献上的日期,你能帮助小明判断有哪些可能的日期对其对应吗?
     
    输入
    ----
    一个日期,格式是"AA/BB/CC"。  (0 <= A, B, C <= 9)  
     
    输出
    ----
    输出若干个不相同的日期,每个日期一行,格式是"yyyy-MM-dd"。多个日期按从早到晚排列。  
     
    样例输入
    ----
    02/03/04  
     
    样例输出
    ----
    2002-03-04  
    2004-02-03  
    2004-03-02  
     
    资源约定:
    峰值内存消耗(含虚拟机) < 256M
    CPU消耗  < 1000ms
     
     
    请严格按要求输出,不要画蛇添足地打印类似:“请您输入...” 的多余内容。
     
    注意:
    main函数需要返回0;
    只使用ANSI C/ANSI C++ 标准;
    不要调用依赖于编译环境或操作系统的特殊函数。
    所有依赖的函数必须明确地在源文件中 #include <xxx>
    不能通过工程设置而省略常用头文件。
     
    提交程序时,注意选择所期望的语言类型和编译器类型。

    小题做完然后就有点困,仔细看了这个题,感觉好像不太怎么用算法啊,就是有点麻烦,不过认真点应该没啥问题....

    然后自己就可怜的在那不知不觉的将近两个小时...................................[失策!!!]

    #include <iostream>
    #include <stdio.h>
    #include <string>
    #include<set> 
     
    using namespace std;
    int md[13]={0,31,28,31,30,31,30,31,31,30,31,30,31};
    struct date
    {
        int year;
        int month;
        int day;
     
        date(int y,int m,int d)
        {
            year = y;
            month = m;
            day = d;
        }
     
        bool operator < (date other)const     //重载小于符号来比较日期的大小 
        {
            if(year == other.year)            //先比较年份 
            {
                if(month == other.month)      //在比较月份 
                    return day<other.day;     //最后比较日期 
                return month<other.month;
            }
            return year<other.year;
        }
        bool vial()                          //判断日期是否非法
        {                  
            if(year < 1960 || year > 2059) 
               return false;
            if(month <= 0 || month > 12) 
               return false;
            if(year % 400 == 0 || year % 100 != 0 && year % 4 == 0)
            {
                //闰年
                if(month == 2)
                {
                    return day >= 1 && day <= 29;
                } 
                return day >= 1 && day <= md[month]; 
            }
            else
            {
                return day >= 1 && day <= md[month];
            }
        }
        void print()const
        {
            printf("%d-%02d-%02d
    ",year,month,day);
        }
    };
    set<date> ss;  //利用set容器来去重排序;{唯一性!!!}
     
    void insert(int a,int b,int c)
    {
        date obj(a,b,c);                 //插入时创建一个实例化对象 
        if(obj.vial()) 
           ss.insert(obj);
    }
    int main()
    {
        int a,b,c;
        scanf("%d/%d/%d",&a,&b,&c);
        //年月日 
        insert(1900+a,b,c);
        insert(2000+a,b,c);
        //月日年
        insert(1900+c,a,b);
        insert(2000+c,a,b);
        //日月年 
        insert(1900+c,b,a);
        insert(2000+c,b,a);
     
        set<date>::iterator it = ss.begin();
        for(; it != ss.end() ; it ++)
        {
            it->print();
        }
        return 0;
    }

    第十题:

    标题: k倍区间
     
    给定一个长度为N的数列,A1, A2, ... AN,如果其中一段连续的子序列Ai, Ai+1, ... Aj(i <= j)之和是K的倍数,我们就称这个区间[i, j]是K倍区间。  
    你能求出数列中总共有多少个K倍区间吗?  
    输入
    -----
    第一行包含两个整数N和K。(1 <= N, K <= 100000)  
    以下N行每行包含一个整数Ai。(1 <= Ai <= 100000)  
    输出
    -----
    输出一个整数,代表K倍区间的数目。  
    例如,
    输入:
    5 2
    1  
    2  
    3  
    4  
    5  
    程序应该输出:
    6
    资源约定:
    峰值内存消耗(含虚拟机) < 256M
    CPU消耗  < 2000ms
    请严格按要求输出,不要画蛇添足地打印类似:“请您输入...” 的多余内容。
    注意:
    main函数需要返回0;
    只使用ANSI C/ANSI C++ 标准;
    不要调用依赖于编译环境或操作系统的特殊函数。
    所有依赖的函数必须明确地在源文件中 #include <xxx>
    不能通过工程设置而省略常用头文件。
     
    提交程序时,注意选择所期望的语言类型和编译器类型

    因为第一题花费太多时间了,所以后面就草草的看了剩下的三个编程题,感觉这个题还可以过上一部分的AC而且分值够大。。。就做了~~

    #include <string.h>
    #include <iostream>
    #include <stdio.h>
     
    using namespace std;
     
    typedef long long ll;                   // 题干中(1 <= N,K <= 100000)(1 <= Ai <= 100000) 
    ll bk[100010]={0};
    ll arr[100010];
    ll k,n;
    int main()
    {
        scanf("%lld%lld",&n,&k);
        for(int i = 0 ; i < n ; i ++)
            scanf("%lld",&arr[i]);
        arr[0] %= k;
        ll sum = 0;
        for(int i = 1 ; i < n ; i ++)
            arr[i] = (arr[i]+arr[i-1])%k;
        for(int i = 0 ; i < n ; i ++)
            sum += (bk[arr[i]]++);         //重了累加!!!
        printf("%lld
    ",sum+bk[0]);
        return 0;
    }

    第八题:

    标题:包子凑数
    
     
    
    小明几乎每天早晨都会在一家包子铺吃早餐。他发现这家包子铺有N种蒸笼,其中第i种蒸笼恰好能放Ai个包子。每种蒸笼都有非常多笼,可以认为是无限笼。
    
    每当有顾客想买X个包子,卖包子的大叔就会迅速选出若干笼包子来,使得这若干笼中恰好一共有X个包子。比如一共有3种蒸笼,分别能放3、4和5个包子。当顾客想买11个包子时,大叔就会选2笼3个的再加1笼5个的(也可能选出1笼3个的再加2笼4个的)。
    
    当然有时包子大叔无论如何也凑不出顾客想买的数量。比如一共有3种蒸笼,分别能放4、5和6个包子。而顾客想买7个包子时,大叔就凑不出来了。
    
    小明想知道一共有多少种数目是包子大叔凑不出来的。
    
     
    
    输入
    
    ----
    
    第一行包含一个整数N。(1 <= N <= 100)
    
    以下N行每行包含一个整数Ai。(1 <= Ai <= 100) 
    
     
    
    输出
    
    ----
    
    一个整数代表答案。如果凑不出的数目有无限多个,输出INF。
    
     
    
    例如,
    
    输入:
    
    2 
    
    4 
    
    5  
    
     
    
    程序应该输出:
    
    6 
    
     
    
    再例如,
    
    输入:
    
    2 
    
    4 
    
    6   
    
     
    
    程序应该输出:
    
    INF
    
     
    
    样例解释:
    
    对于样例1,凑不出的数目包括:1, 2, 3, 6, 7, 11。 
    
    对于样例2,所有奇数都凑不出来,所以有无限多个。 
    
     
    
    资源约定:
    
    峰值内存消耗(含虚拟机) < 256M
    
    CPU消耗  < 1000ms
    
     
    
     
    
    请严格按要求输出,不要画蛇添足地打印类似:“请您输入...” 的多余内容。
    
     
    
    注意:
    
    main函数需要返回0;
    
    只使用ANSI C/ANSI C++ 标准;
    
    不要调用依赖于编译环境或操作系统的特殊函数。
    
    所有依赖的函数必须明确地在源文件中 #include<xxx>
    
    不能通过工程设置而省略常用头文件。
    
     
    
    提交程序时,注意选择所期望的语言类型和编译器类型

    这道题,真的是好难....不会....

    大神说:

    这是扩展欧几里德变形的,有个定理。如果满足所有数的最大公约数不为1则有无穷个,否则都是有限个。然后利用完全背包就可以统计了

    代码:

    #include <string.h>
    #include <iostream>
    #include <stdio.h>
    using namespace std;
     
    int gcd(int a,int b)
    {
        if(b == 0) return a;
        return gcd(b,a%b);
    }
     
    int arr[110],n;
    const int N = 10010;
    bool bk[N];
     
    int main()
    {
        scanf("%d",&n);
        for(int i = 0 ; i < n ; i ++)
            scanf("%d",&arr[i]);
        int g = arr[0];
        for(int i = 1 ; i < n ; i ++)
            g = gcd(g,arr[i]);
        if(g != 1)
        {
            printf("INF
    ");
        }else{
            bk[0] = true;
            for(int i = 0 ; i < n ; i ++)
            {
                for(int j = 0 ; j + arr[i] < N ; j ++)
                    if(bk[j])bk[j+arr[i]]= true;
            }
            int count = 0;
            for(int i = N-1 ; i >= 0 ; i --){
                if(bk[i] == false) count++;
            }
            printf("%d
    ",count);
        }
        return 0;
    }

    第九题:

    标题: 分巧克力
     
        儿童节那天有K位小朋友到小明家做客。小明拿出了珍藏的巧克力招待小朋友们。
        小明一共有N块巧克力,其中第i块是Hi x Wi的方格组成的长方形。
     
        为了公平起见,小明需要从这 N 块巧克力中切出K块巧克力分给小朋友们。切出的巧克力需要满足:
     
        1. 形状是正方形,边长是整数  
        2. 大小相同  
     
    例如一块6x5的巧克力可以切出6块2x2的巧克力或者2块3x3的巧克力。
     
    当然小朋友们都希望得到的巧克力尽可能大,你能帮小Hi计算出最大的边长是多少么?
     
    输入
    第一行包含两个整数N和K。(1 <= N, K <= 100000)  
    以下N行每行包含两个整数Hi和Wi。(1 <= Hi, Wi <= 100000) 
    输入保证每位小朋友至少能获得一块1x1的巧克力。   
    输出
    输出切出的正方形巧克力最大可能的边长。
    样例输入:
    2 10  
    6 5  
    5 6  
    样例输出:
    2
    资源约定:
    峰值内存消耗(含虚拟机) < 256M
    CPU消耗  < 1000ms
    请严格按要求输出,不要画蛇添足地打印类似:“请您输入...” 的多余内容。
    注意:
    main函数需要返回0;
    只使用ANSI C/ANSI C++ 标准;
    不要调用依赖于编译环境或操作系统的特殊函数。
    所有依赖的函数必须明确地在源文件中 #include <xxx>
    不能通过工程设置而省略常用头文件。
     
    提交程序时,注意选择所期望的语言类型和编译器类型。

    当时看完这个题也心动了,但时间有点来不及了,就挑分值大点的做了。下来的时候自己把它实现了一下...........

    代码:

    #include <algorithm>
    #include <string.h>
    #include <iostream>
    #include <stdio.h>
     
    using namespace std;
     
    typedef long long ll;
    ll hi[100010];
    ll wi[100010];
     
    int n;  //n块巧克力
    ll k;   //k个小朋友
    bool is_ok(ll ans)
    {
        ll sum = 0;
        for(int i = 0 ; i < n ; i ++)
        {
            sum += (hi[i]/ans)*(wi[i]/ans);
            if(sum >= k) 
                return true; 
        }
        return false;
    }
    int main()
    {
        scanf("%d%lld",&n,&k);
        for(int i = 0 ; i < n ; i ++)
            scanf("%lld%lld",&hi[i],&wi[i]);
        ll big_size = 1,r = 100000,ans;
        while(big_size <= r)
        {
            ans = (big_size + r)/2;
            if(is_ok(ans))
                big_size = ans + 1;
            else 
                r = ans - 1;
        }
        big_size++;
        while(big_size--)
        {
            if(is_ok(big_size))
                break;
        }
        printf("%lld
    ",big_size);
        return 0;
    }
  • 相关阅读:
    做汉堡
    作业三 读《构建之法》
    一个程序员的生命周期--有感
    阅读《构建之法》第13-17章
    阅读<构建之法>10、11、12章
    阅读《构建之法》第8,9,10章
    测试与封装5.2-5.3
    作业5 四则运算 测试与封装 5.1
    阅读5-7章
    做汉堡
  • 原文地址:https://www.cnblogs.com/cjn123/p/10665470.html
Copyright © 2011-2022 走看看