在C&C++指针的惯用方法

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 稳定性。

相关推荐
MM_MS7 分钟前
Halcon变量控制类型、数据类型转换、字符串格式化、元组操作
开发语言·人工智能·深度学习·算法·目标检测·计算机视觉·视觉检测
꧁Q༒ོγ꧂35 分钟前
LaTeX 语法入门指南
开发语言·latex
njsgcs41 分钟前
ue python二次开发启动教程+ 导入fbx到指定文件夹
开发语言·python·unreal engine·ue
alonewolf_9942 分钟前
JDK17新特性全面解析:从语法革新到模块化革命
java·开发语言·jvm·jdk
古城小栈1 小时前
Rust 迭代器产出的引用层数——分水岭
开发语言·rust
ghie90901 小时前
基于MATLAB的TLBO算法优化实现与改进
开发语言·算法·matlab
恋爱绝缘体11 小时前
2020重学C++重构你的C++知识体系
java·开发语言·c++·算法·junit
wuk9981 小时前
VSC优化算法MATLAB实现
开发语言·算法·matlab
AI小怪兽2 小时前
基于YOLOv13的汽车零件分割系统(Python源码+数据集+Pyside6界面)
开发语言·python·yolo·无人机
Z1Jxxx2 小时前
加密算法加密算法
开发语言·c++·算法