【OJ】机试指南 第一章

机试指南

第一章 从零开始

OJ 网站

做题结果反馈

  • Accepted

    • 答案正确
  • Wrong Answer

    • 答案错误:程序实现或者思路出现问题,也可能是数据范围边界没有考虑到
  • Runtime Error

    • 运行时错误:数组越界或者递归过深导致栈溢出
  • Presentation Error

    • 输出格式错误:一般是末尾多了或少了空格或少了换行
  • Time Limit Exceeded

    • 程序运行超时:算法不够优秀,程序运行时间过长
  • Memory Limit Exceeded

    • 运行内存超限:程序申请太大了空间,超过了题目规定的空间大小
  • Compile Error

    • 编译错误:代码存在语法错误,检查是不是选择错误的语言提交了
  • Output Limit Exceeded

    • 输出超限:程序输出过多的内容,一般是循环出了问题导致多次输出或者调试信息忘记删除了
  • Submitting

    • 提交中,等待题目结果的返回,由于判题机有性能差异所以返回结果也不一样

输入输出技巧

//输入int型变量
scanf("%d",&x);
//输入double型变量
scanf("%lf",&x);//不用float直接double
//输入char类型变量
scanf("%c",&x);
//输入字符串数组
scanf("%s",s);
//输出与输入表示方式一致
printf("%s\n",s);
  • scanf 输入解析

    • 输入日期 2020 - 03 - 21

    • int year,month,day;
      scanf("%d-%d-%d",&year,&month,&day);
      printf("%d %d %d\n",year,month,day);
      
    • 输入时间 18 : 21 : 30

    • int hour,minute,second;
      scanf("%d:%d:%d",&hour,&minute,&second);
      printf("%d %d %d\n",hour,minute,second);
      
  • scanf 和 gets

输入一行字符串带空格的话,使用 gets,scanf 遇到空格会自动结束

char s[105];
gets(s);
printf("%s\n",s);
  • getchar 和 putchar

读入单个字符和输出单个字符,一般在 scanf 和 gets 中间使用 getchar 用于消除回车 '\n' 的影响

  • 输出进制转换
int a = 10;
printf("%x\n",a);//小写十六进制输出a
printf("%X\n",a);//大写十六进制输出A
printf("%o\n",a);//八进制输出答案12
  • 输出增加前置0
int a = 5;
printf("%02d\n",a);//2代表宽度,不足的地方用0补充
输出结果 05
printf("%04d\n",a);
输出结果 0005
  • 输出保留小数
double a = 3.6;
printf("%.2lf\n",a);//2表示保留两位小数
输出结果 3.60

注:有小数输出小数,没小数输出整数,前提是输入的类型浮点类型

%g

Example:

#include <stdio.h>
int main(){
    
    double a = 2;
    double b = 2.5467;
    printf("a = %g\n",a);
    printf("b = %g\n",b);
    
    return 0;
} 

Test Result:

image-20200321110921091
  • long long 的使用

很多情况下的计算会超过 int,比如求 N!,N 比较大的时候 int 就存不下了,这个时候我们就要使用 long long。

注:int 取值范围:-1e9 到 1e9,long long 取值范围:-1e18 到 1e18

long long x;
scanf("%11d",&x);
printf("%lld\n",x);
  • 字符的 ASCII 码
printf("%d\n",'a');

注:若遇到需要 ASCII 码的题目,记住 char 字符和 int 值是可以相互转化的。

  • cin 和 cout

很多时候使用 c++ 的输入输出更简单,在应对输入输出量不是很大的题目的时候,我们会采用 cin 和 cout 来提高我们的解题速度。

Example:求两个数之和

#include <iostream>//输入输出函数的头文件 
using namespace std;

int main(){
   
   int a,b;
   cin >> a >> b;
   cout << a + b;
   
   return 0;
} 

注:平时练习的时候不要排斥混合编程,即 C 与 C++ 混用,然后用 C++ 提交。但是注意:printf 尽量不要和 cout 同时使用,会发生一些不可控的意外。

头文件技巧

cpp 文件中推荐一个万能头文件:

#include <bits/stdc++.h>
using namespace std;

注:要看考试的评测机型支不支持,绝大部分都是支持的,当然准备一个完整的头文件还是有必要的,作为备用。

#include <stdio.h>
#include <string.h>
#include <math.h>
#include <stdlib.h>
#include <time.h>
#include <algorithm>
#include <iostream>
#include <queue>
#include <stack>
#include <vector>
#include <string>
using namespace std;

int main(){
    
    return 0;
} 

注:头文件可以多,但是不能少

数组使用技巧

数组除了可以存储数据以外,还可以用来进行标记

例题 01:

image-20200321120529994

【代码实现】

#include <bits/stdc++.h>//万能头文件 
using namespace std;

int f[105]={0};//注意:尽量将数组开在全局 
int main(){
    int n,x;
    scanf("%d",&n);
    for(int i=0;i<n;i++){
        scanf("%d",&x);
        f[x]++;
    } 
    for(int i=1;i<=100;i++){
        if(f[i]>0){
            printf("%d %d\n",i,f[i]);
        }
    }
    return 0;
} 

例题 02:

image-20200321134145614

【代码实现】

#include <bits/stdc++.h>//万能头文件 
using namespace std;

int f[105] = {0};//注意:尽量将数组开在全局 
int p[105] = {0};//p[i]表示有 i个这样的数的最大值是多少 
int main(){
    int n,x;
    scanf("%d",&n);
    for(int i=0;i<n;i++){
        scanf("%d",&x);
        f[x]++;
    } 
    for(int i=0;i<100;i++){
        p[f[i]] = i;
    } 
    for(int i=1;i<=100;i++){
        if(p[i]>0){
            printf("%d %d\n",p[i],i);
        }
    }
    return 0;
} 

例题 03:

【题目】二维数组实现存储地图

####
#.##
##@#
####

【代码实现】

#include <bits/stdc++.h>//万能头文件 
using namespace std;

char mpt[10][10];
int main(){
    for(int i=0;i<4;i++){
        scanf("%s",mpt[i]);
    }
    for(int i=0;i<4;i++){
        for(int j=0;j<4;j++){
            printf("%c",mpt[i][j]);
        }
        printf("\n");
    }

    return 0;
} 

复杂度与是否可做

做好审时度势

算法 循环嵌套 时间复杂度 空间复杂度
冒泡排序 两个 for 循环 O(N^2​)

注:空间复杂度一般不会限制,如果遇到了再想办法优化空间。

时限 1 s 情况下的复杂度:

时间复杂度 N 取值(时限 1 s)
O(N) N 最大在 500w 左右
O(NlogN​) N 最大在 20w 左右
O(N^2​) N 最大在 2000 左右
O(N^2logN​) N 最大在 700 左右
O(N^3​) N 最大在 200 左右
O(N^4​) N 最大在 50 左右
O(2^N​) N 最大在 24 左右
O(N!)​ N 最大在 10 左右

注:如果是 2S、3S 对应的乘以 2 和 3 就可以。

C++ STL 的使用

?> C++ 的算法头文件里有很多实用的函数,我们可以直接拿来用。

#include <algorithm>

排序函数

  • sort()

  • 依次传入三个参数,要排序区间的起点,要排序区间的终点+1,比较函数。比较函数可以不填,则默认为从小到大排序

  • 示例:

  • #include <bits/stdc++.h>
    using namespace std;
    
    int a[105];
    int main(){
        int n;
        scanf("%d",&n);
        for(int i=0;i<n;i++){
          scanf("%d",&a[i]);
      }
      sort(a,a+n);
      for(int i=0;i<n;i++){
          printf("%d ",a[i]);
      }
        return 0;
    }
    

查找函数

  • lower_bound() 函数

  • upper_bound() 函数

  • lower_bound() 和 upper_bound() 都是利用二分查找的方法在一个排好序的数组中进行查找的。

  • 在从小到大的排序数组中:

  • lower_bound(begin,end,num):从数组的 begin 位置到 end-1 位置二分查找第一个大于或等于 num 的数字,找到返回该数字的地址,不存在则返回 end。通过返回的地址减去起始地址 begin,得到找到数字在数组中的下标

  • upper_bound(begin,end,num):从数组的 begin 位置到 end-1 位置二分查找第一个大于 num 的数字,找到返回该数字的地址,不存在则返回 end。通过返回的地址减去起始地址 begin,得到找到数字在数组中的下标

  • 重载 lower_bound() 和 upper_bound()

  • lower_bound(begin,end,num,greater< type >()):

  • 从数组的 begin 位置到 end-1 位置二分查找第一个小于或等于 num 的数字,找到返回该数字的地址,不存在则返回 end。通过返回的地址减去起始地址 begin,得到找到数字在数组中的下标

  • upper_bound(begin,end,num,greater< type >()):

  • 从数组的 begin 位置到 end-1 位置二分查找第一个小于 num 的数字,找到返回该数字的地址,不存在则返回 end。通过返回的地址减去起始地址 begin,得到找到数字在数组中的下标

  • 示例:

  • //查找函数
    #include <bits/stdc++.h>
    using namespace std;
    
    int cmp(int a,int b);
    int main(){
        int num[6]={1,2,4,7,15,34};
        sort(num,num+6);//按从小到大排序
        //返回数组中第一个大于或等于被查数的值
      int pos1 = lower_bound(num,num+6,7)-num;
      //返回数组中第一个大于被查数的值
      int pos2 = upper_bound(num,num+6,7)-num;
      cout<<pos1<<" "<<num[pos1]<<endl;
      cout<<pos2<<" "<<num[pos2]<<endl;
      sort(num,num+6,cmp);//按从大到小排序
      //返回数组中第一个小于或等于被查数的值
      int pos3 = lower_bound(num,num+6,7,greater<int>())-num;
      //返回数组中第一个小于被查数的值
      int pos4 = upper_bound(num,num+6,7,greater<int>())-num;
      cout<<pos3<<" "<<num[pos3]<<endl;
      cout<<pos4<<" "<<num[pos4]<<endl;
       
        return 0;
    }
    
    int cmp(int a,int b){
      return a>b;
    }
    

优先队列

  • 通过 priority_queue< int >q 来定义一个储存整数的空的 priority_queue。当然 priority_queue 可以存任何类型的数据,比如 priority_queue< string >q 等等。

  • 示例:

  • //#include <bits/stdc++.h>
    #include <iostream>
    #include <queue>
    using namespace std;
    
    int main(){
        priority_queue<int> q;//定义一个优先队列
      q.push(1);//入队 
      q.push(2);
      q.push(3);
      while(!q.empty()){//判读队列不为空 
          cout << q.top() << endl; //队首元素 
          q.pop();//出队 
      } 
        return 0;
    }
    

C++ 的 STL (标准模板库)是一个非常重要的东西,可以极大的帮助更快速的解决题目

vector

  • 通过 vector< int >v 来定义一个储存整数的空的 vector。当然 vector 可以存任何类型的数据,比如 vector< string >v 等等

  • 示例:

  • //#include <bits/stdc++.h>
    #include <iostream>
    #include <vector>
    using namespace std;
    
    int main(){
        vector<int> v;//定义一个空的 vector
      for(int i=1;i<=10;i++){
          v.push_back(i*i);//加入到 vector 中 
      } 
      for(int i=0;i<v.size();i++){
          cout<<v[i]<<" ";//访问 vector 元素 
      } 
      cout << endl; 
        return 0;
    }
    
  • Result:

image-20200322201845985

queue

  • 通过 queue< int >q 来定义一个储存整数的空的 queue。当然 queue 可以存任何类型的数据,比如 queue< string >q 等等

  • 示例:

  • #include <iostream>
    #include <queue>
    using namespace std;
    int main()
    {
      queue<int> q;//定义一个队列
      q.push(1);
      q.push(2);
      q.push(3);
      while(!q.empty()){//当队列不为空 
          cout << q.front() << endl;//取出队首元素
          q.pop();//出队 
      } 
      
      return 0;
    }
    

stack

  • 通过 stack< int >S 来定义一个全局栈来储存整数的空的 stack。当然 stack 可以存任何类型的数据,比如 stack< string >S 等等。

  • 示例:

  • #include <iostream>
    #include <stack>
    using namespace std;
    int main()
    {
      stack<int> s;//定义一个栈 
      s.push(1);
      s.push(10);
      s.push(7);
      while(!s.empty()){//当栈不为空 
          cout << s.top() << endl;//取出栈顶元素
          s.pop();//出栈 
      } 
      
      return 0;
    }
    

map

  • 通过 map< string, int >dict 来定义一个 key:value 映射关系的空的 map。当然 map 可以存任何类型的数据,比如 map< int, int >m 等等

  • 示例:

  • #include <iostream>
    #include <string>
    #include <map>
    
    using namespace std;
    int main()
    {
      map<string, int> dict;//定义一个 map
      dict["Tom"] = 1;//定义映射关系
      dict["Jone"] = 2;
      dict["Mary"] = 1;
      if(dict.count("Mary")){//查找 map
          cout << "Mary is in class " << dict["Mary"]; 
      } 
      //使用迭代器遍历map和value
      for(map<string, int>::iterator it = dict.begin();it != dict.end();++it){
          cout << it->first << " is in class " << it->second << endl;
      } 
      dict.clear();//清空 map 
      return 0;
    }
    

set

  • 通过 set< string >country 来定义一个储存字符串的空的 set。当然 set 可以存任何类型的数据,比如 set< int >s 等等。

  • #include<iostream>
    #include<set>
    using namespace std;
    int main(){
      set<string> country;//定义一个存放string的集合
      country.insert("China");//插入操作
      country.insert("America");
      country.insert("France");
      set<string>::iterator it;
      //使用迭代器遍历集合元素
      for(it=country.begin();it!=country.end();++it){
          cout<<*it<<" ";
      }
      cout<<endl;
      country.erase("American");//删除集合内的元素
      country.erase("England");
      if(country.count("China")){//统计元素个数
          cout<<"China in country."<<endl;
      }
      country.clear();//清空集合
      return 0;
    }
    

多组输入的问题

即循环输入输出结果

【题目描述】输入两个数,输出两个数的和,要求多组输入

【代码实现】

  • Example 01:C 循环读入代码
#include <bits/stdc++.h> 
using namespace std;
int main(){
    int a,b;
    while(scanf("%d%d",&a,&b)!=EOF){
        printf("%d\n",a+b);
    }
    return 0;
}

!> 不能使用 while(1) 这样死循环,!=EOF 的意思一直读取到文件末尾(Endoffile)另外,多组输入一定要注意初始化问题,数组和变量的初始化要放在 while 循环内,否则上一次的运算的结果会影响当前的结果。

  • Example 02:C++ 循环读入代码
#include <bits/stdc++.h> 
using namespace std;
int main(){
    int a,b;
    while(cin >> a >> b){
        cout << a + b << endl;
    }
    return 0;
}
  • Example 03:Java 循环读入代码
Scanner stdin = new Scanner(System.in);
while(stdin.hasnext()){
    String s = stdin.next();
    int n = stdin.nextInt();
    double b = stdin.nextDouble();
}
  • Example 04:Python 循环读入代码
while true:
    try:
        a, b = map(int, input().split())
        c = a + b
        print(c)
    except:#读到文件末尾抛出异常结束循环
        break

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,039评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,223评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,916评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,009评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,030评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,011评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,934评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,754评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,202评论 1 309
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,433评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,590评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,321评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,917评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,568评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,738评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,583评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,482评论 2 352

推荐阅读更多精彩内容