VS2017 MFC无法向控制台输出消息

时间:2018-04-16 07:23:11

标签: c++ mfc console visual-studio-2017

我创建了一个基于MFC对话框的应用程序。在VS2013中,我可以创建一个控制台窗口和输出消息。当我升级到VS2017时,执行相同的代码,创建控制台窗口但没有消息输出。以下是我的代码:

bool Initialize(void)
{
    HWND hWnd = GetConsoleWindow();
    if (NULL != hWnd)
    {
        return true;
    }

    if (!AllocConsole())
    {
        return false;
    }

    HANDLE m_hStdOut = GetStdHandle(STD_OUTPUT_HANDLE);
    if (INVALID_HANDLE_VALUE == m_hStdOut)
    {
        return false;
    }

    int m_hCrt = _open_osfhandle((intptr_t)m_hStdOut, _O_TEXT);
    if (-1 == m_hCrt)
    {
        return false;
    }

    FILE* m_pCrtFile = _fdopen(m_hCrt, "w");
    *stdout = *m_pCrtFile;
    int ret = setvbuf(stdout, NULL, _IONBF, 0);
    if (-1 == ret)
    {
        return false;
    }
    return true;
}

void WriteLine(LPCTSTR lpszText)
{
    Initialize();
    std::wcout << lpszText;
    std::wcout << std::endl;
    std::wcout.flush();
    system("pause");
}

BOOL CMFCApplication1App::InitInstance()
{
    WriteLine(_T("test"));
    ...
}

问题: 如何在VS2017中修改我的代码以将消息输出到控制台?

1 个答案:

答案 0 :(得分:3)

致电freopen(或freopen_s以避免安全警告)重新开启stdout。同时添加_dup2以将stdout与控制台的文件描述符关联。

需要

_setmode(_fileno(stdout), _O_U16TEXT)打印Unicode(std::cout将不可用,除非您将模式设置回_O_TEXT)。

另见Redirecting cout

bool Initialize()
{
    if(GetConsoleWindow())
        return true;

    if(!AllocConsole())
        return false;

    HANDLE m_hStdOut = GetStdHandle(STD_OUTPUT_HANDLE);
    if(m_hStdOut == INVALID_HANDLE_VALUE)
        return false;

    //get file descriptor from handle
    int m_hCrt = _open_osfhandle((intptr_t)m_hStdOut, _O_TEXT);
    if(m_hCrt == -1)
        return false;

    FILE* m_pCrtFile = _fdopen(m_hCrt, "w");
    if(!m_pCrtFile)
        return false;

    FILE* notused;
    freopen_s(&notused, "CONOUT$", "w", stdout);

    //associate m_pCrtFile with `stdout`
    if(_dup2(_fileno(m_pCrtFile), _fileno(stdout)) != 0)
        return false;

    setvbuf(stdout, NULL, _IONBF, 0);

    _setmode(_fileno(stdout), _O_U16TEXT);
    std::wcout << L"123 ελληνικά Иванчо English\n";
    return true;
}