首页 > 编程语言 >C++面试八股文:什么是构造函数?

C++面试八股文:什么是构造函数?

时间:2023-07-13 23:44:30浏览次数:35  
标签:初始化 Singleton 八股文 int 面试官 C++ Foo 构造函数

C++面试八股文:什么是构造函数?

某日二师兄参加XXX科技公司的C++工程师开发岗位第29面:

面试官:什么是构造函数?

二师兄:构造函数是一种特殊的成员函数,用于创建和初始化类的对象。构造函数的名称与类的名称相同,并且没有返回类型。构造函数在对象被创建时自动调用。

struct Foo
{
  Foo(int v):val(i){}	//构造函数
private:
  int val;
};

面试官:什么是默认构造函数?什么情况下默认构造函数会被创建?

二师兄:没有任何参数的构造函数(所有参数都要默认参数的构造函数也是)。一般定义类时没有显式的声明任何构造函数,默认构造函数会被编译器自动创建。

struct Foo
{
private:
  int val;
};	//此时默认构造函数会被创建

二师兄:当然就算为类自定义了构造函数,我们也可以通过Foo()=default为类显式定义一个默认构造函数。

面试官:什么是构造函数初始值列表?

二师兄:是为了初始化成员变量所传入的参数列表:

class Foo
{
public:
    Foo(int i, long l):ival_(i),lval_(l){}	//初始值列表
private:
    int ival_;
    long lval_;
};

面试官:上面的构造函数和以下的构造函数有什么区别?

Foo(int i, long l)
{
    ival_ = i;
    lval_ = l;
}

二师兄:这是初始化与赋值的区别。这段代码中的ival_lval_先被默认初始化,然后被赋值。而初始化列表是直接初始化,少了一步赋值。

面试官:如果把构造函数写成Foo(int i, long l):lval(l),ival_(i){}会有什么问题吗?

二师兄:成员初始化的顺序尽量要和定义的顺序保持一致。如下面的代码,就是未定义的:

class Foo
{
public:
    Foo(int i):jval_(i),ival_(jval_){}	//未定义的行为,因为ival先被初始化,这时候jval是未定义的
private:
    int ival_;
    int jval_;
};

面试官:什么是委托构造函数?

二师兄:构造函数在构造对象的时候把一部分任务委托给其他构造函数进行构造,这是C++11引入的新特性:

class Foo
{
public:
    Foo(int i, long l):ival_(i),lval_(l){}
    Foo(int i):Foo(i,0){}   //委托给Foo(int i, long l)
private:
    int ival_;
    long lval_;
};

面试官:如果构造函数没有初始化任何成员变量,使用这个构造函数会发生什么?

二师兄:成员变量将会被默认初始化。

面试官:什么是默认初始化

二师兄:如果是内置类型(如boolintdouble),将不被初始化,如果是类类型,将执行类类型的的默认构造函数初始化变量。如果类类型的默认构造函数是删除的(=delete)或定义了其他构造函数但是没有定义默认构造函数的,将不能通过编译。

二师兄:类类型的初始化时一个循环的过程,如果类类型中有类类型成员,初始化方式和以上描述的一致。

struct Foo{ int a;}
struct Goo
{
    int b;
    Foo f;
};	
Goo g;	//此g.b是默认初始化,值不确定。Foo中的a也是默认初始化,所以g.f.a的值也是不确定的。

面试官:可以使用virtual修饰构造函数吗?

二师兄:不可以,因为构造函数在对象构造阶段调用,虚表尚未建立,所以无法调用虚函数实现多态。

面试官:可以使用const修饰构造函数吗?

二师兄:不可以,因为构造函数需要初始化成员变量,这与const修饰成员函数的意义相悖。

面试官:可以使用constexpr修饰构造函数吗?

二师兄:可以。这表明类的对象可以在编译器构造。我们所熟悉的std::array的构造函数在C++20下就是constexpr的。

面试官:什么情况下会将一个类的构造函数定义为私有的?

二师兄:一般不希望直接通过类型定义对象,如C++的单例模式:

class Singleton
{
public:
    static Singleton& Instance()
    {
        static Singleton instance;
        return instance;
    }
    Singleton(const Singleton&) = delete;
    Singleton(Singleton&&) = delete;
    Singleton& operator=(const Singleton&) = delete;
    Singleton& operator=(Singleton&&) = delete;
private:
    Singleton() = default;
    ~Singleton() = default;
};

Singleton s; //编译失败
Singleton& s = Singleton::Instance();	//编译成功

面试官:最后一个问题,你知声明、定义、初始化、赋值的区别吗?

二师兄:声明是告诉编译器这里有个符号,但不分配内存。定义告诉编译器,这里有个符号,要分配一块内存给它。初始化时在分配内存的时候给它一个初始值。赋值是将这块内存原来的值擦除,给它填入一个新值。

面试官:好的,今天的面试结束了,请回去等通知吧。

C++类的构造函数的基本考点都在这里了,小伙伴本要理解这些设计及设计背后的取舍,面对面试官的拷问才能对答如流哦。

标签:初始化,Singleton,八股文,int,面试官,C++,Foo,构造函数
From: https://www.cnblogs.com/bujidao1128/p/17552505.html

相关文章

  • c++ 段错误(核心已转储)
    一、什么是段错误?段错误应该就是访问了不可访问的内存,这个内存区要么是不存在的,要么是受到系统保护的,还有可能是缺少文件或者文件损坏。二、段错误产生的原因1、访问不存在的内存地址#include<iostream>#include<algorithm>#include<vector>#include<stdio.h>#include<st......
  • C++ 文件和流
     到目前为止,我们已经使用了 iostream 标准库,它提供了 cin 和 cout 方法分别用于从标准输入读取流和向标准输出写入流。本教程介绍如何从文件读取流和向文件写入流。这就需要用到C++中另一个标准库 fstream,它定义了三个新的数据类型:数据类型描述ofstream该数据类......
  • C++ STL容器之vector、list
    (1)vector连续存储的容器,动态数组,在堆上分配空间底层实现:数组扩容机制:vector增加(插入)新元素时,如果未超过当时的容量,则还有剩余空间,那么直接添加到最后(插入指定位置),然后调整迭代器。如果没有剩余空间了,则会重新配置原有元素个数的两倍空间,然后将原空间元素通过复制的方式初始......
  • 现代C++(Modern C++)基本用法实践:一、类型推导
    概述类型推导主要是依赖auto关键字和decltype关键字/运算符实现的,具体用法参考下面的例子。二者特点:auto用于声明时推导遍历decltype用于推导各种表达式,decltype(var)中var也是一种称为变量表达式的表达式二者都是在编译时进行推导。引用类型推断:decltype推断变量类型时......
  • 现代C++(Modern C++)基本用法实践:五、智能指针(Smart Pointers)
    概述c++效率较高的一个原因是我们可以自己定制策略手动申请和释放内存,当然,也伴随着开发效率降低和内存泄漏的风险。为了减少手动管理内存带来的困扰,c++提出了智能指针,可以帮助我们进行内存管理,有三种:std::unique_ptr是一种独占所有权的智能指针,它不允许多个指针指向同一个对......
  • 现代C++(Modern C++)基本用法实践:四、模板
    概述C++的模板是泛型编程思想的一种实现。C++是强类型语言,处处强调类型。同样的加法运算,int和float的加法运算需定义两个函数(重载),而使用模板则可以只用一个函数(见下面示例)。这类似我们面向对象所说的多态(定义加法运算,各个类型有不同的实现),所以是所谓静多态的一种实现方式,不同的......
  • 现代C++(Modern C++)基本用法实践:三、移动语义
    概述移动移动(move)语义C++引入了一种新的内存优化,以避免不必要的拷贝。在构造或者赋值的时候,如果实参是右值(或者左值由std::move转换成右值),便会匹配移动语义的函数调用如下述举例的Str(Str&&obj)。移动语义的本质是将资源(内存/句柄)转移给另一个对象,被转移资源的对象不应再被使......
  • 现代C++(Modern C++)基本用法实践:二、Lambda表达式
    概述lambda表达式,有时也被称为匿名函数。他提供了简短的,内联的函数对象。用法形式如:[capture](parameters)->return_type{body}具体用法如下文举例它的实现是由编译器决定的,在我的编译器上他是通过创建一个匿名类,通过重载()运算符,成为一个可调用对象,从而实现调用,类似://......
  • 现代C++(Modern C++)基本用法实践:七、范围遍历
    概述c++的for循环在语法上有些刻板,近几个版本对此进行了优化,支持了基于范围的for循环用法举例参考测试项目代码ModernCppTest/modrenc_range_for.cpp主要内容:数组遍历vector遍历字符串遍历map遍历#include"ModernCppTestHeader.h"#include<vector>#include<map>......
  • 现代C++(Modern C++)基本用法实践:六、constexpr编译时计算
    概述constexpr修饰的变量、函数、对象构造函数表示在编译时就可以确定。它经常用来计算一些编译期可以确定常数,和常数组成的表。比如编译时确定10000以内所有的素数,运行时用的时候直接查表。用法举例参考测试项目代码ModernCppTest/modrenc_constexpr.cpp主要内容:constexpr......