笔试面试中CC++重要知识点整理_第1页
笔试面试中CC++重要知识点整理_第2页
笔试面试中CC++重要知识点整理_第3页
已阅读5页,还剩14页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

1、笔试面试中C/C+重要知识点整理(不定期更新)1. C和C+语言中的优先级规则C语言中语言声明的优先级规则如下 (以后分析的基础):A声明从它的第一个名字开始读取,然后按照优先级顺序依次读取B优先级从高到低依次是B. 1声明中被括号括起来的那部分B. 2后缀操作符:括号()表示这是一个函数括号表示这是一个数组B. 3前缀操作符:星号*表示 指向的指针”下面我们使用上述规则来分析以下例子(1) char * const *(*next)();适用觇则解释A宵先,右变凰名“皿就”,幷注意到它直接陂韬号所招住BJ所以先把括号里的东西作为一个整体,得出"next 个播向鳥B然后丸虔抬号外面的

2、东函.在星号前纽利括号后缀之间作出选择B.2R.2规则告诉我们优先级较高的是右边的函放抠号所以得出S函数揃针,指向一个返回W的蹄数"B3燃局 处理就缀"J 補出抬针所聒的内容C最后*把"char * const为指向字符的常量指忡把上述分析结果加以概括这个声明表示"next是一个指针.它指向一堅凹另一个指针.该指针捋向一个类型为char的常蓋拾科X大功告时优(2) char* (*c10)(int *p)一步步分析:先分析括号里面的容,我们知道C是一个数组,保存的是 “.:的指针”然后根据规则B,要先分析后缀,得到指针是一个函数指针。该函数参数为P返回值

3、为char*。最后得到:C是一个数组元素,它的元素类型是函数指针,其所指向的函数的返回值是一个指向char的指针。(3) void(*signal(int sig,void(*func)(int)(int);从signal所在的括号开始提取:void(*signal( ) )(int);首先signal后缀跟的是括号,我们得到signal 是一个函数,然后得到前缀为*表示此函数返回的是一个” 指针”最后得到signal是一个函数,返回函数指针,函数所指向的指针接受一个 int类型的参数并且返回 void。然后我们看signal函数参数本身:void(*func)(int) 表示func是一个函

4、数指针,此指针指向的函数接收一个int参数,返回值是void。如果我们定义typedef void(*ptr_to_func)(int)则表示 ptr_to_func是一个函数指针,该函数接受一个int参数,返回值为void那么上述函数可以写为ptr_to_funcsignal(int sig, ptr_to_func); 表示 signal 是一个函数,接收参数为 int 和ptr_to_func,返回 ptr_to_func ;2. typedef int x10与 #define x int10的区别typedef与宏文本替换之间存在关键性的区别。如下:第一:可以用其他类型说明符对宏名进

5、行扩展,但对typedef所定义的类型名却不能这样做。如下:#define peach intunsigned peach i ; / 可以typedef int banana;unsigned banana i ; / 错误第二:在连续声明的变量中用typedef定义的类型能够保证声明中所有变量均同一种类型,而用#define定义的类型却无法保证。如下:#define int_ptr int*int_ptr chalk, cheese;经过宏扩展,第二行变为:int *chalk, cheese; 这使得chalk与cheese为不同的类型。chalk为int类型的指针,而 cheese只是

6、int 类型变量。typedef char * char_ptr;char_ptr Benley, Royce ; Benley,和Royce类型是相同的。都是指向char的指针原因:#define在编译时仅仅是名称替换而typedef可以被看成一个彻底封装的类型”在了解typedef中变量具体表达什么意思的时候可以按照前面说的优先级规则进行解析。3. 指针与typedeftypedef中使用指针往往带来意外的结果。例:typedef string *pstring;const pstring cstr;c_str究竟代表什么类型。我们知道pstring是指向string的指针很多人都会误认为

7、真正的类型是 const string* cstr。错误原因是将typedef当成#define 直接进行文本扩展了,其实 const修饰的是 pstring 而pstring是 一个指针,因此,正确的等价形式应该是string *const cstr;4. 类与面向对象编程4.1类接口与实现的概念:每个类都定义了一个接口(可以不是很确切的理解为类中访问级别为public的函数为接口)和一个实现。接口由使用该类的代码需要执行的操作组成。实现一般包括该类所需要的数据。实现还包括定义该 类需要的但又不供一般性使用的函数。定义类时,通常先要定义该类的接口,即该类所提供的操作。通过这些操作,可以决定该

8、类完 成其功能所需要的数据,以及是否需要定义一些函数来支持该类的实现。public派生类继承基类的接口,它具有与基类相同的接口。设计良好的类层次中,public派生类的对象可以用在任需要基类对象的地。4.2用struct关键字与class关键定义类以及继承的区别(1)定义类差别struct关键字也可以实现类,用class和struct关键字定义类的唯一差别在于默认访问级别:默认情况下,struct成员的访问级别为 public,而class成员的为private。语法使用也相同,直接将class改为struct即可。(2 )继承差别使用class保留字的派生类默认具有private继承,而用s

9、truct保留字定义的类某人具有public继承。其它则没有任区别。class Base /*.*/;struct D1: Base /*/;/ 默认是 public 继承class D2: Base/*/; / 默认是 private 继承4.3类设计与protected成员可以认为protected访问标号是private和public的混合:(1 )像private成员一样,protected成员不能被类的用户访问(2 )像public成员一样,protected成员可以被该类的派生类访问。class Baseprotected:int price;;class ltem_Base :p

10、ublic Base;Base b;Item_Base d;b.price; / errord.price: / OK小结(帮助理解为什么设置 protected类型):如果没有继承,类只有两种用户:类本身的成员以及该类的用户,将类划分为private和public访问级别反映了用户类型的这一分割:用户只能访问public接口,类成员和友元既能访问public成员也能访问private成员。有了继承,就有了第三种用户:从派生类定义新类的程序员。派生类提供者通常(但不总是)需要访问(类型为private的)基类实现(见4.1实现概念)。为了允这种访问而仍然禁止对实现的一般访问。所以提供了附加的p

11、rotected访问标号。类的protected部分仍然不能被一般程序访问,但可以被派生类访问。定义基类时,将成员设置为 public的标准并没有改变:仍然是接口函数应该 为public而数据一般不应为 public。被继承的类必须决定实现那些部分为protected哪些部分为private。希望禁止派生类访问的成员应该设为private,提供派生类实现所需操作或数据的成员设为protected。换句话说,提供给派生类的接口是protected成员和public成员的组合。4.4派生类与虚函数概述(1)定义为virtual的函数是希望派生类重新定义。希望派生类继承的函数不能定义为虚函数。如果派

12、生类没有重新定义某个虚函数,则在调用的时候会使用基类中定义的版本。(2) 派生类中函数的声明必须与基类中定义的式完全匹配,但有一个例外:返回对基类类型的引用(或指针)的虚函数。派生类中的虚函数可以返回基类函数所返回类型的派生类的引用(或指针)。比如:ltem_base类可以定义返回ltem_base*的函数。如果这样,派生类Bulk_item类中定义的实例可以定义返回为 ltem_base* 或者 Bulk_item*(3) 一旦函数在基类中声明为虚函数,它就一直为虚函数,派生类无法改变该函数为虚函数这一事实。派生类重新定义虚函数时,可以使用virtual保留字,也可以省略。4.5 virtu

13、al函数详解(待更新)要触发动态绑定,必须满足两个条件:第一:只有指定为虚函数的成员函数才能进行动态绑定。第 二,必须通过基类类型的引用或者指针进行函数调用。下面重点讲下第二个条件。由于每个派生类都包含基类部分,所以可将基类对象引用或者指针绑定到派生类对象的基类部分 (派生类对象本身不会改变)。如下:double print_total(const ltem_base&, size_t);ltem_base item;print_total(item, 10); / OKBulk_item bulk;print_total(bulk, 10);/ OK 引用 bulk 中 ltem_b

14、ase 的部分。ltem_base *item = &bulk; / OK , 指针指向 bulk 的 ltem_base 部分。通过引用或者指针调用虚函数时,编译器将生成代码,在运行时确定调用哪个函数。比如:假定print_total为虚函数,在基类ltem_base和派生类Bulk_item中都有定义。函数原型: void print_total(ostream& os, const ltem_base &i tem, size_t n);ltem_base base;Bulk_item derived;print_total(count, base, 10); /

15、将调用基类 ltem_base 中的 print_total 函数print_total(count, derivede,10); /将调用派生类中的 print_total 函数。在某些情况下,希望覆盖虚函数的机制并强制函数使用虚函数的特定版本,这时可以使用作用域操 作符。ltem_base *baseP = &derived;double d = baseP->ltem_base:net_price(42);小结:引用和指针的静态类型与动态类型可以不同,这是C+支持多态性的基。当通过基类引用或者指针滴哦啊用基类中定义的函数时,我们并不知道执行函数的对象的确切类型,执行函数的对

16、象可能是基类类型的,也可能是派生类类型的。女保调用非虚函数,则无论实际对象是什么类型,都执行基类中所定义的哦函数。如果调用 虚函数,则直到运行时才能确定调用哪个函数。4.5派生类到基类的转换:C+ primer 488没有想好怎么整理4.6基类与派生类中构造函数和复制控制:构造函数和复制控制成员不能被继承,每个类定义自己的构造函数和复制控制成员,如果不定义,则 编译器将合成一个。继承对基类中构造函数的唯一影响是,某些类需要只希望派生类使用的特殊构造函数,这样的构造函 数应该定义为 protected。派生类构造函数派生类构造函数受继承关系的影响,每个派生类构造函数除了初始化自己的数据成员之外,

17、 还要初始化基类。对于合成的派生类默认构造函数,先调用基类的默认构造函数初始化(问题,女口果基类没有定义默认构造函数咋整,要试验下 )再默认初始 化自己的对象成员。具体语法参见 P491 c+ primer派生类析构函数派生类析构函数不负责撤销基类对象的成员。编译器总是显式调用派生类对象基类部分的析构函数。每个析构函数只负责清除自己的成员:class Derived: public Base/ Base:Base()函数会自动被调用Derived。;对象的撤销顺序与构造顺序相反:首先运行派生类析构函数,然后按照继承层次依次向上调用各基类的析构函数。463虚析构函数当阐述指向动态分配对象的指针时

18、,需要运行析构函数在释放对象之前清除对象。如果把析构函数设置为虚函数,运行哪个析构函数将因指针所指向对象类型的不同而不同:ltem_base *itemP = new ltem_base;delete itemP; /基类的析构函数被调用itemP = new Bulk_item;delete itemP; /派生类的析构函数被调用如果不把析构函数定义为虚函数,则会一直调用基类的析构函数,从而引发程序异常。像其他虚函数一样,析构函数的虚函数性质将继承,因此,如果层次中根类的析构函数为虚函 数,则派生类析构函数也将是虚函数,无论派生类显式定义析构函数还是使用合成析构函数,派生类析构 函数都是虚函

19、数。构造函数不是虚函数:构造函数实在对象完全构造之前运行的,在构造函数运行的时候,对象的动态类型还不够完整(待理解),所以构造函数不是虚函数。4.7继承情况下的类的作用域继承层次中函数调用遵循以下四个步骤:(1)首先确定进行函数调用的对象,引用或者指针的静态类型。(2)在该类中查找函数,如果找不到,就直接在基类中查找,如此循环着类的继承链往上找,直到找到该函数或者查找完最后一个类。如果不能再类或者相关基类中找到该名字,则调用是错误的。(3)一旦找到了该名字,进行常规类型检查(参数类型检查等),查看该函数调用是否合法(4)假定函数调用合法,编译器就生成代码,如果函数是虚函数并且通过引用或者指针调

20、用,则编译器生成代码以确定根据对象的动态类型运行哪个函数版本,否则,编译器生成代码直接调用函数。举例1:Bulk_item bulk;cout << bulk.book();book的使用将这样确定:(1) bulk是Bulk_item类对象,在Bulk类中查找,找不到名字 book (根据上面第一步,确 定静态类型为Bulk_item,然后进入第二步)(2) 因为从ltem_base派生Bulk_item,所以接着在ltem_base类中查找,找到 book,名字成功确定。举例2:struct Baseint menfcn();;struct Derived: Baseint m

21、enfcn(int);Derived d; Base b;b.memfcn(); /调用基类的函数d.menfcn(10); /调用派生类函数d.menfcn();/ 错误:d.Base:menfcn(); / 调用基类函数第三个调用中出现错误, 原因是,Derived中的么么fcn声明隐藏了 Base中的声明。 原因 是,根据上面规则,一旦找到了名字,编译器就不会再继续查找了。 而是进行常规检查,由于调用与Derived 中的memfcn不匹配,该定义希望接受int实参,而这个函数调用没有提供那样的实参,所以错误如果派生类重新定义了重载成员,则通过派生类行只能访问派生类中重新定义的那些成 员

22、。举例3 :通过基类指针或者引用调用假定print_total为虚函数,在基类Item_base和派生类Bulk_item中都有定义。函数原型: void print_total(ostream& os, const Item_base &i tem, size_t n);Item_base base;Bulk_item derived;print_total(count, derivede,10); /将调用派生类中的 print_total 函数。如果print_total不是虚函数,根据上面的步骤,将直接调用基类ltem_base中的print_total版本由于prin

23、t_total中第二个参数的静态类型为ltem_base所以根据规则(1),先ltem_base中查找print_total ,然后进行常规检查,参数没有错,由于函数是虚函数之后根据规则(4), print_total(count, base,10);用基类 ltem_base 中的 print_total 函数,print_total(count, derivede,10);,调用派生类中的 print_total 函 数。现在可以理解为什么虚函数在基类和派生类中拥有同一原型了,如果没有同一原型,比如基类与派生类中参数不同,根据规则 3,确定基类中参数没有问题时,如果根据规则4实际调用的是派

24、生类中的函数时由于参数不同就会岀现错误。4.8细节知识点4.8.1 explicit 关键字我们可以将构造函数声明为explicit,来防止在需要隐式转换的上下文中使用构造函数。例如class Sales_itempublic:Sales_item(const string &book = ""):isbn(book),units_sold(0)bool same_isbn(const Sales_item &rhs) const;每个构造函数都定义了一个隐式转换。因此,在期待一个Sales_item类型对象的地,可以使用一个 string或者istream

25、 :如下string null_book = "9-1111-1111"item.same_isbn(null_book);以上程序中,本来程序期待一个Sales_item对象作为实参,编译器使用接受一个string的Sales_item构造函数从 null_book生成一个新的 Sales_item对象,新生成的临时的 Sales_item对 象被传递给same_isbn。如果我们不想要编译器隐式的转换,可以将构造函数声明为explicit o 注意的是explicit关键字只能用于类部的构造函数声明上。在类定义外部所作的定义中不再重复它。比如以下是错explicit S

26、ales_item: sales_item(istrea m& is)/错误,explicit只在类构造函数的声明上加上explicit关键字后,以下就不能编译通过item.same_isbn(null_book); / error : string constructor is explicit当然我们可以显式的使用构造函数来生成转换,如下item.same_isbn(Sales_item(null_book); / OK总结:通常,除非有明显的理由需要隐式转换,否则,构造函数应该为explicit o将构造函数设置为explicit可以避免错误,并且当转换有用时用户可以显式的构造对

27、象4. 9虚函数与纯虚函数区别(1)虚函数在子类里面也可以不重载的;但纯虚必须在子类去实现(2)带纯虚函数的类叫虚基类,这种基类不能直接生成对象,而只有被继承,并重写其虚函数后,才能使用。这样的类也叫抽象类。虚函数是为了继承接口和默认行为5. C+存分配与释放浅析(以后会深入讲解更新)我们一般以一下的形式对 C+进行存配置和释放。class Foo ;Foo *pf = new Foo; /配置存,然后构造对象其中new算式含两个阶段操作:调用::operator new操作符配置存(2)调用Foo:Foo()构造对象容delete算式也包含两个阶段操作:(1)调用Foo:Foo()将对象析构

28、(2)调用:operator delete 释放存如果我们想要创建对象的时候初始化可以用如下形式:string *ps = new string("Hello") ;/ *ps is "Hello"与malloc和free的区别(1) new/delete 调用 constructor/destructor.Malloc/free 不会(2) new不需要类型强制转换。.Malloc要对放回的指针强制类型转换(3) new/delete操作符可以被重载,malloc/free 不会(4) new并不会强制要求你计算所需要的存(不像malloc)6. C语

29、言数组与字符串1. 数组的定义与初始化在数组定义时,如果没有显式提供元素初值贝U数组元素会像普通变量一样初始化:在函数体外定义的置数组,其元素均初始化为0;在函数体定义的置数组,其元素无初始化不管数组在哪里定义,如果元素为类类型,则自动调用该类的默认构造函数进 行初始化;如果该类没有默认构造函数则U必须为该数组的元素提供显式初始化2. 特殊的字符数组字符数组有两种初始化式(1)用一组由花括号括起来、逗号隔开的字符字面值进行初始化。(2)用一个字符串字面值进行初始化。两种初始化形式有所不同。第(2)种初始化式包含一个额外的空字符(null)用 于结束字符串。举例如下:char ca1 = C;

30、+',+ ;/ no nullchar ca2 = C; +',+: 0 ; / explicit null (也可以写为 C:'+ ',o )char ca3 = C+ ”;/ null added automatically上例中ca1的维数为3, ca2和ca3的维数为4注:strlen计算字符串的大小时从字符串开头直到遇到null为止,在计算strlen(ca1)时结果是未知的。原因是不知道时会遇到null 而 strlen(ca2)与 strlen(ca3)的结果都将是33. string s(n, c;/将s4初始化为字符 '的n个副本vector<T> v(n, i); / v 包含n个值为i的元素4. vector类型的值初始化vector中如果没有指定元素的初始化式,那

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论