news 2026/6/10 0:51:27

【C++】--函数参数传递:传值与传引用的深度解析

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
【C++】--函数参数传递:传值与传引用的深度解析

前言:在 C++ 函数调用中,参数传递方式直接影响程序的效率、正确性和可读性。传值和传引用是两种最基本的参数传递机制,它们在内存使用、性能表现和使用场景上有着显著差异。本篇博客将详细解析这两种传递方式的工作原理、优缺点及适用场景。

一.传值传参

传值传参是最直观的参数传递方式,当函数被调用时,实参的值会被复制一份,然后传递给形参。这意味着函数内部对形参的任何修改都不会影响到外部的实参。

传值传参的工作原理

代码语言:javascript

AI代码解释

#include <iostream> using namespace std; // 传值传参示例 void increment(int num) { num++; // 仅修改函数内部的副本 cout << "函数内部: " << num << endl; // 输出11 } int main() { int x = 10; cout << "调用前: " << x << endl; // 输出10 increment(x); // 传递x的值 cout << "调用后: " << x << endl; // 仍输出10,未被修改 return 0; }

在这个例子中,increment函数接收的是x的一个副本,函数内部对num的修改不会影响到外部的x。这是因为实参和形参存储在内存中的不同位置。

传值传参的特点
  1. 安全性高:函数内部操作不会影响外部变量,避免了意外修改
  2. 独立性强:形参是独立的变量,有自己的内存空间
  3. 存在拷贝开销:会复制实参的值,对于大型对象可能影响性能
  4. 适合小型数据:对于基本数据类型(int、float 等)非常高效
传值传参的代价

当传递大型对象(如复杂的类实例或大型数组)时,传值方式的性能问题会变得明显:

代码语言:javascript

AI代码解释

#include <iostream> #include <string> using namespace std; // 定义一个大型对象 class LargeObject { private: string data[1000]; // 包含大量数据 public: LargeObject() { // 构造函数:初始化数据 for (int i = 0; i < 1000; i++) { data[i] = "sample data"; } cout << "LargeObject 构造函数被调用" << endl; } // 拷贝构造函数 LargeObject(const LargeObject& other) { for (int i = 0; i < 1000; i++) { data[i] = other.data[i]; } cout << "LargeObject 拷贝构造函数被调用" << endl; } }; // 传值传递大型对象 void processObject(LargeObject obj) { // 处理对象... } int main() { LargeObject lo; processObject(lo); // 会触发拷贝构造函数 return 0; }

运行这段代码会发现,传递大型对象时会调用拷贝构造函数,执行大量的数据复制操作,这在性能敏感的场景中是不可接受的。


二.传引用传参

传引用传参是将实参的引用(别名)传递给函数,函数内部对形参的操作实际上是直接操作实参本身。这种方式不会产生数据拷贝,效率更高。

传引用传参的工作原理

代码语言:javascript

AI代码解释

#include <iostream> using namespace std; // 传引用传参示例 void increment(int& num) { // 使用&声明引用参数 num++; // 直接修改实参 cout << "函数内部: " << num << endl; // 输出11 } int main() { int x = 10; cout << "调用前: " << x << endl; // 输出10 increment(x); // 传递x的引用 cout << "调用后: " << x << endl; // 输出11,已被修改 return 0; }

在这个例子中,increment函数接收的是x的引用(别名),函数内部对num的修改会直接反映到外部的x上。实参和形参实际上指向内存中的同一个位置。

传引用传参的特点
  1. 无拷贝开销:不会复制实参,直接操作原始数据,效率高
  2. 可以修改实参:函数内部的修改会影响外部变量
  3. 适合大型对象:传递大型对象时性能优势明显
  4. 语法简洁:使用方式与传值类似,但效率更高
常量引用

有时我们希望享受引用传递的效率,又不希望函数修改实参,这时可以使用常量引用(const reference):

代码语言:javascript

AI代码解释

#include <iostream> #include <string> using namespace std; // 使用常量引用传递,保证不修改实参 void printString(const string& str) { // str += " modified"; // 错误!常量引用不能被修改 cout << str << endl; } int main() { string message = "Hello, World!"; printString(message); // 传递引用但不允许修改 return 0; }
常量引用的优势:
  • 避免不必要的拷贝,提高效率
  • 明确告知函数设计者和使用者,该参数不会被修改
  • 可以接收临时对象(如字符串字面量)作为参数

三.传值与传引用的对比分析

特性

传值传参

传引用传参

内存开销

有拷贝,开销较大

无拷贝,开销小

实参修改

不会影响实参

会直接修改实参

空值处理

无此问题

引用不能为 null,必须指向有效对象

适用场景

基本数据类型、小型结构体

大型对象、需要修改实参的场景

安全性

高(不会意外修改外部数据)

中(需注意对实参的修改)

临时对象

可以接收

常量引用可以接收,普通引用不行

何时使用传值,何时使用传引用?
优先使用传值的场景:
  1. 传递基本数据类型(int、float、char 等)
  2. 传递小型结构体或类(拷贝成本低)
  3. 不希望函数修改实参的值
  4. 函数需要保留参数的原始值用于内部操作

代码语言:javascript

AI代码解释

// 传值适合小型数据 int square(int num) { return num * num; // 仅使用值,不修改 }
优先使用传引用的场景:
  1. 传递大型对象(避免高昂的拷贝成本)
  2. 需要在函数内部修改实参
  3. 传递容器或复杂数据结构
  4. 函数需要返回多个结果(通过修改引用参数)

代码语言:javascript

AI代码解释

// 传引用适合大型对象和需要修改实参的场景 void splitString(const string& input, string& first, string& second) { size_t spacePos = input.find(' '); if (spacePos != string::npos) { first = input.substr(0, spacePos); second = input.substr(spacePos + 1); } }
常见陷阱与最佳实践
  • 避免返回局部变量的引用:局部变量在函数返回后会被销毁,引用将指向无效内存

代码语言:javascript

AI代码解释

// 错误示例:返回局部变量的引用 int& badFunction() { int temp = 10; return temp; // 危险!temp将在函数结束后被销毁 }
  • 优先使用 const 引用接收只读参数:既保证效率,又防止意外修改
  • 基本类型尽量传值:对于 int、double 等基本类型,传值和传引用效率差异很小,但传值更直观安全
  • 明确你的意图:如果函数需要修改参数,使用引用;如果不需要,考虑 const 引用或传值
  • 注意默认参数与引用:引用参数不能有默认值,这是 C++ 的语法规定

传值和传引用是 C++ 中两种基本的参数传递方式,它们各有优劣:

  • 传值方式简单直观,安全性高,但存在拷贝开销,适合传递小型数据
  • 传引用方式效率高,无拷贝开销,适合传递大型对象或需要修改实参的场景
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/10 10:06:36

快速掌握 eSpeak NG:面向新手的终极语音合成指南

快速掌握 eSpeak NG&#xff1a;面向新手的终极语音合成指南 【免费下载链接】espeak-ng espeak-ng: 是一个文本到语音的合成器&#xff0c;支持多种语言和口音&#xff0c;适用于Linux、Windows、Android等操作系统。 项目地址: https://gitcode.com/GitHub_Trending/es/esp…

作者头像 李华
网站建设 2026/6/10 10:13:04

“每单多收3欧元”的时代来了:跨境卖家如何把低价小包从“风险”改造成“可控利润”

很多跨境团队这两年吃到的红利&#xff0c;本质来自一件事&#xff1a;低价值小包的成本结构不透明——关税、清关、平台履约约束都相对“松”。但从 2025 年底到 2026 年初&#xff0c;欧盟与平台规则给出了非常清晰的信号&#xff1a;低价小包要被系统性“重新定价”。这篇文…

作者头像 李华
网站建设 2026/6/10 10:12:02

Open-AutoGLM部署教程:云服务vLLM启动参数匹配要点

Open-AutoGLM部署教程&#xff1a;云服务vLLM启动参数匹配要点 1. Open-AutoGLM – 智谱开源的手机端AI Agent框架 你有没有想过&#xff0c;让AI帮你操作手机&#xff1f;不是简单的语音助手&#xff0c;而是真正“看懂”屏幕、理解界面、自动点击滑动&#xff0c;像真人一样…

作者头像 李华
网站建设 2026/6/10 10:03:07

OOTDiffusion:重新定义虚拟试衣的AI技术革命

OOTDiffusion&#xff1a;重新定义虚拟试衣的AI技术革命 【免费下载链接】OOTDiffusion 项目地址: https://gitcode.com/GitHub_Trending/oo/OOTDiffusion 还在为网购"买家秀"与"卖家秀"的巨大差距而烦恼吗&#xff1f;OOTDiffusion基于前沿的潜在…

作者头像 李华
网站建设 2026/6/10 10:08:26

Qwen1.5-0.5B能耗测试:低功耗设备运行可行性验证

Qwen1.5-0.5B能耗测试&#xff1a;低功耗设备运行可行性验证 1. 背景与目标&#xff1a;为什么关注小模型的能效表现&#xff1f; 在AI模型不断“变大”的趋势下&#xff0c;我们常常忽略了真实世界中大量场景对低功耗、低成本、可离线运行的需求。从工业边缘网关到便携式语音…

作者头像 李华
网站建设 2026/6/9 21:33:15

亲测verl实战效果:从安装到生产部署全流程分享

亲测verl实战效果&#xff1a;从安装到生产部署全流程分享 1. verl框架核心特性与技术优势 verl 是由字节跳动火山引擎团队开源的强化学习&#xff08;RL&#xff09;训练框架&#xff0c;专为大型语言模型&#xff08;LLMs&#xff09;的后训练阶段设计。作为 HybridFlow 论…

作者头像 李华