c++ 概念解析

new, delete, malloc, free 之间的关系

malloc 和 free 是 c/c++ 标准库函数,new 和 delete 是 c++的运算符。
运算符 :编译器能够识别并且对运算符语句进行编译。
malloc 和 free 只是动态申请内存和释放内存。
new 和 delete 是经过c++编译器解析为:动态申请内存,转换指针类型,调用对象构造函数  (new的解析) 和  调用对象析构函数,释放内存(delete 的解析)。

new, delete, new[], delete[]

new 和 delete 是用来动态申请单个对象,如果该对象不是内建数据类型,则会额外增加 对象的构造和析构。
new[] 和 delete[] 是用来创建多个对象。c++ 会将 new[] 解析为 申请内存,构造多个对象;将 delete[] 解析为 析构多个对象,释放内存。
总结:使用new 则配对 delete, 使用 new[] 则配对 delete[]. 当使用 new[] 配对 delete 则可能会出现有些对象没有别调用析构函数。

c++有哪些性质(面向对象的特点)

封装,继承和多态。

继承与组合之间的构造函数和析构函数顺序

构造函数:
	继承:先调用基类的构造函数,再调用派生的构造函数。当有多个基类时,调用顺序是继承时从左到右的调用,和构造函数中初始化序列无关
	组合:先调用内嵌对象的构造函数,再调用类本身的构造函数。当有多个内嵌对象时,调用顺序是根据定义对象的先后顺序,和构造函数中初始化序列排序无关。
	继承+组合:先执行继承,后执行组合。

析构函数(与构造函数顺序相反):
	继承:先调用派生类的构造函数,再调用基类的析构函数;当有多个基类时,调用顺序是继承时从右到左的调用。
	组合:先调用类的析构函数,再调用内嵌对象的析构函数;当有多个内嵌对象时,调用顺序是后定义的对象先析构。
	继承+组合:先执行组合+后执行继承

多态,虚函数,纯虚函数,函数和运算符重载

多态
多态:对于不同对象接收相同的消息产生不同的动作。c++ 的多态体现在运行时多态(动态多态) 和编译时多态(静态多态)。
运行时多态是通过虚函数和继承来实现
编译时多态是通过函数重载和运算符重载,以及类模板。
虚函数 和纯虚函数
虚函数:以 virtual 开头声明的成员函数,虚函数允许派生类重新定义。
纯虚函数:以 virtual type fun(type) = 0; 声明的成员函数,有纯虚函数的类不能被创建对象,具有纯虚函数的类称为抽象类,派生类继承抽象类必须实现纯虚函数。
函数重载
函数重载是体现在:函数名相同但 参数个数,参数类型,参数顺序的不同。
	注意:1.函数重载与函数返回值无关,
		  2. 参数类型需注意: const char * p  与 char *p 是类型不同,当两个函数除了 指针是否被const 修饰不同之外,其他都相同,编译器也会通过。
		  如:
		  	void func(int i, char *p);
		  	void func(int i, const char *p); 	//函数重载
		  3. 成员函数 被 const 修饰的真实意义:
		  	class A{
		  		void func(int i) const;
		  		void func(int i);
		  	};
		  	class A 类中两个成员函数也体现函数重载。两个成员函数被编译器编译为:
		  		void func( const A *this, int i);
		  		void func( A *this, int i);

全局对象

全局对象的创建是在main()函数执行之前被调用

float 与 0 比较

float a;
const float flage = 0.000001;
if ( a < flage && a > -falge ) { ...为 0;}

基类析构函数为虚函数

基类析构函数不为虚函数,可能导致派生类的析构函数没有调用,造成内存泄露。

strcpy 函数的实现

char *strcpy(char *strDest, const char *strSrc)
{
	if (strDest == nullptr || strSrc == nullptr ) 
		return nullptr;
	if ( strDest == strSrc )
		return strDest;
		
	char *retAddr= strDest;
	if ( strDest < strSrc ) {
		//strSrc 的值不会被覆盖
		while(  (*strDest++ = *strSrc++) != '\0' );
	}
	else {
		//strSrc 的值可能会被覆盖
		int len =0;
		//算出strSrc的长度
		while( *strSrc != '\0' ) {
			len++;
			strSrc++;
		}
		//从后往前复制
		strDest+= len;	//strSrc 地址 指向 此时应该赋值 '\0'
		while(  retAddr < strDest ) {
			*strDest-- = *strSrc--;
		}
	}
	return retAddr;
}

c 和 c++ 混合编程

头文件
extern "C" {
	void func(int);
}
告诉c++编译器按照c语言方式编译 extern "C" { ... } 里面的函数,不然函数其他C语言无法正真调用到 func函数。因为C语言编译的方式该函数名为 func, 但是C++语言编译函数名可能为 func_int, 反正不是func。所以避免在函数调用中找不到。

类的大小

class A{}; 	//空类大小为 1,对象 为 1
class A{ static int i};	//静态变量的存放在静态存储区 大小为 1 , 对象大小为 1
class A{ int i; }; // 类的大小为 int, 对象为 大小为 int,
class A{ virtual void func() {} }; //虚函数存在虚指针,大小为 指针大小。
派生类继承两个基类,两个基类有虚函数,则大小 增加两个虚指针。

转载内容

链接: link精选 30 个 C++ 面试题(含解析)

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值