Effective C++ 2e Item19

部署运行你感兴趣的模型镜像

 

条款19: 分清成员函数,非成员函数和友元函数

成员函数和非成员函数最大的区别在于成员函数可以是虚拟的而非成员函数不行。所以,如果有个函数必须进行动态绑定(见条款38),就要采用虚拟函数,而虚拟函数必定是某个类的成员函数。关于这一点就这么简单。如果函数不必是虚拟的,情况就稍微复杂一点。

看下面表示有理数的一个类:

class Rational {
public:
  Rational(int numerator = 0, int denominator = 1);
  int numerator() const;
  int denominator() const;

private:
  ...
};

这是一个没有一点用处的类。(用条款18的术语来说,接口的确最小,但远不够完整。)所以,要对它增加加,减,乘等算术操作支持,但是,该用成员函数还是非成员函数,或者,非成员的友元函数来实现呢?

当拿不定主意的时候,用面向对象的方法来考虑!有理数的乘法是和Rational类相联系的,所以,写一个成员函数把这个操作包到类中。

class Rational {
public:

  ...

  const Rational operator*(const Rational& rhs) const;
};

(如果你不明白为什么这个函数以这种方式声明——返回一个const值而取一个const的引用作为它的参数——参考条款21-23。)

现在可以很容易地对有理数进行乘法操作:

Rational oneEighth(1, 8);
Rational oneHalf(1, 2);

Rational result = oneHalf * oneEighth;   // 运行良好

result = result * oneEighth;             // 运行良好

但不要满足,还要支持混合类型操作,比如,Rational要能和int相乘。但当写下下面的代码时,只有一半工作:

result = oneHalf * 2;      // 运行良好

result = 2 * oneHalf;      // 出错!

这是一个不好的苗头。记得吗?乘法要满足交换律。

如果用下面的等价函数形式重写上面的两个例子,问题的原因就很明显了:

result = oneHalf.operator*(2);      // 运行良好

result = 2.operator*(oneHalf);      // 出错!

对象oneHalf是一个包含operator*函数的类的实例,所以编译器调用了那个函数。而整数2没有相应的类,所以没有operator*成员函数。编译器还会去搜索一个可以象下面这样调用的非成员的operator*函数(即,在某个可见的名字空间里的operator*函数或全局的operator*函数):

result = operator*(2, oneHalf);      // 错误!

但没有这样一个参数为int和Rational的非成员operator*函数,所以搜索失败。

再看看那个成功的调用。它的第二参数是整数2,然而Rational::operator*期望的参数却是Rational对象。怎么回事?为什么2在一个地方可以工作而另一个地方不行?

秘密在于隐式类型转换。编译器知道传的值是int而函数需要的是Rational,但它也同时知道调用Rational的构造函数将int转换成一个合适的Rational,所以才有上面成功的调用(见条款M19)。换句话说,编译器处理这个调用时的情形类似下面这样:

const Rational temp(2);      // 从2产生一个临时
                             // Rational对象

result = oneHalf * temp;     // 同oneHalf.operator*(temp);

当然,只有所涉及的构造函数没有声明为explicit的情况下才会这样,因为explicit构造函数不能用于隐式转换,这正是explicit的含义。如果Rational象下面这样定义:

class Rational {
public:
  explicit Rational(int numerator = 0,     // 此构造函数为
                    int denominator = 1);  // explicit
  ...

  const Rational operator*(const Rational& rhs) const;

  ...

};

那么,下面的语句都不能通过编译:

result = oneHalf * 2;             // 错误!
result = 2 * oneHalf;             // 错误!

这不会为混合运算提供支持,但至少两条语句的行为一致了。

然而,我们刚才研究的这个类是要设计成可以允许固定类型到Rational的隐式转换的——这就是为什么Rational的构造函数没有声明为explicit的原因。这样,编译器将执行必要的隐式转换使上面result的第一个赋值语句通过编译。实际上,如果需要的话,编译器会对每个函数的每个参数执行这种隐式类型转换。但它只对函数参数表中列出的参数进行转换,决不会对成员函数所在的对象(即,成员函数中的*this指针所对应的对象)进行转换。这就是为什么这个语句可以工作:

result = oneHalf.operator*(2);      // converts int -> Rational

而这个语句不行:

result = 2.operator*(oneHalf);      // 不会转换
                                    // int -> Rational

第一种情形操作的是列在函数声明中的一个参数,而第二种情形不是。

尽管如此,你可能还是想支持混合型的算术操作,而实现的方法现在应该清楚了:使operator*成为一个非成员函数,从而允许编译器对所有的参数执行隐式类型转换:

class Rational {

  ...                               // contains no operator*

};

// 在全局或某一名字空间声明,
// 参见条款M20了解为什么要这么做
const Rational operator*(const Rational& lhs,
                         const Rational& rhs)
{
  return Rational(lhs.numerator() * rhs.numerator(),
                  lhs.denominator() * rhs.denominator());
}

Rational oneFourth(1, 4);
Rational result;

result = oneFourth * 2;           // 工作良好
result = 2 * oneFourth;           // 万岁, 它也工作了!

这当然是一个完美的结局,但还有一个担心:operator*应该成为Rational类的友元吗?

这种情况下,答案是不必要。因为operator*可以完全通过类的公有(public)接口来实现。上面的代码就是这么做的。只要能避免使用友元函数就要避免,因为,和现实生活中差不多,友元(朋友)带来的麻烦往往比它(他/她)对你的帮助多。

然而,很多情况下,不是成员的函数从概念上说也可能是类接口的一部分,它们需要访问类的非公有成员的情况也不少。

让我们回头再来看看本书那个主要的例子,String类。如果想重载operator>>和operator<<来读写String对象,你会很快发现它们不能是成员函数。如果是成员函数的话,调用它们时就必须把String对象放在它们的左边:

// 一个不正确地将operator>>和
// operator<<作为成员函数的类
class String {
public:
  String(const char *value);

  ...

  istream& operator>>(istream& input);
  ostream& operator<<(ostream& output);

private:
  char *data;
};

String s;

s >> cin;                   // 合法, 但
                            // 有违常规

s << cout;                  // 同上

这会把别人弄糊涂。所以这些函数不能是成员函数。注意这种情况和前面的不同。这里的目标是自然的调用语法,前面关心的是隐式类型转换。

所以,如果来设计这些函数,就象这样:

istream& operator>>(istream& input, String& string)
{
  delete [] string.data;

  read from input into some memory, and make string.data
  point to it

  return input;
}

ostream& operator<<(ostream& output,
                    const String& string)
{
  return output << string.data;
}

注意上面两个函数都要访问String类的data成员,而这个成员是私有(private)的。但我们已经知道,这个函数一定要是非成员函数。这样,就别无选择了:需要访问非公有成员的非成员函数只能是类的友元函数。

本条款得出的结论如下。假设f是想正确声明的函数,C是和它相关的类:

·虚函数必须是成员函数。如果f必须是虚函数,就让它成为C的成员函数。

·operator>>和operator<<决不能是成员函数。如果f是operator>>或operator<<,让f成为非成员函数。如果f还需要访问C的非公有成员,让f成为C的友元函数。

·只有非成员函数对最左边的参数进行类型转换。如果f需要对最左边的参数进行类型转换,让f成为非成员函数。如果f还需要访问C的非公有成员,让f成为C的友元函数。

·其它情况下都声明为成员函数。如果以上情况都不是,让f成为C的成员函数。

您可能感兴趣的与本文相关的镜像

Qwen-Image-Edit-2509

Qwen-Image-Edit-2509

图片编辑
Qwen

Qwen-Image-Edit-2509 是阿里巴巴通义千问团队于2025年9月发布的最新图像编辑AI模型,主要支持多图编辑,包括“人物+人物”、“人物+商品”等组合玩法

<think> 我们参考了用户提供的引用内容,其中提到了内存管理、多线程编程和STL等内容。用户已经学完C++基础语法、STL和多线程,接下来应该学习什么? 根据C++的学习路径,在掌握了这些基础之后,可以继续深入学习以下方向: 1. 深入模板元编程:C++的模板是强大且复杂的特性,学习模板元编程可以提高代码的抽象能力和复用性。 2. 现代C++特性(C++11/14/17/20):学习lambda表达式、智能指针、移动语义、并发库等现代特性。 3. 设计模式:掌握常用的设计模式,提高软件设计能力。 4. 网络编程:学习使用C++进行网络编程,如socket编程,以及网络库(如Boost.Asio)的使用。 5. 系统级编程:包括操作系统相关的API调用、进程管理、文件系统操作等。 6. 性能优化:学习如何分析和优化C++程序的性能,包括内存访问模式、缓存友好设计等。 7. 跨平台开发:了解如何在不同的操作系统(如Windows、Linux)上开发和移植C++程序。 8. 深入学习并发编程:虽然已经学习了多线程,但并发编程还有更多高级主题,如无锁编程、线程池、条件变量等。 我们根据这些方向,为用户推荐接下来的学习内容,并生成相关问题。 </think>在学完C++基础语法、STL和多线程后,建议按照以下路径继续深入学习,这些内容能帮助你构建更专业的C++开发能力: ### 1. 现代C++特性(C++11/14/17/20) ```cpp // 示例:现代C++核心特性 auto lambda = [](auto x) { // C++14泛型lambda return std::make_unique<int>(x*2); // 智能指针 }; std::vector<int> v{1,2,3}; // 初始化列表 for(auto&& item : v) { // 范围for循环 std::cout << item << std::endl; } ``` - **关键特性**:智能指针(`unique_ptr/shared_ptr`)、移动语义、lambda表达式、类型推导(`auto`)、并发库(`<future>`) - **学习资源**:《Effective Modern C++》(Scott Meyers)、cppreference.com ### 2. 内存管理进阶 ```cpp // 智能指针应用场景 class Resource { public: Resource() { std::cout << "资源创建\n"; } ~Resource() { std::cout << "资源释放\n"; } }; auto processResource() { auto res = std::make_shared<Resource>(); // 自动内存管理 return res; // 引用计数管理生命周期 } ``` - **重点内容**: - 智能指针的原理与使用场景 - 自定义内存分配器 - 内存泄漏检测工具(Valgrind、AddressSanitizer) - RAII(资源获取即初始化)设计模式 ### 3. 网络编程(结合多线程) ```cpp // 简易TCP服务器示例 void start_server() { boost::asio::io_context io; tcp::acceptor acceptor(io, tcp::endpoint(tcp::v4(), 8080)); auto handle_connection = [&] { tcp::socket socket(io); acceptor.accept(socket); std::thread(handle_client, std::move(socket)).detach(); }; } ``` - **推荐方向**: - Socket编程基础 - 网络库:Boost.Asio/Poco/libevent - HTTP/TCP协议实现 - 高并发服务器架构 ### 4. 设计模式与架构 ```cpp // 工厂模式示例 class Widget { public: virtual void draw() = 0; static std::unique_ptr<Widget> create(WidgetType type); // 工厂方法 }; ``` - **常用模式**: - 创建型:工厂/单例/建造者 - 结构型:适配器/代理/组合 - 行为型:观察者/策略/命令 - **学习建议**:《设计模式:可复用面向对象软件的基础》 ### 5. 跨平台开发 - **关键技术**: - CMake构建系统 - 平台特定API封装(Windows/Linux/macOS) - 条件编译与宏定义 - 跨平台GUI框架(Qt/WxWidgets) ### 6. 性能优化 ```cpp // 缓存友好设计示例 struct alignas(64) CacheLine { // 64字节对齐 int data[16]; }; ``` - **优化方向**: - CPU缓存机制 - 分支预测优化 - SIMD指令集(AVX/SSE) - 性能分析工具(perf、VTune) ### 7. 项目实践建议 ```mermaid graph LR A[基础语法] --> B[STL容器] B --> C[多线程] C --> D[网络编程] D --> E[设计模式] E --> F[性能优化] F --> G[完整项目] ``` - **项目类型**: - 简易数据库系统 - HTTP服务器 - 实时数据处理系统 - 游戏引擎模块
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值