1、(C+版)NOIP 复习资料主 编 葫芦岛市一高中 李思洋完成日期 2012 年 8 月 27 日前 言有一天,我整理了 NOIP 的笔记,并收集了一些经典算法。不过我感觉到笔记比较凌乱,并且有很多需要修改和补充的内容,于是我又搜集一些资料,包括一些经典习题,在几个月的时间内编写出了NOIP 复习资料。由于急于在假期之前打印出来并分发给同校同学(我们学校既没有竞赛班,又没有懂竞赛的老师。我们大家都是自学党),NOIP 复习资料 有很多的错误,还有一些想收录而未收录的内容。在“减负”的背景下,暑期放了四十多天的假。于是我又有机会认真地修订NOIP 复习资料。我编写资料的目的有两个:总结我学过(包
2、括没学会)的算法、数据结构等知识;与同学共享 NOIP 知识,同时使我和大家的 RP+。大家要清醒地认识到,NOIP 复习资料页数多,是因为程序代码占了很大篇幅。这里的内容只是信息学的皮毛。对于我们来说,未来学习的路还很漫长。基本假设作为自学党,大家应该具有以下知识和能力: 能够熟练地运用 C+语言编写程序(或熟练地把 C+语言“翻译”成 Pascal 语言); 能够阅读代码,理解代码含义,并尝试运用; 对各种算法和数据结构有一定了解,熟悉相关的概念; 学习了高中数学的算法、数列、计数原理,对初等数论有一些了解; 有较强的自学能力。代码约定N、M 、 MAX、INF 是事先定义好的常数( 不会
3、在代码中再次定义,除非代码是完整的程序 )。N 、M、MAX针对数据规模而言,比实际最大数据规模大;INF 针对取值而言,是一个非常大,但又与 int 的最大值有一定差距的数,如 100000000。对于不同程序,数组下标的下限也是不同的,有的程序是 0,有的程序是 1。阅读程序时要注意。阅读顺序和方法没听说过 NOIP,或对 NOIP 不甚了解的同学,应该先阅读附录 E,以加强对竞赛的了解。如果不能顺利通过初赛,你就应该先补习初赛知识。这本NOIP 复习资料总结的是复赛知识。如果没有学过 C+语言,应该先选择一本 C+语言教材。一般情况下,看到“面向对象编程”一章的前一页就足够了(NOIP
4、不用“面向对象编程 ”,更不用摆弄窗口对话框 )。附录 G 介绍了一些书籍和网站。你应该选择一本书,认真地学习。再选择一个网站,作为练习的题库。第一单元对竞赛中常用的操作和简单的算法分析进行了总结,算作对 C+语言的巩固。同时,阅读这一单元之后,你应该选择一个合适的 C+代码编辑器。第二到第六单元介绍了竞赛常用的算法。阅读每一章时,应该先阅读“小结”名曰“小结”,实际上是“导读”。这五个单元除了经典习题,还有某些思想和算法的具体实现方法。这些信息可能在明处,也可能在暗处,阅读时要注意挖掘和体会。如果有时间,应该在不看解析和代码的前提下独立完成这些题。第七单元是第六单元的一个部分,由于它的内容来
5、自背包九讲,所以单独放在一个单元。从第八单元开始,到第十三单元,基本上就没有习题了。换句话说,该“背课文”了。第八单元介绍了常用的排序算法。你可以有选择地学习,但一定要掌握“STL 算法”和“快速排序”。第九单元介绍了基本数据结构,你一定要掌握第九单元前五小节的内容(本单元也有应该优先阅读的“小结”)。有余力的话,第六小节的并查集也应该掌握。第十单元介绍了与查找、检索有关的数据结构和算法。你也可以有选择地学习。第十一单元与数学有关。数学对于信息学来说具有举足轻重的地位。标有“!”的应该背下来,至于其他内容,如果出题,你应该能把它解决。第十二单元仍与数学有关。第十三单元是图论。学习时要先阅读“小
6、结”,把概念弄清楚。之后要掌握图的实现方法。接下来要掌握一些经典图论算法:Kruskal 算法、Dijkstra 算法、 SPFA、Floyd 算法、拓扑排序。附录 F 总结了 2004 年以来 NOIP 考察的知识点,可以作为选择性学习的参考。在学习算法和数据结构的同时,应该阅读和学习附录 A。如果你还有余力,你应该学习第十四单元。第十四单元的内容不是必须要掌握的,但是一旦学会,可以发挥 C+语言的优势,降低编程复杂度。临近竞赛时,应该阅读附录 B 和附录 C,以增加经验,减少失误。面临的问题1. 这是复赛复习资料需要有人能用心总结、整理初赛的知识,就像这份资料一样。2. 潜在的问题还是相当
7、多的,只是时间不够长,问题尚未暴露。3. 部分代码缺少解说,或解说混乱。4. 个人语文水平较差,资料也是如此。5. 没有对应的 Pascal 语言版本。如果有人能为 P 党写一个 Pascal 版的 STL,他的 RP 一定会爆增!6. 希望有人能用 整理资料,并以自由文档形式发布。LA T E X最后,欢迎大家以交流、分享和提高为目的修改、复制、分发本资料,同时欢迎大家将资料翻译成 Pascal 语言版供更多 OIer 阅读!谢谢大家的支持!葫芦岛市一高中 李思洋2012 年 8 月 27 日目 录标题上的符号:1. !:表示读者应该熟练掌握这些内容,并且在竞赛时能很快地写出来。换句话说就是
8、应该背下来。2. *:表示内容在 NOIP 中很少涉及,或者不完全适合 NOIP 的难度。3. #:表示代码存在未更正的错误,或算法本身存在缺陷。前 言 1目 录 I第一单元 C+语言基础 11.1 程序结构 .11.2 数据类型 .41.3 运算符 61.4 函数 81.5 输入和输出! 91.6 其他常用操作! .101.7 字符串操作! .131.8 文件操作! .131.9 简单的算法分析和优化 141.10 代码编辑器 .16第二单元 基础算法 172.1 经典枚举问题 172.2 火柴棒等式 182.3 梵塔问题 192.4 斐波那契数列 192.5 常见的递推关系! 202.6
9、选择客栈 222.7 2k进制数 .232.8 Healthy Holsteins .242.9 小结 .25第三单元 搜索 273.1 N 皇后问题 273.2 走迷宫 .293.3 8 数码问题 313.4 埃及分数 343.5 Mayan 游戏 .363.6 预处理和优化 403.7 代码模板 413.8 搜索题的一些调试技巧 433.9 小结 .44第四单元 贪心算法 464.1 装载问题 464.2 区间问题 464.3 删数问题 474.4 工序问题 474.5 种树问题 474.6 马的哈密尔顿链 .474.7 三值的排序 494.8 田忌赛马 504.9 小结 .50第五单元
10、分治算法 515.1 一元三次方程求解 .515.2 快速幂 .515.3 排序 .515.4 最长非降子序列 .535.5 循环赛日程表问题 .535.6 棋盘覆盖 545.7 删除多余括号 555.8 聪明的质监员 565.9 模板 .585.10 小结 59第六单元 动态规划 606.1 导例:数字三角形 .606.2 区间问题:石子合并 .636.3 坐标问题 656.4 背包问题 676.5 编号问题 676.6 递归结构问题 686.7 DAG 上的最短路径 .716.8 树形动态规划* .726.9 状态压缩类问题:过河 746.10 Bitonic 旅行 766.11 小结 7
11、7第七单元 背包专题 787.1 部分背包问题 787.2 0/1 背包问题! .787.3 完全背包问题 797.4 多重背包问题 797.5 二维费用的背包问题 .807.6 分组的背包问题 .817.7 有依赖的背包问题 .817.8 泛化物品 817.9 混合背包问题 827.10 特殊要求 .827.11 背包问题的搜索解法 837.12 子集和问题 .84第八单元 排序算法 858.1 常用排序算法 858.2 简单排序算法 878.3 线性时间排序 888.4 使用二叉树的排序算法* .898.5 小结 .90第九单元 基本数据结构 .919.1 线性表(顺序结构) .919.2
12、 线性表(链式结构) .919.3 栈 .939.4 队列 .949.5 二叉树 .959.6 并查集! 999.7 小结 102第十单元 查找与检索 10410.1 顺序查找 10410.2 二分查找 !.10410.3 查找第 k 小元素! .10510.4 二叉排序树 10610.5 堆和优先队列 * .10810.6 哈夫曼( Huffman)树 .11010.7 哈希( Hash)表 111第十一单元 数学基础 11611.1 组合数学 11611.2 组合数的计算 ! .11711.3 排列和组合的产生(无重集元素)! 11711.4 排列和组合的产生(有重集元素) .12011.
13、5 秦九韶算法 12211.6 进制转换(正整数) .12311.7 高精度算法(压位存储)! .12311.8 快速幂 ! 12811.9 表达式求值 12911.10 解线性方程组 * 133第十二单元 数论算法 13512.1 同余的性质 !.13512.2 最大公约数、最小公倍数! .13512.3 解不定方程 axby c!* .13512.4 同余问题 *.13612.5 素数和素数表 13612.6 分解质因数 137第十三单元 图与图论算法 13913.1 图的实现 13913.2 图的遍历 14113.3 连通性问题 14213.4 欧拉回路 邻接矩阵 .14613.5 最小
14、生成树 (MST) 14713.6 单源最短路问题 (SSSP 问题) .14813.7 每两点间最短路问题(APSP 问题)!15213.8 拓扑排序 15213.9 关键路径 15513.10 二分图初步 .15713.11 小结 160第十四单元 STL 简介 16414.1 STL 概述 .16414.2 常用容器 16414.3 容器适配器 17014.4 常用算法 17114.5 迭代器 .17514.6 示例:合并果子 .175附录 A 思想和技巧 177A.1 时间/空间权衡 177A.2 试验、猜想及归纳 177A.3 模型化 177A.4 随机化* .178A.5 动态化静
15、态 .178A.6 前序和! .179A.7 状态压缩* 180A.8 抽样测试法* 182A.9 离散化* .183A.10 Flood Fill* .184附录 B 调试 .185B.1 常见错误类型 .185B.2 调试过程 .185B.3 调试功能 .185B.4 符号 DEBUG 的应用 .186B.5 代码审查表 .186B.6 故障检查表 .187B.7 命令行和批处理* .188附录 C 竞赛经验和教训 192C.1 赛前两星期 .192C.2 赛前 30 分钟 192C.3 解题表 193C.4 测试数据 .195C.5 交卷前 5 分钟 .196C.6 避免偶然错误 .19
16、6C.7 骗分 197附录 D 学习建议 .198D.1 学习方法 .198D.2 学习能力 .198D.3 关于清北学堂 .198附录 E 竞赛简介 .199E.1 从 NOIP 到 IOI199E.2 NOIP 简介 .199E.3 常用语 201E.4 第一次参加复赛 202附录 F NOIP 复赛知识点分布 204附录 G 资料推荐 .205G.1 书籍 205G.2 网站 205参考文献 206计算机专业是朝阳还是夕阳? .207杜子德在 CCF NOI2012 开幕式上的讲话 209多数奥赛金牌得主为何难成大器 .210第单元 C+语言基础1.1 程序结构(1) 程序框架 注释:注
17、释有两种,一种是“/”,另一种是“/* */”。“/ ”必须单独放置一行,或代码所在行的后面;而“/*”、“*/ ”成对存在,可以插入到代码的任意位置。 引用头文件:在代码开头写“#include ”。如果想引用自己的头文件,需要把尖括号(表示只从系统目录搜索头文件)换成双引号(表示先从 cpp 所在文件夹搜索,然后再到系统文件夹搜索)。 命名空间:很多 C+的东西都要引用 std 命名空间,所以代码中会有“using namespace std;”。 main():所有程序都要从 main()开始。在所有的算法竞赛中,main()的返回值必须是 0,否则视为程序异常结束,得分为 0 分。 语
18、句和语句块:1. 语句:一般情况下,一条语句要用一个分号“;”结束。为了美观和可读性,可以把一条语句扩展成几行,也可以把多个语句写到同一行上。2. 语句块:用“”和“”包围的代码是语句块。无论里面有多少代码,在原则上,语句块所在的整体都视为一条语句。(2) 选择结构1. if 语句: if 表示判断。如果条件为真,就执行接在 if 后的语句(语句块),否则执行 else 后的语句(语句块)。如果没有 else,就直接跳过。if 有以下几种格式:if (条件) / 如果条件成立,就执行 if后面的语句或语句块。语句或语句块if (条件) / 如果条件成立,就执行 if后面的 A,否则执行B。语句
19、或语句块Aelse语句或语句块Bif (条件1) / 实际上,这是 if语句内的if语句,即if的嵌套。所以else和if中间要有空格。语句或语句块Aelse if (条件 2)语句或语句块Belse语句或语句块N2. switch 语句:switch 表示选择。它根据条件的不同取值来执行不同的语句。格式如下:switch (表达式 )case 值 1:代码段Abreak;case 值 2:代码段Bbreak;default:代码段Nbreak;如果表达式的值是值 1,就执行代码段 A;如果表达式的值是值 2,就执行代码段 B否则执行代码段N。注意: default 一部分可以省略。 如果不使
20、用 break,那么紧随其后的 case 部分代码也会被执行,直到遇到 break 或 switch 语句结束为止! switch 结尾要有一个分号。3. if、switch 都可以嵌套使用。【问题描述】输入一个日期,判断它所在年份是否为闰年,并输出所在月份的天数。闰年的判断方法:四年一闰,百年不闰,四百年又闰。int year,month,day;bool b=false;cinyearmonthday;/ 判断是否为闰年if (n%400=0)b=true;else if (n%100!=0 if (b)cout ”,还是“=”。 逆序循环时,不要把自减“-”写成自增“+ ”!【问题描述】
21、输入 n,输出 n!(n! 123 4n)。结果保证小于 long long 的范围。当输入值为负数时结束程序。int n;long long r=1;cinn;while (n-1)r=1;for (int i=1; in;(4) goto 语句goto 语句用于无条件跳转。要想跳转,首先要定义标签(在代码开头的一段标识符,后面紧跟冒号),然后才能 goto 那个标签。很多教程不提倡使用无条件跳转,因为它破坏了程序结构,还容易给代码阅读者带来麻烦。不过,这不代表 goto 没有使用价值。 goto 的一个用途是跳出多层循环:for (int i=0; i应该改成。 C 程序运行速度稍优于 C
22、+。不过也没快多少。总之,C 能做的一切事情, C+也能做;C+能做的一切事情,C 不一定能做。1.2 数据类型(1) 基本数据类型名称 占用空间 别名 数据范围int 4 signed, signed int,long, long int2,147,483,6482,147,483,647unsigned int1 4 unsigned, unsignedlong,unsigned long int04,294,967,295char 1 char 128127unsigned char 1 unsigned char 0255short2 2 short int,signed short
23、int32,76832,767unsigned short 2 unsigned short int 065,535long long3 8 signed long long 9,223,372,036,854,775,8089,223,372,036,854,775,8074unsigned long long8 018,446,744,073,709,551,615bool 1 true 或 falsechar 1 128127signed char 1 128127unsigned char 1 0255float 4 3.4E +/- 38 (7 位有效数字 )double 8 lon
24、g double 1.7E +/- 308 (15 位有效数字 )1 一般都使用有符号整数,除非范围不够大,或者你确定你的减法运算不会出现“小数减大数”。2 一般来说,使用 int、long long 更保险一些,除非内存不够用。3 不要使用 “_int64”!它是 Visual C+特有的关键字。4 假如 a 是 long long 类型,把超过 231的值赋给它时要使用字面值 LL(ULL ):a=123456789012345LL。(2) 变量与常量1. 定义变量:“变量类型 标识符”,如“int i;”定义了一个名字为 i 的整型变量。注意,此时 i 并未初始化,所以 i 的值是不确定
25、的。2. 定义常量:“const 变量类型 标识符=初始值”,如:const int N=90;3. 合法的标识符: 标识符不能和关键字(在 IDE 中会变色的词语)相同。 标识符只能包括字母、数字和下划线“_”,并且开头只能是字母或下划线。 标识符必须先定义后使用。 在同一作用域内,标识符不能重复定义(即使在不同作用域内也不应重复,否则容易产生歧义)。 C+区分大小写。所以 A 和 a 是两个不同的标识符。(3) 数组1. 定义一个一维数组:int a10;这个数组一共 10 个元素,下标分别为 09。访问某个元素时,直接用 a 加方括号,如 a5。2. 定义一个二维数组:int b53;这
26、个数组一共 5315 个元素,分别是 b00、b01、b02、b10b42 。访问某个元素时要用两个方括号,如 b21。多维数组的定义和使用方法与此类似。3. 数组名和元素的寻址:以上面的 a、b 为例 数组名是一个指针,指向整个数组第一个元素所在的地址。如 a 就是定义多个指针时,每个字母的前面都要有“*”。注意,如果 p 没有被初始化,它就会指向一个未知的内存空间,而错误地操作内存会导致程序崩溃!4. 指针使用实例:int a = 0, b = 1; int c = 1,2,3,4,5,6,7,8,9,10;int *p; / 定义一个指针p= / 让p指向a(*p)=3; / 相当于a=
27、3(*p)=b; / 相当于a=b,此时a等于1/ p=b; / 非法操作,左边是int *,右边是int ,类型不匹配。p= / 让p指向b,从此 p和a没关系了p=c+6; / 让p指向c6,p和b 又没关系了cout”):p.value、(j = j + (+i); / i 先自增,变成 1,然后再和 j 相加。执行之后 i=1,j=9。k = k + (i+); / i 先和 k 相加,使 k=6。然后 i 再自增。执行之后 i=2,k=6。 前缀运算符返回引用类型,后缀运算符返回数值类型。 为了避免错误,不要让+、- 和其他能够改变变量的操作在同一行出现!7. 赋值运算符: 在 C+
28、中赋值是一种运算符。所以你会看到 i=j=0、 dx=y、return c=i+j 之类的代码。 +=、-= 、 *=、可以简化书写。例如 a*=2+3 相当于 a=a*(2+3)。1 例如计算 “a float c=6.4/(float)i; / 把i的值变成float 类型。两个操作数进行四则运算,如果想保留小数位,那么两个操作数应该都是浮点数。上面的代码就是这样。1.4 函数(1) 定义和使用函数1. 定义和调用函数:下面定义了一个函数,返回值是 double 类型的,其中有两个参数 i、j,分别是 int和 float 类型的。double foo(int j, float j) 如果
29、函数不需要返回任何值,可定义为 void 类型。 函数的定义必须在函数调用的前面。只有在前面添加了函数定义,才能把具体实现放到调用的后面:double foo(int, float); / 放到调用之前2. 返回值:return 值; 函数是 void 类型的,那么 return 后面除了分号,什么都不跟。 调用之后,函数立刻结束。 不可以直接对函数名赋值(学过 Pascal 或 Basic 语言的同学要特别注意)。3. 如果你的函数需要返回指针或引用,你必须注意:不要引用函数内部的变量!因为函数一结束,函数内部的变量就烟消云散,不复存在了。正确做法是引用静态变量(static)或全局变量。4
30、. 内联函数(inline):当一个函数内部只有寥寥几句时,如“华氏度变摄氏度”,可以考虑将其定义成内联函数,通知编译器省略函数入栈等操作,直接展开函数内容,以加快运行速度。inline int FtoC(int f) return (f-32)/9*5; (2) 传递实参1. 按值传递:例如 int foo(int n),在调用 foo 时,程序会把参数复制一份给 n。这样,对 n 的任何修改都不会反映到调用 foo 的参数上面。对于按值传递数组,一定要慎重。因为复制数组的元素要浪费很多时间。2. 传递指针:例如 int foo(int *n)。对 n 的修改会反映到调用 foo 的参数上面
31、。 修改 n 的值时要注意,必须用取值运算符,否则改变的是 n 指向的内存空间 1。 此外,这种方法可以用于传递数组调用时只需把数组名作为参数。这时不需要取值运算符。3. 传递引用:例如 int foo(int fin、fout 分别代表输入文件和输出文件。把它们换成 stdin 和stdout,就是从屏幕输入和从屏幕输出。“1.5 字符串操作 ”也使用了同样的变量。1. 输出字符串或变量的值:printf(“格式字符串“, );或 fprintf(fout, “格式字符串“, );格式字符:“%”后连接一个字母。字符 含义 字符 含义d 整数 1 e, E 用科学记数法表示的浮点数u 无符号
32、整数 f 浮点数o 八进制整数 c 字符x, X 十六进制整数(小写、大写) s 字符串(字符数组)常见的修饰符: %5d:5 位数,右对齐。不足 5 位用空格补齐,超过 5 位按实际位数输出。 %-5d:5 位数,左对齐。不足 5 位用空格补齐,超过 5 位按实际位数输出。 %05d:5 位数,右对齐。不足 5 位用0补齐,超过 5 位按实际位数输出。 %+d:无论是正数还是负数,都要把符号输出。 %.2f:保留 2 位小数。如果小数部分超过 2 位就四舍五入,否则用 0 补全。1. 输入到变量 读取不含空白的内容:scanf(“格式字符串“, 或 fscanf(fin, “格式字符串“,
33、格式字符和 printf 基本一致。 不要忘记“ 首先要判断它是否为 EOF(文件结束)。如果不是,就可以用强制类型转换变成 char。读取到行末时,要注意对换行符的处理。 Windows、Linux、Mac 的回车字符是不同的。Linux 是n,Mac 是r ,Windows 下是两个字符r和n 。(2) 使用流输入 /输出头文件:1. 输入到变量:cinn;2. 输出到屏幕上:coutnm; cout 右对齐,长度为 n,不足的部分用空格补齐:cout.width(n);cout.fill( ); / 如果想用“0 ”补齐,就可以把空格换成“0”cout、 、 以及、等。C+的流、容器、算
34、法等都需要引用 std 命名空间。所以需要在#include 后面、你的代码前面加上一句:using namespace std;(1) 库函数1. 数组的整体操作:头文件: 将 a初始化:memset(a, 0, sizeof(a);第二个参数应该传入 0、-1 或 0x7F。传入 0 或-1 时,a 中每个元素的值都是 0 或-1 ;如果传入0x7F 时,那么 a中每个元素的值都是 0x7F7F7F7F(不是 0x7F!),可认为是“无穷大”。 将 a整体复制到 b中:memcpy(b, a, sizeof(a); 判断 a和 b是否等价:memcmp(a, b, sizeof(a); /
35、 返回 0 表示等价2. 字符操作:头文件: tolower(c)、toupper(c):将 c 转化为小写或大写。 isdight(c)、isalpha(c)、isupper(c)、islower(c)、isgraph(c) 、isalnum(c):分别判断 c 是否为十进制数字、英文字母、大写英文字母、小写英文字母、非空格、字母或数字。3. 最大值/最小值 :头文件:max(a,b)返回 a 和 b 中的最小值, min(a,b)返回 a 和 b 中的最大值。其实我们可以自己写:4. 交换变量的值:swap(a,b)头文件:其实我们可以自己写:inline void swap(int a=
36、b; b=t; 5. 执行 DOS 命令或其他程序:system(“命令行“); 头文件: 暂停屏幕:system(“pause“); 竞赛交卷或 OJ 提交代码之前必须删除 system,否则会被视为作弊( 如果是 tyvj 甚至都无法提交)。 如果使用输入重定向,那么命令提示符不会接受任何键盘输入直接用文件内容代替键盘了。6. 立刻退出程序:exit(0);这种方法常用于深度优先搜索。执行后,程序立刻停止并返回 0,所以在调用前应该输出计算结果。头文件:7. 计时:double a = (double)clock() / (double)CLOCKS_PER_SEC;上面的 a 对应一个时
37、刻。而将两个时刻相减,就是时间间隔。可用这种方法卡时。头文件:8. 断言:assert( 条件) 条件为假时,程序立刻崩溃。 头文件: 如果定义了 NDEBUG 符号,那么它将不会起任何作用。 断言和错误处理不同:例如出现“人数为负数”的情况,如果责任在于用户,那么应该提示错误并重新输入,而不是用断言;如果发生在计算过程,应该用断言来使程序崩溃,以帮助改正代码中的错误。换句话说,错误处理防的是用户的错误,断言防的是代码的错误。9. 快速排序:qsort(首项的指针, 待排序元素个数, 每个元素所占字节, 比较函数) 头文件: 这是留给 C 党的快速排序,它比 STL 的排序算法啰嗦一些。 比较
38、函数返回 int 类型,用于对两个元素的比较。原型如下:int compare(const void *i, const void *j);如果*i*j ,则应返回一个小于 0 的数;如果*i=*j 则应返回 0,否则返回一个大于 0 的数。10.随机数发生器: 头文件: 产生随机数: 032767 的随机数: rand() 粗略地控制范围:rand()%范围注意,这种方法产生的随机数的分布是不均匀的。 精确地控制范围:(double)rand()/RAND_MAX*范围 控制在a, b)之间:a + (int) (double)rand()/RAND_MAX*(b-a) 初始化随机数种子:
39、srand(数字 ):初始化随机数种子。 注意,这条语句在程序开头使用,并且最多用一次。同一程序、同一平台,srand 中的参数相等,用 rand()产生的随机数序列相同。 使随机数更加随机:引用,然后这样初始化随机数种子,srand(time(NULL)。不要在循环中使用这条语句(例如批量产生随机数据),因为 time 只精确到秒。11.数学函数: 头文件: abs(x):求 x 的绝对值(该函数同时包含于)。 sin、cos、tan、asin、acos 、atan:三角函数,角的单位为弧度。可用 atan(1)*4 表示 。 sinh、cosh、tanh、asinh、acosh 、atan
40、h:双曲函数 sqrt:求平方根 ceil(x)、floor(x):分别返回大于等于 x 的最小整数、小于等于 x 的最大整数。注意,参数和返回值都是浮点数类型。 exp(x)、log(x)、log10:分别求 ex、ln x、lgx(顺便提一句,指数可以把加法问题转化为乘法问题,对数可以把乘法问题转化为加法问题。) pow(a,b):计算 ab。由于精度问题,你仍然需要学会快速幂。 fmod(a,b):计算 a 除以 b 的余数。当然,这是浮点数的版本。(2) 宏定义宏定义是 C 语言的产物。在 C+中,它真的 out 了。1. 第一种用法配合条件编译:#define DEBUG定义一个叫
41、DEBUG 的标识符。它应该与#ifdef 或#ifndef 配合使用。举例如下:#define DEBUG#ifdef DEBUGvoid print(int v) cout 。printf 和 scanf 在中,cin 和 cout 在头文件中且位于std 命名空间内。下面假设待处理的字符串为 str 和 str2,即:char strMAX, str2MAX;牢记,字符串的最后一个字符一定是0。如果字符串内没有 0,进行以下操作(输入除外)时可能会造成意外事故。1. 输出字符串 str: coutstr;以上两种方法在输入时会忽略空格、回车、TAB 等字符,并且在一个或多个非空格字符后面
42、输入空格时,会终止输入。 fgets(str, MAX, fin); 每调用一次,就会读取一行的内容(即不断读取,直到遇到回车停止)。3. 求字符串 str 的长度:strlen(str) / 这个长度不包括末尾的0。4. 把字符串 str2 连接到字符串 str 的末尾:strcat(str, str2) str 的空间必须足够大,能够容纳连接之后的结果。 连接的结果直接保存到 str 里。函数返回值为格式化字符串:sprintf(str, “%d“, i);它们和 fscanf、fprintf 非常像,用法也类似。可以通过这两个函数进行数值与字符串之间的转换。1.8 文件操作!正式竞赛时,
43、数据都从扩展名为“in”的文件读入,并且需要你把结果输出到扩展名为“out”的文件中;在 OJ( Online Judge,在线测评器)中则不需要文件操作。具体情况要仔细查看题目说明,以免发生悲剧。(1) 输入/输出重定向头文件:或 方法:只需在操作文件之前添加以下两行代码。freopen(“XXXXX.in“,“r“,stdin);freopen(“XXXXX.out“,“w“,stdout); 调用两次 freopen 后,scanf、printf、cin、cout 的用法完全不变,只是操作对象由屏幕变成了指定的文件。 使用输入重定向之后,“命令提示符”窗口将不再接受任何键盘输入(调用 s
44、ystem 时也是如此),直到程序退出。这时不能再用 system(“pause“)暂停屏幕。(2) 文件流头文件:流的速度比较慢,在输入/输出大量数据的时候,要使用其他文件操作方法。 方法:定义两个全局变量。ifstream fin(“XXXXX.in“);ofstream fout(“XXXXX.out“); fin、fout 和 cin、cout 一样,也用“”运算符输入/输出,如:fina; fout 或 方法:定义两个指针。FILE *fin, *fout;int main()fin = fopen(“XXXXX.in“, “r“);fout = fopen(“XXXXX.out“,
45、 “w“);fclose(fin); fclose(fout); / 在某些情况下,忘记关闭文件会被认为是没有产生文件。return 0; 进行输入/输出操作时,要注意函数名的前面有 f,即 fprintf、fscanf、fgets ,并且这些函数的第一个参数不是格式字符串,而是 fin 或 fout,如 fprintf(fout, “%d“, ans)。 想改成从屏幕上输入/输出时,不用对代码动手术,只需把含 fopen 和 fclose 的代码注释掉,并改成:fin=stdin; fout=stdout;1.9 简单的算法分析和优化(1) 复杂度为了描述一个算法的优劣,我们引入算法时间复杂
46、度和空间复杂度的概念。时间复杂度:一个算法主要运算的次数,用大 O 表示。通常表示时间复杂度时,我们只保留数量级最大的项,并忽略该项的系数。例如某算法,赋值做了 3n3+n2+8 次,则认为它的时间复杂度为 O(n3);另一算法的主要运算是比较,做了 42n+2n4+700 次,则认为它的时间复杂度为 O(2n)。当然,如果有多个字母对算法的运行时间产生很大影响,就把它们都写进表达式。如对 mn 的数组遍历的时间复杂度可以写作 O(mn)。空间复杂度:一个算法主要占用的内存空间,也用大 O 表示。在实际应用时,空间的占用是需要特别注意的问题。太大的数组经常是开不出来的,即使开出来了,遍历的时间
47、消耗也是惊人的。(2) 常用算法的时空复杂度1s 运算次数约为 5,000,0001。也就是说,如果把 n 代入复杂度的表达式,得数接近或大于5,000,000,那么会有超时的危险。常见的数量级大小:O(1) O(logn)O(n)O(nlogn)O( n2)O(n 3)O(2 n)O(n!)数量级 能承受的大致规模 常见算法O(1) 任意 直接输出结果O(logn) 任意 二分查找、快速幂O(n) 以百万计(五六百万) 贪心算法、扫描和遍历O(nlogn) 以十万计(三四十万) 带有分治思想的算法,如二分法O(n2) 以千计数(两千) 枚举、动态规划O(n3) 不到两百 动态规划O(2n)
48、24 搜索O(n!) 10 产生全排列O(nn) 8 暴力法破解密码O(1)叫常数时间;O(n)、O(n 2)、O(n 3)、O(n 4)叫做多项式时间;O(2 n)、O(3 n)叫做指数时间。1 不同资料给出的数值是不同的,不过这不要紧。在 NOIP 中,只要你的算法正确,就不会在运行时间上 “打擦边球”。(3) 简单的优化方法1. 时间的简单优化方法时间上的优化在于少做运算、做耗时短的运算等。有几个规律需要注意: 整型运算耗时远低于实型运算耗时。 位运算速度极快。 逻辑运算比四则运算快。 +、 -、*运算都比较快(-、* 比+慢一点点,可以忽略不计)。 /运算比 +、-、*慢得多 (甚至慢几十倍)。 取余%和除法运算速度相当。 调用函数要比直接计算慢(因为要入栈和出栈)。这些规律我们可以从宏观上把握。事实上,究竟做了几步运算、几次赋值、变量在内存还是缓存等多数由编译器、系统决定。但是,少做运算(尤其在循环体、递归体中)一定能很大程度节省时间。2. 空间的简单优化方法空间上的优化主要在于减小数组大小、降低数组维数等。常用的节省内存的方法有:压缩储存见 180 页“A.7 状态压缩*”。覆盖旧数据