C2440 static_cast 无法从基类转换为派生类

Fre*_*red 2 c++ polymorphism casting static-cast

我试图理解为什么使用指针从基类到派生类的转换编译得很好,但使用非指针对象进行转换会产生错误 C2440。

ThreadedMessage下面我有一个由 class 继承的基类GPSMessage

struct ThreadedMessage
{
  ThreadedMessage()
    : m_Type(0), m_ID(0)
  { }

  ThreadedMessage(uint Type, uint ID = 0) :
    m_Type(Type), m_ID(ID)
  { }

  uint m_Type;
  uint m_ID;
};    

struct GPSMessage : public ThreadedMessage
{
  GPSMessage()
    : ThreadedMessage()
  { }

  GPSMessage(double lat, double lon)
    : ThreadedMessage(1), m_lat(lat), m_lon(lon)
  { }

  double m_lat;
  double m_lon;
};
Run Code Online (Sandbox Code Playgroud)

我试图myFunction从基类转换为派生类。

void myFunction(const ThreadedMessage& msg)
{
  const GPSMessage* message = static_cast<const GPSMessage*>(&msg); // Compiles fine
  const GPSMessage message1 = static_cast<const GPSMessage>(msg);   // Error C2440
}
Run Code Online (Sandbox Code Playgroud)

调用myFunction()看起来像这样:

GPSMessage msg(21.123, 12.321);
myFunction(msg);
Run Code Online (Sandbox Code Playgroud)

当我编译时,指针的转换编译正常,但非指针转换失败并出现以下错误:

错误 C2440:“static_cast”:无法从“const ThreadedMessage”转换为“const GPSMessage”没有构造函数可以采用源类型,或者构造函数重载解析不明确

为什么我无法使用非指针变量从基类转换为派生类?

编译器是MS VS 2008 C++。

是的,我看过其他类似的问题,但我读过的问题似乎没有回答我的问题。

gez*_*eza 6

这两个演员有不同的含义。

第一个演员阵容:

const GPSMessage* message = static_cast<const GPSMessage*>(&msg); // Compiles fine
Run Code Online (Sandbox Code Playgroud)

这意味着它msg实际上是一个GPSMessage(或派生的)对象。您要求编译器msgGPSMessage. 不会创建新对象,message将指向msg. 如果msg实际上不是GPSMessage(或派生),则此转换具有未定义的行为。

顺便说一句,以下强制转换具有相同的含义(强制转换为引用):

const GPSMessage& message = static_cast<const GPSMessage&>(msg); // same meaning, which results in a reference instead of a pointer
Run Code Online (Sandbox Code Playgroud)

关于第二部演员阵容:

const GPSMessage message1 = static_cast<const GPSMessage>(msg);   // Error C2440
Run Code Online (Sandbox Code Playgroud)

这意味着,您从 .创建一个对象。您应该提供一种方法来使这种转换成为可能。例如,您应该创建一个具有参数的构造函数:message1msgGPSMessageThreadedMessage

struct GPSMessage {
    GPSMessage(const ThreadedMessage &); // needed constructor
};
Run Code Online (Sandbox Code Playgroud)

ThreadedMessage或者为to创建一个转换运算符GPSMessage

struct ThreadedMessage {
    operator GPSMessage(); // conversion operator
};
Run Code Online (Sandbox Code Playgroud)