C 和 C++ 中指针是核心特性,但两者的惯用法(idioms)有显著差异:C 语言依赖指针实现动态内存、数据结构、函数抽象;C++ 则通过智能指针、引用、RAII等机制减少裸指针使用,提升安全。
下面系统梳理 C/C++ 指针的最常用、最实用的惯用方法,涵盖安全使用、性能优化、现代替代方案等。
一、C 语言指针惯用法
1. 指针作为输出参数(模拟"引用")
cpp
// 返回多个值
int parse_int(const char* str, int* out_value) {
if (!str || !out_value) return -1; // 防御性检查
*out_value = atoi(str);
return 0; // success
}
// 使用
int val;
if (parse_int("123", &val) == 0) {
printf("Parsed: %d\n", val);
}
✔️ C 无引用,用
T*实现"写回"语义。
2. 指针遍历数组(高效且惯用)
cpp
void print_array(int* arr, size_t n) {
for (int* p = arr; p != arr + n; ++p) {
printf("%d ", *p);
}
}
✔️ 比下标更贴近硬件,常用于嵌入式/高性能代码。
3. 函数指针实现回调与多态
cpp
typedef void (*Logger)(const char*);
void log_info(const char* msg) { printf("[INFO] %s\n", msg); }
void log_error(const char* msg) { printf("[ERROR] %s\n", msg); }
void process(Logger log_fn) {
log_fn("Processing...");
}
// 使用
process(log_info);
✔️ Linux 内核、qsort、事件系统广泛使用。
二、C++ 指针惯用法(现代 C++)
核心原则:尽量避免裸指针(raw pointer),优先使用引用、智能指针、容器。
1. 用引用替代输出参数(更安全)
cpp
bool parse_int(std::string_view str, int& out_value) {
try {
out_value = std::stoi(std::string(str));
return true;
} catch (...) {
return false;
}
}
// 使用
int val;
if (parse_int("123", val)) { /* ... */ }
✔️ 引用不可为空,语义更清晰。
2. 智能指针管理动态内存(RAII)
std::unique_ptr(独占所有权)
cpp
auto ptr = std::make_unique<int>(42);
// 自动析构,无需手动 delete
std::shared_ptr(共享所有权)
cpp
auto sp1 = std::make_shared<MyClass>();
auto sp2 = sp1; // 引用计数 +1
惯用法 :优先使用
make_unique/make_shared,避免裸new。
3. 裸指针仅用于"非拥有式观察"
cpp
class Observer {
public:
// 接收指针,但不负责 delete
void watch(const Widget* widget) {
if (widget) {
current_ = widget; // 仅观察
}
}
private:
const Widget* current_ = nullptr; // 观察者指针
};
4. 成员函数返回 this 指针(流式接口)
cpp
class Builder {
std::string data_;
public:
Builder& add(const std::string& s) {
data_ += s;
return *this; // 返回引用(本质是 this 解引用)
}
};
// 使用
Builder b;
b.add("Hello").add(" World");
5. "Pimpl" 惯用法(Pointer to Implementation)
cpp
// widget.h
class Widget {
public:
Widget();
~Widget();
void doSomething();
private:
class Impl;
std::unique_ptr<Impl> pImpl;
};
cpp
// widget.cpp
class Widget::Impl {
public:
int data;
void helper() { /* ... */ }
};
Widget::Widget() : pImpl(std::make_unique<Impl>()) {}
void Widget::doSomething() { pImpl->helper(); }
优点:隐藏实现细节、减少编译依赖、提升 ABI 稳定性。