C++ 类访问修饰符

数据封装是面向对象编程的一个重要特点,它防止函数直接访问类类型的内部成员。类成员的访问限制是通过在类主体内部对各个区域标记 public、private、protected 来指定的。关键字 public、private、protected 称为访问修饰符。一个类可以有多个 public、protected 或 private 标记区域。每个标记区域在下一个标记区域开始之前或者在遇到类主体结束右括号之前都是有效的。成员和类的默认访问修饰符是 private。

class Base {
public: // 公有成员 protected: // 受保护成员 private: // 私有成员
};

公有(public)成员
公有成员在程序中类的外部是可访问的。您可以不使用任何成员函数来设置和获取公有变量的值,如下所示:

#include <iostream>
using namespace std;
class Line
{
public:
double length;
void setLength(double len);
double getLength(void );
}; double Line::getLength(void)
{
return length ;
} void Line::setLength( double len )
{
length = len;
} int main( )
{
Line line;
line.setLength(6.0);
cout << "Length of line : " << line.getLength() <<endl;
line.length = 10.0;
cout << "Length of line : " << line.length <<endl;
return ;
}

编译并执行:

Length of line :
Length of line :

私有(private)成员

私有成员变量或函数在类的外部是不可访问的,甚至是不可查看的。只有类和友元函数可以访问私有成员。
默认情况下,类的所有成员都是私有的。例如在下面的类中,width 是一个私有成员,这意味着,如果您没有使用任何访问修饰符,类的成员将被假定为私有成员:

class Box
{
double width;
public:
double length;
void setWidth( double wid );
double getWidth( void );
};

实际操作中,我们一般会在私有区域定义数据,在公有区域定义相关的函数,以便在类的外部也可以调用这些函数,如下所示:

#include <iostream>
using namespace std; class Box
{
public:
double length;
void setWidth( double wid );
double getWidth( void ); private:
double width;
}; double Box::getWidth(void)
{
return width ;
} void Box::setWidth( double wid )
{
width = wid;
} int main()
{
Box box; box.length = 10.0; //length is public
cout << "Length of box : " << box.length <<endl; //box.width = 10.0; // Error: the width is private
box.setWidth(10.0); // member function to set the width
cout << "Width of box : " << box.getWidth() <<endl; return ;
}

编译并执行:

Length of box :
Width of box :

保护(protected)成员

保护成员变量或函数与私有成员十分相似,但有一点不同,保护成员在派生类(即子类)中是可访问的。
现在您可以看到下面的实例中,我们从父类 Box 派生了一个子类 smallBox。下面的实例与前面的实例类似,在这里 width 成员可被派生类 smallBox 的任何成员函数访问。

#include <iostream>
using namespace std; class Box
{
protected:
double width;
}; class SmallBox:Box // SmallBox 是派生类
{
public:
void setSmallWidth( double wid );
double getSmallWidth( void );
}; // 子类的成员函数
double SmallBox::getSmallWidth(void)
{
return width ;
} void SmallBox::setSmallWidth( double wid )
{
width = wid;
} int main( )
{
SmallBox box; // 使用成员函数设置宽度
box.setSmallWidth(5.0);
cout << "Width of box : "<< box.getSmallWidth() << endl; return ;
}

编译并执行:

Width of box : 

继承中的特点
有public, protected, private三种继承方式,它们相应地改变了基类成员的访问属性。
基类 public 成员,protected 成员,private 成员的访问属性在派生类中分别变成:public, protected, private
但无论哪种继承方式,下面两点都没有改变:

1. private 成员只能被本类成员(类内)和友元访问,不能被派生类访问;
2. protected 成员可以被派生类访问。

public继承

#include <iostream>
#include <assert.h>
using namespace std; class A{
public:
int a;
A() {
a1 = ;
a2 = ;
a3 = ;
a = ;
} void fun() {
cout << a << endl; //right
cout << a1 << endl; //right
cout << a2 << endl; //right
cout << a3 << endl; //right
}
public:
int a1;
protected:
int a2;
private:
int a3;
}; class B : public A{
public:
int a;
B(int i) {
A();
a = i;
}
void fun() {
cout << a << endl; //right, public成员
cout << a1 << endl; //right, 基类的public成员,在派生类中仍是public成员
cout << a2 << endl; //right,基类的protected成员,在派生类中仍是protected,可以被派生类访问
// cout << a3 << endl; //wrong,基类的private成员,不能被派生类访问
}
}; int main() {
B b();
cout << b.a << endl;
cout << b.a1 << endl; //right
// cout << b.a2 << endl; //wrong, 类外不能访问protected成员
// cout << b.a3 << endl; //wrong, 类外不能访问private成员
return ;
}

编译输出:

10
11

protected继承

#include <iostream>
#include <assert.h>
using namespace std;
class A{
public:
int a;
A(){
a1 = ;
a2 = ;
a3 = ;
a = ;
}
void fun(){
cout << a << endl; //正确
cout << a1 << endl; //正确
cout << a2 << endl; //正确
cout << a3 << endl; //正确
}
public:
int a1;
protected:
int a2;
private:
int a3;
};
class B : protected A{
public:
int a;
B(int i){
A();
a = i;
}
void fun(){
cout << a << endl; //正确,public成员。
cout << a1 << endl; //正确,基类的public成员,在派生类中变成了protected,可以被派生类访问。
cout << a2 << endl; //正确,基类的protected成员,在派生类中还是protected,可以被派生类访问。
// cout << a3 << endl; //错误,基类的private成员不能被派生类访问。
}
}; int main(){
B b();
cout << b.a << endl; //正确。public成员
//cout << b.a1 << endl; //错误,public成员不能在类外访问。
//cout << b.a2 << endl; //错误,protected成员不能在类外访问。
//cout << b.a3 << endl; //错误,private成员不能在类外访问。
return ;
}

编译输出:

10

private继承

#include <iostream>
#include <assert.h>
using namespace std;
class A{
public:
int a;
A(){
a1 = ;
a2 = ;
a3 = ;
a = ;
}
void fun(){
cout << a << endl; //正确
cout << a1 << endl; //正确
cout << a2 << endl; //正确
cout << a3 << endl; //正确
}
public:
int a1;
protected:
int a2;
private:
int a3;
};
class B : private A{
public:
int a;
B(int i){
A();
a = i;
}
void fun(){
cout << a << endl; //正确,public成员。
cout << a1 << endl; //正确,基类public成员,在派生类中变成了private,可以被派生类访问。
cout << a2 << endl; //正确,基类的protected成员,在派生类中变成了private,可以被派生类访问。
// cout << a3 << endl; //错误,基类的private成员不能被派生类访问。
}
};
int main(){
B b();
cout << b.a << endl; //正确。public成员
// cout << b.a1 << endl; //错误,private成员不能在类外访问。
// cout << b.a2 << endl; //错误, private成员不能在类外访问。
// cout << b.a3 << endl; //错误,private成员不能在类外访问。
return ;
}

编译输出:

10

注意:在类里面不写是什么类型,默认是 private 的。

include <iostream>
using namespace std;
class Line{
int a;
};
int main() {
Line line;
line.a = ;
cout<<line.a<<endl;
}

这个是会报错的,应该改成:

class Line{
public:
int a;
};

C++解析二的更多相关文章

  1. C#使用zxing,zbar,thoughtworkQRcode解析二维码,附源代码

    最近做项目需要解析二维码图片,找了一大圈,发现没有人去整理下开源的几个库案例,花了点时间 做了zxing,zbar和thoughtworkqrcode解析二维码案例,希望大家有帮助. zxing是谷歌 ...

  2. Java生成与解析二维码

    1.下载支持二维码的jar包qrcode.jar和qrcode_swetake.jar, 其中qrcode_swetake.jar用于生成二维码,rcode.jar用于解析二维码,jar包下载地址(免 ...

  3. java 生成和解析二维码

    public class QRCode { /** * 解析二维码(QRCode) * @param imgPath * @return */ public static String decoder ...

  4. java代码生成二维码以及解析二维码

    package com.test; import java.awt.Color; import java.awt.Graphics2D; import java.awt.image.BufferedI ...

  5. asp.net C#生成和解析二维码代码

    类库文件我们在文件最后面下载 [ThoughtWorks.QRCode.dll 就是类库] 使用时需要增加: using ThoughtWorks.QRCode.Codec;using Thought ...

  6. Fixflow引擎解析(二)(模型) - BPMN2.0读写

    Fixflow引擎解析(四)(模型) - 通过EMF扩展BPMN2.0元素 Fixflow引擎解析(三)(模型) - 创建EMF模型来读写XML文件 Fixflow引擎解析(二)(模型) - BPMN ...

  7. java二维码之利用谷歌的zxing生成二维码,解析二维码

    生成二维码 @RequestMapping("/123") public void test(HttpServletRequest request,HttpServletRespo ...

  8. ZXing 生成、解析二维码图片的小示例

    概述 ZXing 是一个开源 Java 类库用于解析多种格式的 1D/2D 条形码.目标是能够对QR编码.Data Matrix.UPC的1D条形码进行解码. 其提供了多种平台下的客户端包括:J2ME ...

  9. Java生成、解析二维码

    今天遇到需求,使用Java生成二维码图片,网搜之后,大神们早就做过,个人总结一下. 目标:借助Google提供的ZXing Core工具包,使用Java语言实现二维码的生成和解析. 步骤如下: 1.m ...

  10. (转)ZXing解析二维码

    1 ZXing解析二维码 上一篇文件已经说过如何用ZXing进行生成二维码和带图片的二维码,下面说下如何解析二维码 二维码的解析和生成类似,也可以参考google的一个操作类 BufferedImag ...

随机推荐

  1. C.【转】C语言字符串与数字相互转换

    1.gcvt 把浮点数转成字符串 - CSDN博客.html(https://blog.csdn.net/dxuehui/article/details/52791412) 1.1. 函数名: gcv ...

  2. .NET Core 管道

    从用户发请求到服务器响应返回数据 请求从 Request进去    先经过 Middleware(中间件) 然后经过AuthoriationFilters授权验证(token验证和 多租户验证) 在经 ...

  3. Eclipse+maven 构建第一个简单的springmvc项目

    先给出项目的目录: 在eclipse下使用maven构建第一个springmvc项目步骤如下: 1.创建maven project(此处默认你已了解maven),此处需要注意以下两点 2.创建完毕后会 ...

  4. 力扣(LeetCode)15. 三数之和

    给定一个包含 n 个整数的数组 nums,判断 nums 中是否存在三个元素 a,b,c ,使得 a + b + c = 0 ?找出所有满足条件且不重复的三元组. 注意:答案中不可以包含重复的三元组. ...

  5. 学习笔记27—python中numpy.ravel() 和 flatten()函数

    简介 首先声明两者所要实现的功能是一致的(将多维数组降位一维).这点从两个单词的意也可以看出来,ravel(散开,解开),flatten(变平).两者的区别在于返回拷贝(copy)还是返回视图(vie ...

  6. 常用命令-python篇

    1. pip 加速命令 pip install --index-url https://pypi.douban.com/simple pip install -i https://pypi.tuna. ...

  7. Codeforces 1009 F - Dominant Indices

    F - Dominant Indices 思路:树上启发式合并 先跑轻子树,然后清除轻子树的信息 最后跑重子树,不清除信息 然后再跑一遍轻子树,重新加回轻子树的信息 由于一个节点到根节点最多有logn ...

  8. java异常:java.lang.NullPointerException

    /** * 功能:空指针异常测试 */ /* Object[] parameters1=null; if(parameters1.length>0&&parameters1!=n ...

  9. 『TensorFlow』分布式训练_其二_单机多GPU并行&GPU模式设定

    建议比对『MXNet』第七弹_多GPU并行程序设计 一.tensorflow GPU设置 GPU指定占用 gpu_options = tf.GPUOptions(per_process_gpu_mem ...

  10. spring boot(十三)小技巧

    一些springboot小技巧.小知识点 初始化数据 我们在做测试的时候经常需要初始化导入一些数据,如何来处理呢?会有两种选择,一种是使用Jpa,另外一种是Spring JDBC.两种方式各有区别下面 ...