protected成员访问权限取决于访问者身份与上下文:类内部可自由访问所有protected成员;派生类内部可访问本类对象及同类型对象的protected成员,但不可访问基类对象或其他派生类对象的;类外部一律不可访问,友元函数除外。

protected 成员在继承中既不是完全公开,也不是彻底私有——它的访问权限取决于“谁在访问”和“在哪里访问”。 理解它不能只看声明位置,而要看访问发生的上下文:是类内部?派生类内部?还是类外通过对象?三者规则完全不同。
类内部:可以自由访问所有 protected 成员
无论该 protected 成员来自当前类、基类,还是更上层的祖先类,只要在类定义内部(包括成员函数、友元函数体内),都能直接访问。
- 比如 Base 有个 protected int x;,Derived 继承 Base,那么 Derived::func() 可以直接写 x = 10;
- 即使 x 是 Base 的 protected 成员,Derived 的成员函数也视为“自己人”,无需限定作用域(如 Base::x)也能用
派生类内部:能访问本对象或同类型对象的 protected 成员,但不能访问其他基类对象的
关键点在于“对象所属的静态类型”。派生类成员函数可以访问:
– 当前对象(this)的基类部分中的 protected 成员
– 同为该派生类类型的其他对象的 protected 成员(包括其基类部分)
但不能访问一个纯基类对象、或其他派生类对象的 protected 成员。
- ✅ void Derived::f(Derived& d) { d.x = 5; } —— 合法,d 静态类型是 Derived
- ❌ void Derived::f(Base& b) { b.x = 5; } —— 编译错误,b 静态类型是 Base,哪怕运行时它真是 Derived 对象也不行
类外部(包括 main 或非成员函数):一律不可通过对象访问 protected 成员
不管你是用基类对象、派生类对象,还是指针/引用,只要不在类内部或派生类内部,就无法直接访问任何 protected 成员。
立即学习“C++免费学习笔记(深入)”;
- ❌ Derived d; d.x = 10; —— 错误,x 对外界不可见
- ❌ Base b; b.x = 10; —— 同样错误,即使 x 在 Base 中声明为 protected
- ⚠️ 友元函数例外:如果某函数被声明为 Base 或 Derived 的友元,它就能像类内代码一样访问对应类的 protected 成员
本质上,protected 是为“继承体系内的协作”设计的权限,不是为“外部使用”开放的。它守住的是封装边界,同时给子类留出扩展空间。不复杂但容易忽略细节。