c++ - 意外地能够从基类ctor调用派生类虚函数

标签 c++ c++11 inheritance pure-virtual stdthread

任何人都可以帮助解释这种意外行为吗?

前提

我创建了包含成员 std::thread 变量的 Thread 类。 Thread 的构造函数构造成员 std::thread,提供指向调用纯虚函数(由基类实现)的静态函数的指针。

代码

#include <iostream>
#include <thread>
#include <chrono>

namespace
{

class Thread
{
public:
    Thread()
        : mThread(ThreadStart, this)
    {
        std::cout << __PRETTY_FUNCTION__ << std::endl; // This line commented later in the question.
    }

    virtual ~Thread() { }

    static void ThreadStart(void* pObj)
    {
        ((Thread*)pObj)->Run();
    }

    void join()
    {
        mThread.join();
    }

    virtual void Run() = 0;

protected:
    std::thread mThread;
};

class Verbose
{
public:
    Verbose(int i) { std::cout << __PRETTY_FUNCTION__ << ": " << i << std::endl; }
    ~Verbose() { }
};

class A : public Thread
{
public:
    A(int i)
        : Thread()
        , mV(i)
    { }

    virtual ~A() { }

    virtual void Run()
    {
        for (unsigned i = 0; i < 5; ++i)
        {
            std::cout << __PRETTY_FUNCTION__ << ": " << i << std::endl;
            std::this_thread::sleep_for(std::chrono::seconds(1));
        }
    }

protected:
    Verbose mV;
};

}

int main(int argc, char* argv[])
{
    A a(42);
    a.join();

    return 0;
}

问题

您可能已经注意到,这里有一个微妙的错误:Thread::ThreadStart(...) 是从 Thread 构造函数上下文中调用的,因此调用了一个纯/虚函数不会调用派生类的实现。运行时错误证实了这一点:

pure virtual method called
terminate called without an active exception
Aborted

但是,如果我在 Thread 构造函数中删除对 std::cout 的调用,则会出现意外的运行时行为:

virtual void {anonymous}::A::Run(){anonymous}::Verbose::Verbose(int): : 042

virtual void {anonymous}::A::Run(): 1
virtual void {anonymous}::A::Run(): 2
virtual void {anonymous}::A::Run(): 3
virtual void {anonymous}::A::Run(): 4

即在 Thread 构造函数中删除对 std::cout 的调用似乎具有能够从基类构造函数调用派生类的纯/虚函数的效果语境!这与之前的学习和经验不符。

在 Windows 10 上的 Cygwin x64 中构建环境。gcc 版本为:

g++ (GCC) 5.4.0
Copyright (C) 2015 Free Software Foundation, Inc.
This is free software; see the source for copying conditions.  There is NO
warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.

我对这种观察感到困惑,并且对正在发生的事情充满好奇。任何人都可以阐明吗?

最佳答案

由于竞争条件,该程序的行为未定义。

但是,如果您想对此进行推理,让我们试试吧。

对于 A的构造,这里是发生了什么:

  1. mThread被初始化。操作系统安排它在未来的某个时间点开始。
  2. std::cout << __PRETTY_FUNCTION__ << std::endl; - 从程序的角度来看,这是一个相当缓慢的操作。

  3. A构造函数运行 - 初始化它的vtable(这不是标准强制要求的,但据我所知,所有实现都这样做)。

    如果这发生在 mThread 之前计划开始,您将获得您观察到的行为。否则,您将获得纯虚拟调用。

因为这些操作没有任何顺序,所以行为是未定义的。

您会注意到您从基类的构造函数中删除了一个相当慢的操作,从而更快地初始化您的派生 - 及其 vtable。说,在操作系统实际安排之前 mThread的线程开始。也就是说,这并没有解决问题,只是降低了遇到它的可能性。

如果稍微修改您的示例,您会注意到删除 IO 代码会使比赛更难找到,但没有解决任何问题。

virtual void Run()
{
    for (unsigned i = 0; i < 1; ++i)
    {
        std::cout << __PRETTY_FUNCTION__ << ": " << i << std::endl;
//      std::this_thread::sleep_for(std::chrono::seconds(1));
    }
}

主要内容:

for(int i = 0; i < 10000; ++i){
    A a(42);
    a.join();
}

demo

关于c++ - 意外地能够从基类ctor调用派生类虚函数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/40317551/

相关文章:

c++ - 为什么 N3421 不提供 noexcept 限定词?

c++ - C++中这两种转换方法有什么区别?

c++ - 为什么条件变量需要锁(因此也需要互斥锁)

C++ 在带有右值缓冲区的 ostream 中使用 snprintf,格式是否正确?

c++ - std::make_shared()、std::weak_ptr 和循环引用

php - PHP中静态类的不同实例

c++ - 与指向继承方法的模板化函数指针一起使用的模板化方法的问题

c++ - 指向字符数组数组的指针

c++ - 检测到堆损坏 | C++

python - 在 Python 中转换为其他类