C++拷贝构造函数_形参为什么只能是引用(不能传值或指针)

本文深入探讨了拷贝构造函数的正确使用方式及其为何不能使用值传递或指针作为形参的原因,通过实例展示了错误使用可能导致的无限递归问题。

先给出答案:

拷贝构造函数不能用数值或指针形参原因,不是为了节省创建副本的空间与时间。而是那样做会无限循环递归下去。

 

举个例子来看一下为什么会这样。(反例)

class Example()
{
public:
    Example(int a):aa(a) {}    //构造函数

    Example(Example ex)        //拷贝构造函数(值传递参数)
    {
        aa = ex.aa;
    }
private:
    int aa;
};

int main()
{
    Example e1(10);
    Example e2 = e1;
    
    return 0;
}

执行 Example e2 = e1; 时

调用了拷贝构造函数(传值型的)

相当于 e2.Example(e1);

在函数中,因为是值传递所以又需要用e1的值拷贝创建一个副本对象ex,则又需要调用拷贝构造函数ex.Example(e1)

这样就会无限重复下去来完成创建副本ex,无限调用ex.Example(e1)

这就是为什么拷贝构造函数不能使用值传递的形参原因。

 

正确拷贝构造函数:

Example(const Example &ex)   
    {
        aa = ex.aa;
    }

 

..................................................不能用指针的原因待更............................

<think>好的,我现在需要回答用户关于C++中常引用作为函数形参的特点及优势的问题。首先,我得回忆一下常引用的基本概念。常引用就是用const修饰的引用,比如const int& a。用户的问题可能涉及到为什么使用常引用是普通引用,以及它有什么好处。 根据提供的引用资料,引用[2]提到常引用可以防止形参被意外修改,这样在函数内部就不能改变实参的,保护了数据。引用[3]和[4]比较了引用指针作为参数的优缺点,指出引用更直观,且需要像指针那样重复解引用,同时常引用还能避免必要的拷贝,提高效率。 接下来,我需要整理这些点,确保覆盖以下几个方面: 1. **防止修改**:常引用阻止函数内部修改实参,保证数据安全。 2. **避免拷贝**:对于大对象,使用常引用避免复制,提升性能。 3. **兼容性**:可以接受常量和非常量实参,提高函数的适用性。 4. **类型安全**:相比于指针引用更安全,易出错。 5. **与指针的对比**:引用需要解引用,语法更简洁。 同时,需要引用用户提供的资料来支持这些观点,比如引用[2]中的例子说明修改常引用会导致错误,引用[3]和[4]强调引用的直观性和效率优势。还要注意用户提到的“使用引用作为函数参数的优点”,特别是常引用可修改性和效率提升。 可能需要注意用户是否有其他潜在疑问,比如常引用和普通引用的区别,者何时应该使用常引用。但根据问题,主要集中在其特点和优势,所以需要紧扣这些方面,结构清晰,分点说明。 最后,生成相关的问题,例如常引用的使用场景、与指针参数的比较、如何选择参方式等,确保问题与主题相关,帮助用户进一步理解。</think>### C++引用作为函数形参的特点及优势 #### 1. **防止意外修改实参** 常引用(`const T&`)通过 `const` 修饰符限制函数内部对实参的修改。例如: ```cpp void print(const std::string& str) { // str[0] = 'A'; // 错误!常引用禁止修改 std::cout << str; } ``` 此特性适用于需要**只读访问**的场景(如打印、计算),避免因误操作破坏原始数据[^2]。 #### 2. **避免拷贝开销,提升性能** 递大型对象(如容器、类实例)时,直接会触发拷贝构造函数,而常引用**直接绑定实参**,无需复制。例如: ```cpp // :触发拷贝(性能低) void processVector(std::vector<int> data); // 常引用:无拷贝(性能高) void processVector(const std::vector<int>& data); ``` 这对性能敏感的场景(如高频调用、大数据处理)尤为重要[^4]。 #### 3. **兼容常量与非常量实参** 常引用可以接受**常量非常量实参**,而普通引用仅接受非常量。例如: ```cpp void func(const int& a); // 可接受 int const int void func(int& a); // 仅接受 int ``` 这一特性增强了函数的通用性,例如标准库中的容器操作(如 `std::vector::size()` 返回 `const_reference`)[^1]。 #### 4. **类型安全与代码简洁性** 相比指针引用无需解引用操作(`*`),语法更直观。例如: ```cpp // 指针形式 void swap(int* a, int* b) { int tmp = *a; *a = *b; *b = tmp; } // 引用形式 void swap(int& a, int& b) { int tmp = a; a = b; b = tmp; } ``` 常引用进一步结合了安全性和简洁性,适合替代指针作为参数[^3]。 #### 5. **与右引用结合优化资源** 在C++11及以后,常引用可与右引用配合实现**移动语义**,避免深拷贝: ```cpp void process(const std::string& str); // 接受左 void process(std::string&& str); // 接受右(可移动资源) ``` ---
评论 7
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值