《从C语言转C++简明教程_v5.1.docx》由会员分享,可在线阅读,更多相关《从C语言转C++简明教程_v5.1.docx(16页珍藏版)》请在taowenge.com淘文阁网|工程机械CAD图纸|机械工程制图|CAD装配图下载|SolidWorks_CaTia_CAD_UG_PROE_设计图分享下载上搜索。
1、从放弃C语言到使用C+刷算法的简明教程by柳姥目录目录使用C+刷算法的好处名称空间using namespace std的解释cin和cout输入输出关于C+的头文件C+的变量声明C+特有的bool变量C+特有的用const定义常量C+里面超好用的string类C+的结构体struct和C语言的结构体的区别C+的引用&和传值的区别C+STL之动态数组vector (矢量)的使用C+ STL之集合set的使用C+ STL之映射map的使用C+ STL之栈stack的使用C+ STL之队列queue的使用C+ STL 之 unordered_m 叩和 unordered_set 的使用C+的位运算
2、bitsetC+中的sort函数C+中使用sort自定义cmp函数关于cctype头文件里的一些函数关于C+11的解释C+11里面很好用的auto声明C+”特性中基于范围的for循环C+11 特性中的 to_stringC+11 特性中的stoi、stod如何在Dev-Cpp中使用C+U中的函数使用C+刷算法的好处 在已经学习过C语言的前提下,学习C+并使用它刷算法的学习成本非常低只需要几个小时就可 以学会 C+向下兼容C, C语言里面的语法完全可以在C+文件中运行,所以学习C+对刷算法时编程语言 的表达能力进行扩充有益无害,例如C语言的输入输出(scanf和printf )比C+快,那么就可
3、以 在使用C+刷算法同时使用scanf和printf提高代码运行效率 C+拥有丰富的STL标准模版库,这也是PAT甲级、LeetCode等题目中经常需要用到的,单纯使用C 语言解决问题会比C+的STL解决该问题麻烦很多 C+的string超级好用比C语言里面的char数组好用多啦用了就再也不想回去的那种 C+可以在某一变量使用前随时定义该变量,非常方便在解决一些较为简单的PAT乙级题目的时候(例如一些时间复杂度限制不严格的题 目),cin s cout输入输出非常方便用过的都说好(g).g)虽然C+是一门面向对象语言,但是对于刷算法这件事而言,我们并不需要掌握它面向对象的部分只 需要掌握刷算法
4、的时候需要用到的部分(基本输入输出、STL标准模板库、string字符串等)就可以 啦C语言和C+有很多相似之处,且C+向下兼容C语言,所以我没有说的地方就直接用C语言的语法 表示就好以下是正文,先来段代码方便讲解:1 #include using namespace std; int main() int n;cin n;cout hello, liuchuo1 n + 1 n;std:cout hello, liuchuo n + 1 endl;我觉得这样比较丑,所以不管要不要用到,直接每道题的代码标配得写using namespace std;就好啦 cin和cout输入输出就如同sca
5、nf和printf在stdio.h头文件中一样,cin和cout在头文件iostream里面,看名字就知 道,io是输入输出input和output的首字母,stream是流,所以这个iostream头文件里包含的方 法就是管理一些输入输出流的cin和cout比较方便,不用像C语言里的scanf、printf那样写得那样繁琐,cinn;和 scanf(%d, &n);一样的意思(而且用cin再也不用担心像scant 一样忘记写取地址符& 了耶),注意cin是向右的箭头,表示将内容输入到n中同样,cout n;和printf(%d, n);一样的意思,此时cout是向左的两个箭头,注意和cin区
6、分开来而且不管n是double还是int或者是char类型,只用写cin n;和cout n;这样简单的语句就 好,不用像C语言中需要根据n的类型对应地写%lf、d、c这样麻烦endl和n是一个意思,endl的全称是end of line,表示一行输出结束,然后输出下一行一般如 果前面是个字符串引号的话直接“n”比较方便,如果是变量之类的我觉得写endl会比较好看cout hello,小可爱n;cout n endl;cin和cout虽然使用起来更方便,但是输入输出的效率不如scanf和printf快,所以如果是做PAT乙 级里面那种简单、对时间复杂度要求不高的题目,直接用cin和cout会觉
7、得写起来比较省事儿;如 果题目对时间复杂度要求比较高,全都改成scanf和printf可以提高代码的输入输出效率,比如有的 时候发现用cin、cout做题目超时了,改成scanf和printf就AC了关于C+的头文件C+的头文件一般是没有像C语言的.h这样的扩展后缀的,一般情况下C语言里面的头文件去掉.h然 后在前面加个c就可以继续在C+文件中使用C语言头文件中的函数啦比如:#include / 相当于C语言里面的#include #include / 相当于C语言里面的#include #include / 相当于C语言里面的#include 4 #include / 相当于C语言里面的#i
8、nclude C+的变量声明C语言的变量声明一般都在函数的开头,但是C+在首次使用变量之前声明即可(当然也可以都放在 函数的开头),而且一般C语言里面会在for循环的外面定义i变量,但是C+里面可以在for循环内 部定义(关于这点,VC+6.0里面可能会发现代码复制进去编译不通过,这是因为这个编译器太老 啦,建议不要用这么上古的编译器啦)而且在for循环里面定义的局部变量,在循环外面就失效啦 (就是脱离这个局部作用域就会查无此变量的意思),所以一个main函数里面可以定义好多次局部 变量i ,再也不用担心写的循环太多变量名i、j、k不够用啦#include using namespace st
9、d; int main() cin n;cout hello, liuchuo m;for (int i = 0; i n; i+) /这个i只在for循环里面有用,出了这个for循环就相当于 不见了cout i;11for (int i = 0; i m; i+) /又可以定义一个i啦,和上面那个i不会冲突13 cout i + 2;14 15 return 0;16 C+特有的bool变量bool变量有两个值,false和true ,以前用C语言的时候都是用int的0和1表示false和true 的,现在C+里面引入了这个叫做bool (布尔)的变量,而且C+把所有非零值解释为true ,
10、零值解 释为false所以直接赋值一个数字给bool变量也是可以的它会自动根据int值是不是零来决定 给bool变量赋值true还是falsebool flag = true;bool flag2 = -2; / flag2为truebool flag3 = 0; / flag3为falseC+特有的用const定义常量之前C语言里面会用#define定义常量,但是C+里面用const这个限定符定义常量,这样做有个好处 就是可以定义常量的类型,比如int类型的常量a这样定义:const int a = 99999999;C+里面超好用的string类以前用char的方式处理字符串很繁琐,现在有
11、了 string类,定义、拼接、输出、处理都更加简单啦 不过string只能用cin和cout处理,无法用scanf和printf处理:string s = hello world; / 赋值字符串string s2 = s;string s3 = s + s2; /字符串拼接直接用十号就可以string s4;cin s4; /读入字符串 cout s; 输出字符串用cin读入字符串的时候,是以空格为分隔符的,如果想要读入一整行的字符串,就需要用getlines的长度可以用s.length()获取(有几个字符就是长度多少,不存在Chart里面的什么末尾的结束 符之类的)string s; /
12、定义一个空字符串sgetline(cin, s); /读取一行的字符串,包括空格cout s.length(); /输出字符串s的长度string中还有个很常用的函数叫做substr ,作用是截取某个字符串中的子串,用法有两种形式:string s2 = s.substr(4); /表示从下标4开始一直到结束string s3 = s.substr(5( 3); /表示从下标5开始,3个字符C+的结构体struct和C语言的结构体的区别定义好结构体stu之后,使用这个结构体类型的时候,C语言需要写关键字struct ,而C+里面可以省 略不写:1 struct stu int grade; f
13、loat score;4;struct stu arrl 10; / C语言里面需要写成struct stustu arr210;/ C+里面不用写struct,直接写stu就好了C+的引用&和传值的区别这个引用符号&要和C语言里面的取地址运算符&区分开来,他们没有什么关系,C+里面的引用是 指在变量名之前加一个&符号,比如在函数传入的参数中int&a ,那么对这个引用变量a做的所有 操作都是直接对传入的原变量进行的操作,并没有像原来int a 一样只是拷贝一个副本(传值),举 两个例子:void func(int &a) /传入的是n的引用,相当于直接对n进行了操作,只不过在func函数中换
14、了个名 字叫a2 a = 99;3 4 int main() 5 int n = 0;func(n); / n由0变成了997 void func(int a) /传入的是0这个值,并不会改变main函数中n的值 a = 99;8 9 int main() int n = 0; func(n);/并不会改变n的值,n还是010 C+STL之动态数组vector (矢量)的使用之前C语言里面用intarr定义数组,它的缺点是数组的长度不能随心所欲的改变,而C+里面有一个 能完全替代数组的动态数组vector (有的书里面把它翻译成矢量,vector本身就是矢量、向量的意 思,但是叫做动态数组或者
15、不定长数组我觉得更好理解,绝大多数中文文档中一般不翻译直接叫它 vector-),它能够在运行阶段设置数组的长度、在末尾增加新的数据、在中间插入新的值、长度任 意被改变,很好用它在头文件vector里面,也在命名空间std里面,所以使用的时候要引入头文 件 include 和 using namespace std;vector x stack、queue、m叩、set这些在C+中都叫做容器,这些容器的大小都可以用.size() 获取到,就像string s的长度用s.lengthQ获取一样 ( string其实也可以用s.size(),不过对于 vector x stack、queue、m叩
16、、set这样的容器我们一般讨论它的大小size ,字符串一般讨论 它的长度length其实string里面的size和length两者是没有区别、可以互换使用的,比如我之前 写过一篇博客C+: string类中size。和length。的区别,最终的结论就是两者没有区别,里面根据官 方文档进行了详细阐述,有兴趣的可以去看一下:1 #include 2 #include 3 int main() vector vl; /定义一个vector v1,定义的时候没有分配大小 cout vl .size(); /输出vector v1的大小,此处应该为06 return 0;7 vector可以一开始
17、不定义大小,之后用resize方法分配大小,也可以一开始就定义大小,之后还可以 对它插入删除动态改变它的大小而且不管在main函数里还是在全局中定义,它都能够直接将所有 的值初始化为0 (不用显式地写出来,默认就是所有的元素为0),再也不用担心C语言里面出现的那 种intarr10;结果忘记初始化为0导致的各种加8啦vector v(10); /直接定义长度为10的int数组,默认这10个元素值都为023/或者vector v1;v1 . resize(8); 先定义一个vector变量vl,然后将长度resize为8,默认这8个元素都是067/在定义的时候就可以对vector变量进行初始化v
18、ector v3(100r 9);/把100长度的数组中所有的值都初始化为9910/访问的时候像数组一样直接用门下标访问即可(也可以用迭代器访问,下面会讲)11v =2;cout v0;不管是vector、stack、queue、map还是set都有很多好用的方法,这些方法都可以在皿电区囱 生曲叵an官方网站中直接查询官方文档,上面有方法的讲解和代码示例官方文档是刷题时候必不 可少的好伙伴(如果你用的是Mac OS系统,下载软件Dash然后在里面下载好C+,平时查文档会 更方便,我平时做开发写算法都在Dash里面查文档,内容和官方文档是一样的)PS:经此教程读 者KeilGlay提醒,Wind
19、ows下有受Dash启发而开发的离线文档浏览器Zeal ( https:/zealdocs.org/),和Dash的功能一样,使用Windows的小伙伴可以下载Zeal看 离线官方文档比如进入官网搜索vector ,就会出现vector拥有的所有方法,点进去一个方法就能看到这个方法的 详细解释和代码示例当然我们平时写算法用不到那么多方法啦,只有几个是常用的以下是一些常 用的vector方法:1 #include 2 #include using namespace std;4 int main() vector a; /定义的时候不指定vector的大小 cout a.size() endl;
20、 / 这个时候size是07 for (int i = 0; i 10; i+) a .push_back(i); /在vector a的末尾添加一个元素i9cout a.size() endl; / 此时会发现a的size变成了 10vector b(15); /定义的时候指定vector的大小,默认b里面元素都是0 cout b.size() endl;for) for1314151617181920212223242526272829(int i = 0; i b.size(); i+) bi = 15;(int i = 0; i b.size(); i+) cout bi ;)cout
21、 endl;vector c(20, 2); /定义的时候指定vector的大小并把所有的元素赋一个指定的值 for (int i = 0; i c.size(); i+) cout ci ;cout endl;for (auto it = c. begin。; it ! = c .end(); it+) / 使用迭代器的方式访问 vector cout *it ;return 0;容器vector . set、map这些遍历的时候都是使用迭代器访问的,c.beginQ是一个指针,指向容器 的第一个元素,c.end()指向容器的最后一个元素的后一个位置,所以迭代器指针it的for循环判断条 件
22、是 it != c.end()我再重复一遍c.end()指向容器的最后一个元素的后一个位置,这是一个重点和难点,我画个图 加深一下小可爱们的记忆(再biu你们一下,biubiubiu这下总该记得了吧)vector七核今)2元索八 2|4|5| ” 7| 幺| 夕 | 同 斗 V. bein(.)V.endi)访问元素的值要对it指针取值,要在前面加星号所以是cout-it;这里的auto相当于vector:iterator的简写,关于auto下文有讲解C+ STL之集合set的使用set是集合,一个set里面的各元素是各不相同的,而且set会按照元素进行从小到大排序以下 是set的常用用法:1
23、 #include 2 #include using namespace std;4 int main() set s; /定义一个空集合ss.insert(1); /向集合s里面插入一个1cout *(s.begin() endl; /输出集合s的第一个元素(前面的星号表示要对指针取值)for (int i = 0; i 6; i+) s.insert(i); /向集合s里面插入i 10)for (auto it = s.begin(); it != s.end(); it+) / 用迭代器遍历集合s里面的每一 个元素cout *it 13)cout endl (s.find(2) ! =
24、s.end() endl; / 查找集合s中的值,如果结果等于 s. end ()表示未找到(因为s. end ()表示s的最后一个元素的下一个元素所在的位置)cout (s.find(10) I = s.end() endl; / s.find(10) ! = s.end()表示能找到 10 这个元素s.erase(l); /删除集合s中的1这个元素cout (s.find(1) != s.end() endl; /这时候元素1就应该找不到啦18 return 0;19C+ STL之映射m叩的使用map是键值对,比如一个人名对应一个学号,就可以定义一个字符串string类型的人名为“键”,学
25、 号int类型为“值,如map m;当然键、值也可以是其它变量类型map会自动将所有的 键值对按照键从小到大排序,map使用时的头文件include 以下是m叩中常用的方法:#include 234567891011121314151617181920212223#include #include using namespace std;int main() map m; /定义一个空的map m,键是string类型的,值是int类型的 mhello = 2; / cout mhello将key为hell。,value为2的键值对(key-value)存入map中 endl; /访问map中
26、key为hello的value,如果key不存在,则返回。 endl;将 w。rid 1键对应的值修改为3cout second获取 for (auto it = m.begin(); it != m.end(); it+) cout first second endl;/访问map的第一个元素,输出它的键和值cout first second endl;/访问map的最后一个元素,输出它的键和值cout first second endl;/输出map的元素个数cout m.size() endl;return 0;C+ STL之栈stack的使用栈stack在头文件#include中,是数
27、据结构里面的栈以下是常用用法:1 #include 2 #include 3 using namespace std;4 int main() stack s; / 定义一个空栈s for (int i = 0; i 6; i+) 5 .push(i); /将元素i压入栈s中8cout s.topO endl; / 访问s的栈顶元素 cout s.size() endl; / 输出s的元素个数 s.pop(); /移除栈顶元素12return 0;13C+ STL之队歹IJqueue的使用队列queue在头文件ttindude中,是数据结构里面的队列以下是常用用法:1 #include 2 #
28、include using namespace std;4 int main() queue q; /定义一个空队列q for (int i = 0; i 6; i+) 7q.push(i); /将i的值依次压入队列q中8cout q.frontO q.back() endl; /访问队列的队首元素和队尾元素cout q.size() endl; /输出队列的元素个数 q.pop(); /移除队列的队首元素12 return 0;13C+ STL 之 unordered_map 和 unordered_set 的使用unordered_map 在头文件 include 中, unordered
29、_set 在头文件 #include 中unordered_m叩和m叩(或者unordered_set和set )的区别是,m叩会按照键值对的键key进行 排序(set里面会按照集合中的元素大小进行排序,从小到大顺序),而unordered_map (或 者unordered.set )省去了这个排序的过程,如果偶尔刷题时候用m叩或者set超时了,可以考虑 用unordered.map (或者unordered.set )缩短代码运行时间、提高代码效率至于用法和m叩、set是一样的C+的位运算bitsetbitset用来处理二进制位非常方便。头文件是#include , bitset可能在PA
30、T、蓝桥0J中不常 用,但是在LeetCodeOJ中经常用到而且知道bitset能够简化一些操作,可能一些复杂的问题能够直接用bitset就很轻易地解决以下是一些常用用法:1 #include 2 #include using namespace std;4 int main() bitset b(M11); 5表示5个二进位6/初始化方式:7/ bitset b;都为08/ bitset b(u); u为unsigned int,如果u = 1,则被初始化为 100009/ bitset b(s); s为字符串,如1101 - *1011010/ bitset b(s, pos, n);从字
31、符串的spos开始,n位长度for(int i = 0; i 5; i+) cout bi;13cout endl b.any(); b中是否存在1的二进制位14cout endl b. none(); b中不存在 1 吗?15cout endl b.count(); b中 1 的二进制位的个数16cout endl b. size(); b中二进制位的个数17cout endl b.test(2); 测试下标为2处是否二进制位为1b.set(4); 把b的下标为4处置1b. reset(); 所有位归零b. reset(3); b的下标3处归零b.flipO; b的所有二进制位逐位取反uns
32、igned long a = b.to_ulong(); b转换为unsigned long类型23return 0;24)C+中的sort函数sort函数在头文件#include 里面,主要是对一个数组进行排序(intarr|数组或者vector数组都行),vector是容器,要用v.begin()和v.endQ表示头尾;而int arrD用arr表示数 组的首地址,arr+n表示尾部1 #include2 #include3 #include4 using namespace std;bool cmp(int a, int b) / cmp函数返回的值是bool类型 return a b;
33、 /从大到小排列7 int main() vector v(10);10 for (int i = 0; i 10; i+) cin vi;12)sort(v.begin(), v.end();/因为这里没有传入参数cmp,所以按照默认,v从小到大排列1415 int arr(10;16 for (int i = 0; i 或者v,比如:return a b;或 者return a b;而不能是 =,(实际上等于号加了也是毫无意义,sort是不稳定的排 序),否则可能会出现段错误C+中使用sort自定义cmp函数 sort默认是从小到大排列的,也可以指定第三个参数cmp函数,然后自己定义一个c
34、mp函数指定 排序规则cmp最好用的还是在结构体中,尤其是很多排序的题目比如一个学生结构体stu有学 号和成绩两个变量,要求如果成绩不同就按照成绩从大到小排列,如果成绩相同就按照学号从小到大 排列,那么就可以写一个cmp数组实现这个看上去有点复杂的排序过程:#include using namespace std;struct stu /定义一个结构体stu, number表示学号,score表示分数 int number;int score;6bool cmp(stu a, stu b) / cmp函数,返回值是bool,传入的参数类型应该是结构体stu类型 if (a.score != b
35、.score) /如果学生分数不同,就按照分数从大到小排列return a.score b.score;else /如果学生分数相同,就按照学号从小到大排列return a.number b.score : a.number b.number;17关于cctype头文件里的一些函数刚刚在头文件那一段中也提到,include 本质上来源于C语言标准函数库中的头文件include ,其实并不属于C+新特性的范畴,在刷PAT一些字符串逻辑题的时候也经常用到,但是很多 人似乎不了解这个头文件中的函数,所以在这里单独提一下可能平时我们判断一个字符是否是字母,可能会写:char c;cin c;if (c
36、 = A & c = a & c = z) cout c is alpha;5但是在cctype中已经定义好了判断这些字符应该所属的范围,直接引入这个头文件并且使用里面的函 数判断即可,无需自己手写(自己手写有时候可能写错或者漏写)1 #include 2 #include 3 using namespace std;4 int main() char c; cin c; if (isalpha(c) cout c is alpha;9 )10 return 0;11 )不仅仅能判断字母,还能判断数字、小写字母、大写字母等C+官方文档中对这些函数归纳成了一个 表格,我也曾经在C+ isalph
37、a、islower、isupper、isalnum、isblank、isspace函数头文件这篇博客中 列出了官网的函数与所属范围总结表,有兴趣的可以看一下:httDS:/总的来说常用的只有以下几个:isalpha字母(包括大写、小写)islower (小写字母)isupper (大写字母)isalnum (字母大写小写+数字)isblank (space和 t )isspace ( space、t、r、n )cctype中除了上面所说的用来判断某个字符是否是某种类型,还有两个经常用到的函数:tolower 和toupper ,作用是将某个字符转为小写或者大写,这样就不用像原来那样手动判断字符
38、c是否是大 写,如果是大写字符就c = c + 32;的方法将chare转为小写字符啦这在字符串处理的题目中也是经 常用到:char c = A;char t = tolower(c); /将c字符转化为小写字符赋值给t,如果c本身就是小写字符也没有关系 cout t; / 此处t为a关于C+11的解释C+11是2011年官方为C+语言带来的新语法新标准,C+11为C+语言带来了很多好用的新特性,比 如 auto、to_string()函数、stoi、stof、unordered_m叩、unordered_set 之类的现在大多数OJ 都是支持C+口语法的,有些编译器在使用的时候需要进行一些设
39、置才能使用C+11中的语法,否则可 能会导致编译器上编译不通过无法运行,比如我曾经写过一篇博客如何在Dev-Cpp中使用C+11中的 函数(在本教程末尾)这个是针对Dev-Cpp编译器的,其他的编译器如果发现不支持也可以百度搜索 一下让编译器支持C+11的方法总之C+11的语法在0J里面是可以使用的而且很多语法很好用以 下讲解一些C+11里面常用的新特性C+11里面很好用的auto声明auto是C+11里面的新特性,可以让编译器根据初始值类型直接推断变量的类型。比如这样:auto x = 100; / x是int变量auto v 1-5; / y是double变量当然这个在算法里面最主要的用处
40、不是这个,而是在STL中使用迭代器的时候,auto可以代替一大长 串的迭代器类型声明:1/本来set的迭代器遍历要这样写:for(set:iterator it = s.begin(); it != s.end(); it+) cout *it ;4 5 /现在可以直接替换成这样的写法:for(auto it = s.begin。; it ! = s .end(); it+) cout *it ;8 C+11特性中基于范围的for循环除了像C语言的for语句for (i = 0; i arr.size(); i+)这样,C+11标准还为C+添加了一种新的for循环方 式,叫做基于范围(range-based)的for循环,这在遍历数组中的每一个