数据结构与算法计算机杂谈每天写1000字

【计算机本科补全计划】CCF计算机职业资格认证 2017-03

2017-10-17  本文已影响155人  张照博

正文之前

我在之前的文章中提到过,我的老师要求我的CCF 考试考个280分来打个底,(没错,我就是那个横跨考研、工作、保研三大领域的男人)相当于是测试下我的能力,所以虽然不知道近期有没有相关的考试,但是我还是开始准备。这种等级考试,当然就是从刷题开始了!!至于什么大纲,什么宝典,见鬼去吧~ 不信这玩意,题海战术从小用到大,骨子里都习惯了。当然还是直接怼题目来得爽了 ~ ~ 而且还可以实践自己的各种知识积淀,自己看书看一遍,简书写笔记写一遍,最后写题写一遍,考试然后再被轮一遍,这么下来还没有十足长进我就不信了!!

正文


2017-03-01题

试题编号 2017-03-1
试题名称 分蛋糕
时间限制 1.0 s
内存要求 256M

本题标准答案(提交后分数为100分)

/* CCF201703-1 分蛋糕 */  
  
#include <iostream>  
  
using namespace std;  
  
int main()  
{  
    int n, k, count=0, val, sub=0;  
  
    cin >> n >> k;  
    for(int i=1; i<=n; i++) {  
        cin >> val;  
  
        if((sub += val) >= k) {  
            count++;  
            sub = 0;  
        }  
    }  
    if(sub > 0)  
        count++;  
  
    cout << count << endl;  
  
    return 0;  
}  

我的答案:

#include <vector>
#include <iostream>
using namespace std;

int main()
{
    int n,k;
    cout<<"First Line:";
    cin>>n>>k;
    vector<int > cake;
    cout<<"Second Line:";
    while(n--)
    {
        int x;
        cin>>x;
        cake.push_back(x);
    }
    int Num=0,weight=0;
    for (vector<int>::const_iterator iter = cake.begin();iter != cake.end(); ++iter)
    {
        weight+=*iter;
        if (weight>=k)
        {
            ++Num;
            weight=0;
        }
    }
    if(weight!=0)
        ++Num;
    cout<<"一共  "<<Num<<"  人获得了  "<<k<<" g 重量的蛋糕"<<endl;
    return 0;
}

可以看到,思想很接近,只是我做了些许的修缮,所以显得很长,另外,满分答案是直接在输入的时候进行了计算,没有使用数组,向量等容器,确实是优秀答案。而我则是多用了一个vector向量,并且把输入与计算分离(因为有了容器存储,自然可以分离)。好吧,后面拿来一看。发现我的就是个弱鸡,人家对内存的要求比我低得多!!而且最关键的是:速度比我的快,但是我算了下时间复杂度,应该没太大的差别才对,难道读写向量很困难????那我就卧了个大槽了~~~~

搞了半天输入有点偏差,所以后来自己动手写了120个混乱数字,然后直接复制进入算作输入,120个数字,占用了608k,至于用时,我个人感觉还挺快的。

时间我没办法。只能是用sublime的gcc自带的计时,算出来不到一秒。感动~ 但是每次超过500感觉就会直接爆炸,程序直接原地不动了。最后exit 9 ,估计是内存不够?不至于哇!不管了不管了!


完成编译使用时间是0.6s,然后运行的时候我感觉还是很快的,但是这个内存泄漏的问题估计也就是上面的那种满分解法了~ 不知道数组怎么样。我试试。好吧,一样的,看样子vector跟数组都突破不了500这个大关???那怎么办。这会0分的吧!!!三题我要280!!!怎么破??没办法!只能继续怼了!


2017-03-02题

试题编号 2017-03-1
试题名称 学生排队
时间限制 1.0 s
内存要求 256M

提交后得一百分的C++版本的代码(简洁版):

/* CCF201703-2 学生排队 */  
  
#include <iostream>  
  
using namespace std;  
  
const int N = 1000;  
  
int sno2pos[N+1];   // 学号所在位置  
int pos2sno[N+1];   // 位置上的学号  
  
int main()  
{  
    int n, m, p, q;  
  
    // 输入数据  
    cin >> n >> m;  
  
    // 初始化  
    for(int i=1; i<=n; i++) {  
        sno2pos[i] = i;  
        pos2sno[i] = i;  
    }  
  
    // 模拟移动过程  
    for(int i=1; i<=m; i++) {  
        int pos1, pos2, sno2;  
  
        cin >> p >> q;  
  
        if(q != 0) {  
            int move = (q > 0) ? 1 : -1;  
            int end = (q > 0) ? q : -q;  
  
            pos1 = sno2pos[p];  
            for(int i=1; i<=end; i++) {  
                sno2 = pos2sno[pos1 + move];  
                pos2 = sno2pos[sno2];  
  
                pos2sno[pos1] = sno2;  
                sno2pos[sno2] = pos1;  
  
                pos1 = pos2;  
            }  
            pos2sno[pos2] = p;  
            sno2pos[p] += q;  
        }  
    }  
  
    // 输出结果  
    cout << pos2sno[1];  
    for(int i=2; i<=n; i++)  
        cout << " " << pos2sno[i];  
    cout << endl;  
  
    return 0;  
}  

复杂版本的100分标准C++答案代码:

#include <iostream>  
  
using namespace std;  
  
//#define DEBUG  
  
const int N = 1000;  
  
int sno2pos[N+1];   // 学号所在位置  
int pos2sno[N+1];   // 位置上的学号  
  
int main()  
{  
    int n, m, p, q;  
  
    cin >> n >> m;  
  
    for(int i=1; i<=n; i++) {  
        sno2pos[i] = i;  
        pos2sno[i] = i;  
    }  
  
    for(int i=1; i<=m; i++) {  
        int pos1, pos2, sno2;  
  
        cin >> p >> q;  
  
        if(q != 0) {  
            int move, end;  
            if(q > 0) {  
                move = 1;  
                end = q;  
            } else {  
                move = -1;  
                end = -q;  
            }  
              
            pos1 = sno2pos[p];  
            for(int i=1; i<=end; i++) {  
                sno2 = pos2sno[pos1 + move];  
                pos2 = sno2pos[sno2];  
  
                pos2sno[pos1] = sno2;  
                sno2pos[sno2] = pos1;  
  
                pos1 = pos2;  
            }  
            pos2sno[pos2] = p;  
            sno2pos[p] += q;  
        }  
  
//        if(q > 0) {  
//            pos1 = sno2pos[p];  
//            for(int i=1; i<=q; i++) {  
//                sno2 = pos2sno[pos1+1];  
//                pos2 = sno2pos[sno2];  
  
//                pos2sno[pos1] = sno2;  
//                sno2pos[sno2] = pos1;  
  
//                pos1 = pos2;  
//            }  
//            pos2sno[pos2] = p;  
//            sno2pos[p] += q;  
//        } else if(q < 0){  
//            pos1 = sno2pos[p];  
//            for(int i=1; i<=-q; i++) {  
//                sno2 = pos2sno[pos1-1];  
//                pos2 = sno2pos[sno2];  
  
//                pos2sno[pos1] = sno2;  
//                sno2pos[sno2] = pos1;  
  
//                pos1 = pos2;  
//            }  
//            pos2sno[pos2] = p;  
//            sno2pos[p] += q;  
//        }  
#ifdef DEBUG  
        cout << "son: ";  
        for(int i=1; i<=n; i++)  
            cout << pos2sno[i] << " ";  
        cout << endl;  
        cout << "pos: ";  
        for(int i=1; i<=n; i++)  
            cout << sno2pos[i] << " ";  
        cout << endl;  
#endif  
    }  
  
    cout << pos2sno[1];  
    for(int i=2; i<=n; i++)  
        cout << " " << pos2sno[i];  
    cout << endl;  
  
    return 0;  
}  

我的答案:

#include <iostream>
using namespace std;

int main()
{
    int a,b,m,n;
    cin>>n;
    int s[n+1];
    for (int i = 1; i <=n; ++i)
        s[i]=i;
    cin>>m;
    while(m--)
    {
        int x,y;
        cin>>x>>y;
        a=s[x];  
        if(s[x]<=y)
        {
            s[x]=1;
            y=s[x]-1;
        }
        else // if (y>0||s[x]<y)
           s[x]-=y;        
        for(int i=1;i<=n;++i)
        {
            if (i!=x)
            {
                if(s[i]<=a&&s[i]>=(a-y))
                    ++s[i];
            }
        }
       
    }
    for (int i =1 ; i <=n; ++i)
        for (int j = 1; j <= n; ++j)
        {
            if(s[j]==i)
                cout<<j;
        }
    return 0;
}


下面推演一下看看正确性:
1 2 3 4 5 6 这是本来的顺序
1 4 2 3 5 6 第一次 4 2 之后的顺序
1 4 3 2 5 6 第二次 3 1 之后的顺序
1 4 3 6 2 5 第三次 6 2 之后的顺序
完美符合题目要求,全体仅仅使用一个数组用于存储。时间复杂度为O(n^2)不知道时间上能否通过要求,空间复杂度的话 256M应该是够用了!

我因为时间所迫。所以只做了正向移动,也就是只向前,不向后的运动!思想与标准答案千里之差,但是我觉得我的比较简洁而且看起来应该简单易懂一些!


正文之后

程序改变现实,软件统治世界。
程序员需要有精益求精的工匠精神,追求逻辑的极简、时间的最少和存储的最省,并且懂得其中的平衡。
数据表示需要优先考虑,对于许多问题,找到表示该问题的数据结构,问题自然就解决了。
CCF计算机职业资格认证的每一道试题都十分经典,覆盖现实世界中方方面面的问题。这个历年试题解全部用C++语言编写,程序中附有注释,力求解题思路清晰简洁,值得珍藏与模仿。
希望获得100分,仅仅使用原题的样例来测试是不够的,需要自己设计一些样例,并且需要考虑特殊的边界条件。
使用STL的包装类和算法是十分必要,这会简化程序逻辑。
---以上出自某提供试题与答案的博客~

上一篇 下一篇

猜你喜欢

热点阅读