函数里改不了原始变量是因为默认值传递,需传指针或引用;传指针时须用&取地址,解引用前要判空,避免野指针、空指针和悬垂指针。
如果调用函数后,原始变量的值没变,大概率是用了值传递而非地址传递。C++ 默认所有参数都是值传递——int x、std::string s、甚至 std::vector 都会拷贝一份副本进函数,改它不影响外面。想改原始变量,必须显式传地址:要么用指针(int*),要么用引用(int&)。指针是最基础、最可控的方式,尤其适合需要“可能为空”或“明确表达地址操作”的场景。
void foo(int* p) 怎么调用才真正修改原变量传指针本身仍是值传递(拷贝的是地址值),但这个“值”指向原始内存,所以能间接修改。关键在调用时传入变量的地址:
int a = 10; foo(&a); —— &a 是 a 的地
foo 内部通过 *p = 20; 就能改 a
foo(a); —— 编译报错:类型不匹配,int 不能隐转成 int*
foo(nullptr); —— 如果函数里没判空就解引用 *p,直接崩溃以 void func(int x, int* p) 为例:
x 是值传递:函数栈上新开一块内存存 x 的副本,sizeof(x) 是 4(32位 int),改 x 对调用方的原始变量零影响p 是指针值传递:函数栈上也开新内存存 p(地址值,通常 4 或 8 字节),但它指向的仍是原始变量的内存位置;*p = 100; 改的是那块原始内存std::vector<:string>)时,值传递要深拷贝,开销巨大;传指针只拷贝 8 字节地址,几乎无成本指针传递不是万能钥匙,三个高频陷阱:
int* get_ptr() { int local = 42; return &local; } —— 函数返回后 local 被销毁,拿到的是野指针,解引用行为未定义if (p != nullptr) *p = 5; 必须写,否则 foo(nullptr) 直接 crashdelete:int* p = new int(3); foo(p); delete p; —— foo 内部若再用 *p,就是访问已释放内存复杂点在于:指针本身不管理内存生命周期,谁分配、谁释放、谁负责判空,全靠程序员手动约定。稍有疏忽,调试起来比引用麻烦得多。