商家名称 | 信用等级 | 购买信息 | 订购本书 |
算法艺术与信息学竞赛.算法竞赛入门经典(刘汝佳著) | |||
算法艺术与信息学竞赛.算法竞赛入门经典(刘汝佳著) |
《算法竞赛入门经典》分为三部分:语言篇、算法篇和竞赛篇。首先以实践导向的方式讲解了C/C++的基本语法,然后介绍了算法和数据结构的基础知识,最后是动态规划、数学和图论三大专题。全书短小精悍,但内容全面,既可作为教材,又方便自学。
算法在计算机科学乃至于整个科学界的作用日益明显。它们不仅具有重要的理论意义,而且解决了生产生活中的很多实际问题。程序设计竞赛就是这样一类以算法为核心但是偏重实用性的比赛。随着各类比赛规模的逐渐扩大,程序设计竞赛在各高校、IT公司和其他社会各界中越来越受到认可和重视。很多研究工作者和从事IT行业的人尽管不参加这类竞赛,但也希望具有这方面的能力,受到这方面的专业训练。
本丛书的前身是5年前的同名图书《算法艺术与信息学竞赛》。5年来,更多的人加入到参赛、命题和组织的队伍中来,各类竞赛的参赛和命题水平也有了长足的进步。作者深知当年的经典之作开始显得题目陈旧,知识的广度和深度也无法达到当今高水平比赛的要求了。因此,将原书的内容扩充、完善后分成三本,以丛书的形式依次展现给读者。这三本书循序渐进,从零语言基础开始讲起,直到超越竞赛本身,真正把算法当成“艺术”。
适合语言零基础的初学者
涵盖算法竞赛的主要知识点
大量经验教训与比赛技巧
简洁、清晰、高效的示例代码
丰富的辅助教学资源与配套习题
刘汝佳,1982年12月生,高中毕业于重庆市外国语学校。
2000年3月获得NO12000全国青少年信息学奥林匹克竞赛一等奖第四名,进入国家集训队,并因此保送到清华大学计算机科学与技术系。大一时获2001年ACM/ICPC国际大学生程序设计竞赛亚洲一上海赛区冠军和2002年世界总决赛银牌(世界第四),2005年获学士学位,2008年获硕士学位。
学生时代曾为中国计算机学会NOI科学委员会学生委员,担任J012002-2008中国国家队教练,并为NOI系列比赛命题十余道。现为NOI竞赛委员会委员,并在NOI 25周年时获得中国计算机学会颁发的“特别贡献奖”。
2004年至今共为ACM/ICPC亚洲赛区命题二十余道,担任6次裁判和2次命题总监,并应邀参加IOI和ACM/lCPC相关国际研讨会,发表论文两篇。
2004年初作为第一作者出版专著《算法艺术与信息学竞赛》,2009年出版译著《编程挑战》。
多年来在全国二十余个城市进行中学生竞赛培训工作,为北京、上海、吉隆坡等地的著名高校授课与宣讲,并多次与TopCodet、百度和网易有道等知名企业合作举办比赛,让更多的IT人才获得展示自我的平台。
第1部分 语言篇
第1章 程序设计入门1
1.1 算术表达式1
1.2 变量及其输入3
1.3 顺序结构程序设计6
1.4 分支结构程序设计9
1.5 小结与习题13
1.5.1 数据类型实验13
1.5.2 scanf输入格式实验13
1.5.3 printf语句输出实验13
1.5.4 测测你的实践能力14
1.5.5 小结14
1.5.6 上机练习15
第2章 循环结构程序设计16
2.1 for循环16
2.2 循环结构程序设计19
2.3 文件操作23
2.4 小结与习题27
2.4.1 输出技巧28
2.4.2 浮点数陷阱28
2.4.3 64位整数28
2.4.4 C++中的输入输出29
2.4.5 小结30
2.4.6 上机练习31
第3章 数组和字符串33
3.1 数组33
3.2 字符数组37
3.3 最长回文子串41
3.4 小结与习题45
3.4.1 必要的存储量45
3.4.2 用ASCII编码表示字符45
3.4.3 补码表示法46
3.4.4 重新实现库函数47
3.4.5 字符串处理的常见问题47
3.4.6 关于输入输出47
3.4.7 I/O的效率47
3.4.8 小结49
3.4.9 上机练习50
第4章 函数和递归51
4.1 数学函数51
4.1.1 简单函数的编写51
4.1.2 使用结构体的函数52
4.1.3 应用举例53
4.2 地址和指针56
4.2.1 变量交换56
4.2.2 调用栈57
4.2.3 用指针实现变量交换59
4.2.4 初学者易犯的错误61
4.3 递归62
4.3.1 递归定义62
4.3.2 递归函数63
4.3.3 C语言对递归的支持64
4.3.4 段错误与栈溢出66
4.4 本章小结67
4.4.1 小问题集锦67
4.4.2 小结68
第2部分 算法篇
第5章 基础题目选解69
5.1 字符串69
5.1.1 WERTYU69
5.1.2 TeX括号70
5.1.3 周期串71
5.2 高精度运算71
5.2.1 小学生算术72
5.2.2 阶乘的精确值72
5.2.3 高精度运算类bign73
5.2.4 重载bign的常用运算符75
5.3 排序与检索77
5.3.1 6174问题77
5.3.2 字母重排78
5.4 数学基础81
5.4.1 Cantor的数表81
5.4.2 因子和阶乘82
5.4.3 果园里的树84
5.4.4 多少块土地86
5.5 训练参考86
5.5.1 黑盒测试86
5.5.2 在线评测系统87
5.5.3 推荐题目88
第6章 数据结构基础89
6.1 栈和队列89
6.1.1 卡片游戏89
6.1.2 铁轨91
6.2 链表93
6.2.1 初步分析93
6.2.2 链式结构95
6.2.3 对比测试96
6.2.4 随机数发生器98
6.3 二叉树99
6.3.1 小球下落99
6.3.2 层次遍历101
6.3.3 二叉树重建105
6.4 图106
6.4.1 黑白图像107
6.4.2 走迷宫108
6.4.3 拓扑排序110
6.4.4 欧拉回路111
6.5 训练参考112
第7章 暴力求解法114
7.1 简单枚举114
7.1.1 除法114
7.1.2 最大乘积115
7.1.3 分数拆分115
7.1.4 双基回文数116
7.2 枚举排列116
7.2.1 生成1~n的排列116
7.2.2 生成可重集的排列118
7.2.3 解答树118
7.2.4 下一个排列119
7.3 子集生成120
7.3.1 增量构造法120
7.3.2 位向量法121
7.3.3 二进制法122
7.4 回溯法123
7.4.1 八皇后问题123
7.4.2 素数环126
7.4.3 困难的串127
7.4.4 带宽128
7.5 隐式图搜索129
7.5.1 隐式树的遍历129
7.5.2 一般隐式图的遍历130
7.5.3 八数码问题131
7.5.4 结点查找表133
7.6 训练参考136
第8章 高效算法设计138
8.1 算法分析初步138
8.1.1 渐进时间复杂度138
8.1.2 上界分析140
8.1.3 分治法140
8.1.4 正确对待算法分析结果142
8.2 再谈排序与检索143
8.2.1 归并排序143
8.2.2 快速排序145
8.2.3 二分查找145
8.3 递归与分治148
8.3.1 棋盘覆盖问题148
8.3.2 循环日程表问题149
8.3.3 巨人与鬼149
8.3.4 非线性方程求根150
8.3.5 最大值最小化151
8.4 贪心法151
8.4.1 最优装载问题151
8.4.2 部分背包问题152
8.4.3 乘船问题152
8.4.4 选择不相交区间152
8.4.5 区间选点问题153
8.4.6 区间覆盖问题154
8.4.7 Huffman编码154
8.5 训练参考156
第3部分 竞赛篇
第9章 动态规划初步158
9.1 数字三角形158
9.1.1 问题描述与状态定义158
9.1.2 记忆化搜索与递推159
9.2 DAG上的动态规划161
9.2.1 DAG模型161
9.2.2 最长路及其字典序162
9.2.3 固定终点的最长路和最短路163
9.3 0-1背包问题167
9.3.1 多阶段决策问题167
9.3.2 规划方向168
9.3.3 滚动数组169
9.4 递归结构中的动态规划170
9.4.1 表达式上的动态规划170
9.4.2 凸多边形上的动态规划171
9.4.3 树上的动态规划171
9.5 集合上的动态规划172
9.5.1 状态及其转移173
9.5.2 隐含的阶段173
9.6 训练参考174
第10章 数学概念与方法176
10.1 数论初步176
10.1.1 除法表达式176
10.1.2 无平方因子的数178
10.1.3 直线上的点179
10.1.4 同余与模算术180
10.2 排列与组合182
10.2.1 杨辉三角与二项式定理182
10.2.2 数论中的计数问题184
10.2.3 编码与解码186
10.2.4 离散概率初步187
10.3 递推关系188
10.3.1 汉诺塔188
10.3.2 Fibonacci数列189
10.3.3 Catalan数191
10.3.4 危险的组合192
10.3.5 统计n-k特殊集的数目193
10.4 训练参考194
第11章 图论模型与算法196
11.1 再谈树196
11.1.1 无根树转有根树196
11.1.2 表达式树197
11.1.3 最小生成树199
11.1.4 并查集200
11.2 最短路问题201
11.2.1 Dijkstra算法202
11.2.2 稀疏图的邻接表203
11.2.3 使用优先队列的Dijkstra算法204
11.2.4 Bellman-Ford算法205
11.2.5 Floyd算法206
11.3 网络流初步207
11.3.1 最大流问题207
11.3.2 增广路算法208
11.3.3 最小割最大流定理210
11.3.4 最小费用最大流问题211
11.4 进一步学习的参考212
11.4.1 编程语言213
11.4.2 数据结构213
11.4.3 算法设计213
11.4.4 数学214
11.4.5 参赛指南214
11.5 训练参考215
附录A 开发环境与方法216
A.1 命令行216
A.1.1 文件系统216
A.1.2 进程217
A.1.3 程序的执行217
A.1.4 重定向和管道218
A.1.5 常见命令218
A.2 操作系统脚本编程入门219
A.2.1 Windows下的批处理219
A.2.2 Linux下的Bash脚本220
A.2.3 再谈随机数221
A.3 编译器和调试器221
A.3.1 gcc的安装和测试221
A.3.2 常见编译选项222
A.3.3 gdb简介223
A.3.4 gdb的高级功能224
A.4 浅谈IDE225
“听说你最近在写一本关于算法竞赛入门的书?”朋友问我。
“是的。”我微笑道。
“这是怎样的一本书呢?”朋友很好奇。
“C语言、算法和题解。”我回答。
“什么?几样东西混着吗?”朋友很吃惊。
“对。”我笑了,“这是我思考许久后做出的决定。”
大学之前的我
12年前,当我翻开Sam A.Abolrous所著《C语言三日通》的第一页时,我不会想到自己会有机会编写一本讲解C语言的书籍。当时,我真的只花了3天就学完了这本书,并且自信满满:“我学会C语言啦!我要用它写出各种有趣、有用的程序!”但渐渐地,我认识到了:虽然浅显易懂,但书中的内容只是语言入门,离实际应用还有较大差距,就好比小学生学会造句以后还要下很大功夫才能写出像样的作文。
第二本对我影响很大的书是Sun公司的Peter van der Linden(PvdL)所著的《C程序设计奥秘》。作者称该书应该是每一个程序员“在C语言方面的第二本书”,因为“书中绝大部分内容、技巧和技术在其他任何书中都找不到”。原先我只是把自己当成是程序员,但在阅读的过程中,我开始渐渐了解到硬件设计者、编译程序开发者、操作系统编写者和标准制定者是怎么想的。继续的阅读增强了我的领悟:要学好C语言,绝非熟悉语法和语义这么简单。
后来,我自学了数据结构,懂得了编程处理数据的基本原则和方法,然后又学习了8086汇编语言,甚至曾没日没夜地用SoftICE调试《仙剑奇侠传》,并把学到的技巧运用到自己开发的游戏引擎中。再后来,我通过《电脑爱好者》杂志上的一则不起眼的广告了解到了全国信息学奥林匹克联赛(当时称为分区联赛,NOIP是后来的称谓)。“学了这么久编程,要不参加个比赛试试?”想到这里,我拉着学校里另外一个自学编程的同学,找老师带我们参加了1997年的联赛——在这之前,学校并不知道有这个比赛。凭借自己的数学功底和对计算机的认识,我在初赛(笔试)中获得全市第二的成绩,进入了复赛(上机)。可我的上机编程比赛的结果是“惨烈”的:第一题有一个测试点超过了整数的表示范围;第二题看漏了一个条件,一分都没得;第三题使用了穷举法,全部超时。考完之后我原以为能得满分的,结果却只得了100分中的20多分,名落孙山。
痛定思痛,我开始反思这个比赛。一个偶然的机会,我拿到了一本联赛培训教材。书上说,比赛的核心是算法(Algorithm),并且推荐使用Pascal语言,因为它适合描述算法。我从别人那里复制来了Turbo Pascal 7.0(那时网络并不发达),开始研究起来。由于先学的是C语言,所以我刚开始学习Pascal时感到有些不习惯:赋值不是“=”而是“:=”,简洁的花括号变成了累赘的begin和end,if之后要加个then,而且和else之间不允许写分号……但很快我就发现,这些都不是本质问题。在编写竞赛题的程序时,我并不会用到太多的高级语法。Pascal的语法虽然稍微啰嗦一点,但总体来说是很清晰的。就这样,我只花了不到一天时间就把语法习惯从C转到了Pascal,剩下的知识就是在不断编程中慢慢地学习和熟练——学习C语言的过程是痛苦的,但收益也是巨大的,“轻松转到Pascal”只是其中一个小小的例子。
我学习计算机,从一开始就不是为了参加竞赛,因此,在编写算法程序之余,我几乎总是使用熟悉的C语言,有时还会用点汇编,并没有觉得有何不妥。随着编写应用程序的经验逐渐丰富,我开始庆幸自己先学的是C语言——在我购买的各类技术书籍中,几乎全部使用的是C语言而不是Pascal语言,尽管偶尔有用Delphi的文章,但这种语言似乎除了构建漂亮的界面比较方便之外,并没有太多的“技术含量”。我始终保持着对C语言的熟悉,而事实证明这对我的职业生涯发挥了巨大的作用。
插图:
本章介绍一些常见的图论模型和算法,包括最小生成树、单源最短路、每对结点的最短路、最大流、最小费用最大流等。限于篇幅,很多算法都没有给出完整的正确性证明(很容易在其他参考资料中找到相关内容),但给出了简单、易懂的完整代码,方便读者参考。
在第6章中,我们第一次接触到二叉树;后来,又接触到了其他树状结构,如解答树、BFS树。本节将继续讨论“树”这一话题。
有,?个顶点的树具有以下3个特点:连通、不含圈、恰好包含n-1条边。有意思的是,具备上述3个特点中的任意两个,就可以推导出第3个,有兴趣的读者不妨试着证明一下。
1 1.1.1无根树转有根树
输入一个,z个结点的无根树的各条边,并指定一个根结点,要求把该树转化为有根树,输出各个结点的父亲编号。
相关阅读:
更多图书资讯可访问读书人图书频道:http://www.reAder8.cn/book/