类
1. 定义一个空类型,里面没有任何成员变量和成员函数,对该类型求sizeof,得到的结果是什么?
答:结果是1。空类型的实例中不包含任何信息,但是当我们声明该类型时,它必须在内存中占有一定的空间,否则无法使用这些实例。至于占用多少空间,由编译器决定。Visual Studio中每个空类型的实例占有1字节的空间。
2.如果在一个空类型中添加一个构造函数和析构函数,再对该类型求sizeof,得到的结果是多少?
答:结果还是1。调用构造函数和析构函数只需要知道函数的地址即可。而这些函数的地址只与类型相关,而与类型的实例无关,编译器也不会因为这两个函数还在实例中添加任何的信息。
3.如果把上面的析构函数修改为虚函数,得到的结果是多少?
答:结果是4或者8。如果类型中含有虚函数,则编译器会在类型中添加一个虚函数表。同时在类型的每一个实例中添加一个虚指针,指向类型中的虚函数表。在32位的机器上,一个指针占有4个字节的空间,因此sizeof为4;而在64位的机器上,一个指针占有8个字节的空间,sizeof为8。
4.string类的默认构造函数、复制构造函数、赋值函数和析构函数?
MyString{
public:
MyString::CMyString(const CMyString &another);
MyString::CMyString(char * pDate = null);
MyString& CMyString::operator = (const CMyString &another);
~MyString(void);
private:
char *m_pDate;
普通构造函数
MyString::MyString(const char *pdate)
{
if(pdate == NULL)
{
m_pDate = new char[1];
*m_pDate = '\0';
}
else
{
int len = sizeof(pdate)/sizeof(char);
m_pDate = new char[len+1];
strcpy(m_pDate , pdate);
}
}
复制构造函数
用于:
(1)根据另一个同类型的对象显示或者隐式初始化一个对象;
(2)复制一个对象,将它作为实参传给一个函数;
(3)从函数返回时复制一个对象;
(4)初始化顺序容器中的元素;
(5)根据元素初始化式列表初始化数组元素。
MyString::MyString(const MyString &another)
{
int len = strlen(another.m_pDate);
m_pDate = new char[len+1];
strcpy(m_pDate , another.m_pDate);
}
赋值操作符函数
MyString & MyString::operator = (const MyString &another)
{
if( this == &another) //如果两种相同,一旦释放了自身的内存,传入的参数的内存也同时被释放了,再也找不到需要赋值的内容
{
return *this;
}
delete []m_pDate;
m_pDate = NULL;
m_pDate = new char[strlen(another.m_pDate)+1];
strcpy( m_pDate, another.m_pDate);
return *this;
}
把返回值的类型声明为该类型的引用,并在函数结束前返回实例自身的引用(return *this)。只有返回一个引用,才允许进行连续赋值。
把传入的参数的类型声明为常量引用。如果不是引用而是实例,那么从形参到实参会调用一次复制构造函数。加上const是防止被修改。
考虑异常安全性的问题。如果new char时候抛出异常,m_pDate将是一个空指针。
MyString & MyString::operator = (const MyString &another)
{
if(this != &another)
{
Mystring strTemp(another);
char * temp = strTemp.m_pDate;
strTemp.m_pDate = m_pDate;
m_pDate = temp;
}
return *this;
}
析构函数
~MyString::MyString(void)
{
delete []m_pDate;
}
5.C++中可以用struct和class来定义类型,这两种类型有什么区别?
答:如果没有标明成员函数或成员变量的访问控制级别,在struct中默认的是public,在class中默认的时private。
6.为什么不能在构造和析构过程中调用虚函数?
答:在构造和析构期间不要调用virtual函数,因为这类调用从不下降至derived class。7.为什么没有默认构造函数的对象、const对象和引用类型的对象需要在函数初始化列表中进行初始化呢?
答:构造函数分为两个阶段执行:(1) 初始化阶段;(2) 普通的计算阶段。计算阶段由构造函数函数体中的所有语句组成。如果没有在初始化列表中进行初始化,则需要在构造函数函数体中对数据成员进行赋值。const对象和引用类型的对象需要在初始化阶段进行初始化,赋值的初始化过程是不允许的。
7.成员变量的初始化顺序?
答:类中成员变量的初始化顺序只与成员变量的声明顺序相关,而与在初始化列表中得顺序无关。