c++ 派生类向基类转换的可访问性问题

c++ 派生类向基类转换的可访问性问题
C++ primer第五版有这样的描述和我实际编码的例子有点冲突,求教

c++ primer(第五版)描述:
假定D继承自B:

1. 如果当D共有继承B时, 用户代码才能使用派生类向基类的转换; 如果D继承B的方式是受保护的或者私有的则用户代码不能使用此转换。(这一点没有什么问题,关键是接下来两点和提示)
2. 不论D以什么方式继承B,D的成员函数和友员都能使用派生类向基类的的转换,派生类向其直接基类的转换对其成员和友员来说是永远可访问的。
3. 无论D继承B的方式是共有的或者保护的,则D的派生类的成员和友员可以使用D向B的类型转换; 反之, 如果D继承B的方式是私有的,则不能使用。
tip. 对于代码中的给定节点来说,如果基类的共有成员是可访问的,则派生类向基类的转化是可访问的,反之则不行

我的代码特别简单,就是想试试友员
class Base {
public:
virtual int get_prot() {return prot;}
protected:
int prot = 3;
};

class Prot_derived : protected Base {
friend void func(Base *b);
public:
virtual int get_prot() {return prot;}
};

void func(Base *b)
{
}

int main()
{
Prot_derived pd;
func(&pd);//error: ‘Base’ is an inaccessible base of ‘Prot_derived’
}


首先这个代码有点怪,
1. func不是基类的友员,所以不可以访问基类的保护成员和私有成员,但是在func应该可以访问基类的共有成员(tip)
2. func是派生类的友员,可是转换不允许, 到底哪里出错,我没有找到, (描述第二点)

在此多谢大家帮忙

3个回答

如果要使你的代码成立,应使用public继承,你的第一点就是正解的,以protected继承的话,是不能转向父类的,这是你的第二点。

jiasuwoniu
jiasuwoniu 我大概知道原因了,c++ primer 里的描述是对的, 至于不能转换,不是因为友员的问题, 其实真是的转换发生在
2 年多之前 回复

我大概知道原因了,c++ primer 里的描述是对的, 至于不能转换,不是因为友员的问题, 其实真实的转换发生在mian函数里,而不是发生在
func里, 而派生类又是保护继承,在main中将&pd付给 Base*, 是不可以的,如果想要成功,需要把main函数变成Prot_derived的友员

但是还有一点不太理解的地方,c++ primer tip 的描述
tip : 对于代码中的给定节点来说,如果基类的共有成员是可访问的,则派生类向基类的转化是可访问的,反之则不行

如例子,main里 func&pd), pd 不能访问基类的共有成员,所以转换不能发生, 但是即使是共有继承pd也访问不到基类的共有成员(只能访问到
内嵌在派生类的基类共有成员), 转换按描述也不应该发生,
所以我觉得tip应该这样描述好像更加合理

tip: 对于代码中的给定节点来说,如果基类内嵌到派生类的共有成员是可访问的,则派生类向基类的转化是可访问的,反之则不行

不知道我的理解对不对

楼主的代码跟友元函数似乎没什么关系,假设b继承与a,那么如果需要从main或c中访问时,
当通过一个子类对象b访问他从基类a继承过来的成员时,是需要考虑b从a的继承方式的。

jiasuwoniu
jiasuwoniu 我想问得问题是不同的继承方式,派生类向基类的转化问题,上面的代码确实有些问题,我在上面的评论中已经说了
2 年多之前 回复
Csdn user default icon
上传中...
上传图片
插入图片
抄袭、复制答案,以达到刷声望分或其他目的的行为,在CSDN问答是严格禁止的,一经发现立刻封号。是时候展现真正的技术了!