首先明确出现二义性这个问题的起因,当多个派生类继承于基类时,派生类中包含基类的对象及方法。这就意味着当有两个派生类继承于同一个基类时,两个派生类都会包含相同的基类副本。那么当有另一个派生类继承于这两个继承于同一基类的派生类时,该派生类中会有它继承的派生类的副本,然而这些副本都有相同的基类副本,从而导致数据冗余,并且存在二义性(最底层的派生类调用基类方法时,是调用其第一个父类的基类的方法还是第二个父类的基类的方法?)。
使用虚拟继承,则根据规则,基类的构造函数只能有最底层的派生类调用。那么这就导致,基类的派生类并不能成功调用基类的构造函数,除非它没有派生类。在菱形继承中,只有最底层的派生类才可以调用构造函数,那么可以理解为,当最底层的派生类构造对象前,先构造基类的构造函数,然后调用两个基类的派生类的构造函数,最后才调用最底层派生类的构造函数。这样就可以保证在最底层的派生类对象中只包含基类的一个副本,从而完美解决菱形继承问题。