Effective C++读书笔记——item23(用非成员,非友元函数取代成员函数)

一、主要观点

  • 在某些情况下,使用 non-membernon-friend 函数来替换 member 函数可以增强封装性和可扩展性,提供更好的软件设计。

二、详细解释

  1. 封装性

    • 类成员函数的封装性考量 :成员函数可以访问类的 private 成员,这在一定程度上破坏了封装性。因为它拥有比实际所需更多的对类内部的访问权限。例如,对于一个类 Widget 的成员函数,它可以访问类的所有 private 数据成员和函数,即使某些数据或函数并不需要被该成员函数操作。
    • non-member、non-friend 函数的优势 :non-member、non-friend 函数只能通过类的 public 接口来访问类,这意味着它们不能直接访问类的 private 成员。这种方式可以将类的内部实现细节更好地隐藏起来,增强封装性。例如,对于一个处理 Widget 类对象的 non-membernon-friend 函数,它只能通过 Widget 类的 public 接口进行操作,不能访问 Widget 类内部的 private 成员,这样类的内部实现可以更加自由地修改而不影响这个外部函数。
  2. 可扩展性

    • 对类功能的扩展更灵活 :使用 non-member、non-friend 函数可以将不同的功能分布在不同的函数中,而不是将所有功能都集中在类的成员函数里。这样,当需要对类的功能进行扩展时,可以方便地添加新的 non-membernon-friend 函数,而不需要修改类的内部实现。
    • 命名空间的使用 :可以将相关的 non-member、non-friend 函数放在一个命名空间中,形成一个功能集合。这样可以更好地组织代码,并且可以根据不同的功能模块将函数分类到不同的命名空间中,方便代码的管理和维护。例如:
    cpp 复制代码
    namespace WidgetStuff {
        class Widget {... };
        void doSomething(Widget& w);
        void doAnotherThing(Widget& w);
    }

    这里,doSomethingdoAnotherThingnon-membernon-friend 函数,它们通过 Widgetpublic 接口操作 Widget 类,当需要添加新的操作 Widget 的函数时,只需在 WidgetStuff 命名空间中添加即可,而不用修改 Widget 类。

  3. 编译依赖关系

    • 减少依赖member 函数的修改通常会导致类的重新编译,因为它是类定义的一部分。而 non-membernon-friend 函数的修改通常只需要重新编译该函数本身,减少了对类的依赖。这在大型项目中可以提高编译速度。
    • 降低耦合non-membernon-friend 函数与类的耦合度相对较低,使得类的接口更加稳定,避免因修改函数而影响类的内部实现和其他使用该类的代码。
  4. 示例说明

    • 考虑一个表示网页浏览器的类 WebBrowser,可能有清除缓存、清除历史记录、清除 Cookies 等操作。可以有两种实现方式:
    cpp 复制代码
    class WebBrowser {
    public:
        void clearCache();
        void clearHistory();
        void clearCookies();
        void clearEverything() {  // member function
            clearCache();
            clearHistory();
            clearCookies();
        }
    };

    或者使用 non-membernon-friend 函数:

    cpp 复制代码
    class WebBrowser {
    public:
        void clearCache();
        void clearHistory();
        void clearCookies();
    };
    
    void clearEverything(WebBrowser& wb) {  // non-member, non-friend function
        wb.clearCache();
        wb.clearHistory();
        wb.clearCookies();
    }

    在这个例子中,使用 non-membernon-friend 函数 clearEverything 可以将清除操作的功能与 WebBrowser 类分离,避免 WebBrowser 类变得臃肿,同时也不会增加 WebBrowser 类的封装性负担,因为 clearEverything 函数无法访问 WebBrowser 类的 private 成员。

三、总结

  • 为了实现更好的封装性、可扩展性、降低编译依赖和降低耦合度,在某些情况下,应该优先考虑使用 non-membernon-friend 函数而不是 member 函数。这有助于将类的功能分离,提高代码的可维护性和组织性,同时使类的内部实现更加独立和灵活。
相关推荐
巨龙之路16 分钟前
C语言中的assert
c语言·开发语言
2301_776681651 小时前
【用「概率思维」重新理解生活】
开发语言·人工智能·自然语言处理
熊大如如2 小时前
Java 反射
java·开发语言
ll7788112 小时前
C++学习之路,从0到精通的征途:继承
开发语言·数据结构·c++·学习·算法
我不想当小卡拉米2 小时前
【Linux】操作系统入门:冯诺依曼体系结构
linux·开发语言·网络·c++
teacher伟大光荣且正确2 小时前
Qt Creator 配置 Android 编译环境
android·开发语言·qt
炎芯随笔2 小时前
【C++】【设计模式】生产者-消费者模型
开发语言·c++·设计模式
乌鸦9443 小时前
《类和对象(下)》
开发语言·c++·类和对象+
炒空心菜菜3 小时前
SparkSQL 连接 MySQL 并添加新数据:实战指南
大数据·开发语言·数据库·后端·mysql·spark
多多*3 小时前
算法竞赛相关 Java 二分模版
java·开发语言·数据结构·数据库·sql·算法·oracle