当前位置: 首页 > news >正文

C++ Primer 访问控制与封装

欢迎阅读我的 【C++Primer】专栏

专栏简介:本专栏主要面向C++初学者,解释C++的一些基本概念和基础语言特性,涉及C++标准库的用法,面向对象特性,泛型特性高级用法。通过使用标准库中定义的抽象设施,使你更加适应高级程序设计技术。希望对读者有帮助!

在这里插入图片描述
在这里插入图片描述

目录

  • 访问控制与封装
    • 使用class或struct关键字
    • 友元
    • 友元的声明

访问控制与封装

到目前为止,我们已经为类定义了接口,但并没有任何机制强制用户使用这些接口。我们的类还没有封装,也就是说,用户可以直达Sales_data对象的内部并且控制它的具体实现细节。在C++语言中,我们使用访问说明符(access specifiers)加强类的封装性:

  • 定义在public说明符之后的成员在整个程序内可被访问,public成员定义类的接口。
  • 定义在private说明符之后的成员可以被类的成员函数访问,但是不能被使用该类的代码访问,private部分封装了(即隐藏了)类的实现细节。

再一次定义Sales_data类,其新形式如下所示:

class Sales_data{
public://添加了访问说明符
    Sales_data()=default;
    Sales & _data(const std::string&s,unsigned n,double p):
    bookNo(s),units_sold(n),revenue(p*n){}
    Sales_data(conststd::string&s):bookNo(s){}
    Sales_data(std::istream&);
    std::string isbn() const{ return bookNo;}
    Sales_data&combine(const Sales_data&);
private://添加了访问说明符
    double avg_price() const
    {
        return units_sold?revenue/units_sold:0;
    }
    std::string bookNo;
    unsigned units_sold=0;
    double revenue=0.0;
}

作为接口的一部分,构造函数和部分成员函数(即isbn和combine)紧跟在public说明符之后;而数据成员和作为实现部分的函数则跟在private说明符后面。

一个类可以包含0个或多个访问说明符,而且对于每个访问说明符能出现多少次也没有严格限定。每个访问说明符指定了接下来的成员的访问级别,其有效范围直到出现下一个访问说明符或者到达类的结尾处为止。

使用class或struct关键字

在上面的定义中我们迦做了一个微妙的变化,我们使用了class关键字而非struct开始类的定义。这种变化仅仅是形式上有所不同,实际上我们可以使用这两个关键字中的任何一个定义类。唯一的一点区别是,struct和class的默认访问权限不太一样。

类可以在它的第一个访问说明符之前定义成员,对这种成员的访问权限依赖于类定义的方式。如果我们使用struct关键字,则定义在第一个访问说明符之前的成员是public的;相反,如果我们使用class关键字,则这些成员是private的。

出于统一编程风格的考虑,当我们希望定义的类的所有成员是public的时,使用struct;反之,如果希望成员是private的,使用class。

使用class和struct定义类唯一的区别就是默认的访问权限

友元

既然Sales_data的数据成员是private的,我们的read、print和add函数也就无法正常编译了,这是因为尽管这几个函数是类的接口的一部分,但它们不是类的成员。

类可以允许其他类或者函数访问它的非公有成员,方法是令其他类或者函数成为它的友元(friend)。如果类想把一个函数作为它的友元,只需要增加一条以friend关键守开始的函数声明语句即可:

class Sales_data{
//为Sales_data的非成员凶数所做的友元声明
friend Sales_data add(const Sales_data&,const Sales_Data&);
friend std::istream&read(std::istream&,Sales_data&);
friend std::ostream&print(std::ostream&,const_Sales_data&);
//其他成员及访问说明符与之前一致
public:
Sales_data()=default;
Sales_data(const std::string&s,unsigned n,double p):
bookNo(s),units_sold(n),revenue(p*n){}
Sales_data(const std::string&s):bookNo(s){}
Sales_data(std::istream&);
std::string isbn()const{return bookNo;】
Sales_data&combine(const Sales_data&);
private:
std::string bookNo;
unsigned units_sold=0;
double revenue=0.0;
// Sales_data接口的非成员组成部分的声明
Sales_data add(const Sales_data&,const Sales_data&);
std::stream& read(std::istream& ,Sales_data&)
std::ostream& print(std::ostream& ,const Sales_data&);

友元声明只能出现在类定义的内部,但是在类内出现的具体位置不限。友元不是类的成员也不受它所在区域访问控制级别的约柬。

一般来说,最好在类定义开始或结束前的位置集中声明友元。


关键概念:封装的益处

封装有两个重要的优点:

  • 确保用户代码不会无意间破坏封装对象的状态。
  • 被封装的类的具体实现细节可以随时改变,而无须调整用户级别的代码。

一旦把数据成员定义成private的,类的作者就可以比较自由地修改数据了。当实现部分改变时,我们只需要检查类的代码本身以确认这次改变有什么影响;换句话说,只要类的接口不变,用户代码就无须改变。如果数据是public的,则所有使用了原来据成员的代码都可能失效,这时我们必须定位并重写所有依赖于老版本实现的代码,之后才能重新使用该程序。

把数据成员的访问权限设成prtvate还有另外一个威处,这么做能防止由于用户的厉因造成数据被破坏。如果我们发现有程序缺陷破坏了对象的状态,则可以在有限的范图内定位缺陷:因为只有实现部分的代码可能产生这样的错误。因此,将查错限制在有限范图内将能极大地降低维护代码及修正程序错误的难度。

尽管当类的定义发生改变时无须更改用户代码,但是使用了该类的源文件引须重新编译。

友元的声明

友元的声明仅仅指定了访问的权限,而非一个通常意义上的函数声明。如果我们希望类的用户能够调用某个友元函数,那么我们就必须在友元声明之外再专门对函数进行一次声明。

为了使友元对类的用户可见,我们通常把友元的声明与类本身放置在同一个头文件中类的外部。因此,我们的sales_data头文件应该为read、print和add提供独立的声明(除了类内部的友元声明之外)。

许多编译器并未强制限定友元函数必须在使用之前在类的外部声明。

一些编译器允许在尚无友元函数的初始声明的情况下就调用它。不过即使你的编译器支持这种行为,最好还是提供一个独立的函数声明。这样即使你更换了一个有这种强制要求的编译器,也不必改变代码。

相关文章:

  • 防御保护第三次练习
  • 【大学生职业规划大赛备赛PPT资料PDF | 免费共享】
  • 论文阅读 DOES END-TO-END AUTONOMOUS DRIVING REALLY NEED PERCEPTION TASKS?
  • 还在为AI模型部署发愁?VSCode插件让你轻松拥有DeepSeek和近百种AI模型!
  • sql注入中,如果information_schema被过滤,该怎么绕过
  • 【Linux系统】—— 调试器 gdb/cgdb的使用
  • 腾讯云大模型知识引擎×DeepSeek赋能文旅
  • 筛选相同项
  • 深入解析TCP/IP协议:从理论到实践的全链路剖析
  • Pytorch论文实现之GAN-C约束鉴别器训练自己的数据集
  • 机器学习的数学基础(三)——概率与信息论
  • SP字体UI放大代码
  • ComfyUI的安装
  • 基于Electron+Vue3创建桌面应用
  • C语言.h头文件的写法
  • 物联网技术赋能预测性维护的深度剖析与前景展望
  • 华为FusionCompute虚拟化平台
  • MacOS 15.3 卸载系统内置软件
  • 微信小程序---计划时钟设计与实现
  • 网络安全三件套
  • 国务院安委会办公室印发通知:坚决防范遏制重特大事故发生
  • 刘诚宇、杨皓宇进球背后,是申花本土球员带着外援踢的无奈
  • 媒体:西安62岁男子当街殴打妻子,警方称打人者已被行拘
  • “五一”假期预计全社会跨区域人员流动累计14.67亿人次
  • 视觉周刊|劳动开创未来
  • 海港通报颜骏凌伤停两至三周,国足面临门将伤病危机