我正在关注另一个帖子Reading a password from std::cin中的示例,您可以获取标准输入的句柄,获取当前控制台模式,更改模式以抑制输入上的回显。出于某种原因,当我调用GetStdHandle()时,它返回一个有效的句柄,但是当我调用GetConsoleMode()时它失败并返回错误代码6.我在cmake项目中使用它。是否有任何调试标志我应该为此按预期工作?还有其他人遇到过吗?
void set_stdin_echo(bool enable) {
HANDLE hStdin = GetStdHandle(STD_INPUT_HANDLE);
if (hStdin == INVALID_HANDLE_VALUE) {
std::cout << "invalid handle " << GetLastError() << std::endl;
return;
}
else if (hStdin == NULL) {
std::cout << "no handle associated" << std::endl;
return;
}
DWORD mode = 0;
if (GetConsoleMode(hStdin, &mode) == 0) {
std::cout << "Could not get console mode" << GetLastError() << std::endl;
}
std::cout << mode << std::endl;
if (!enable)
mode &= ~ENABLE_ECHO_INPUT;
else
mode |= ENABLE_ECHO_INPUT;
std::cout << mode << std::endl;
if (SetConsoleMode(hStdin, mode) == 0) {
std::cout << "Could not set input mode" << std::endl;
std::cout << GetLastError() << std::endl;
}
}
编辑:使用此代码复制。
的CMakeLists.txt
cmake_minimum_required (VERSION 2.6)
project(tester)
if(CMAKE_COMPILER_IS_GNUCXX)
set(CMAKE_CXX_FLAGS "-g -w -Wall -pedantic-errors -std=c++11")
elseif(MSVC)
set(CMAKE_CXX_FLAGS "/EHsc")
set(CMAKE_CXX_FLAGS_DEBUG "/EHsc /MTd")
set(CMAKE_CXX_FLAGS_RELEASE "/EHsc /MT")
endif()
set(SOURCE_FILES "main.cpp")
add_executable(tester ${SOURCE_FILES})
的main.cpp
#include <iostream>
#include <string>
#include <windows.h>
void set_stdin_echo(bool enable) {
DWORD error;
HANDLE hStdin = GetStdHandle(STD_INPUT_HANDLE);
error = GetLastError();
if (hStdin == INVALID_HANDLE_VALUE) {
std::cout << "invalid handle " << error << std::endl;
return;
}
else if (hStdin == NULL) {
std::cout << "no handle associated" << std::endl;
return;
}
DWORD mode = 0;
if (GetConsoleMode(hStdin, &mode) == 0) {
error = GetLastError();
std::cout << "Could not get console mode" << error << std::endl;
}
std::cout << mode << std::endl;
if (!enable)
mode &= ~ENABLE_ECHO_INPUT;
else
mode |= ENABLE_ECHO_INPUT;
std::cout << mode << std::endl;
if (SetConsoleMode(hStdin, mode) == 0) {
error = GetLastError();
std::cout << "Could not set input mode" << std::endl;
std::cout << error << std::endl;
}
}
int main() {
std::string input;
std::cout << "password: ";
set_stdin_echo(false);
std::cin >> input;
set_stdin_echo(true);
return 0;
}
Visual Studio 12.在某个目录中调用它们。我得到“无法获得控制台模式6”,暗示GetLastError()返回6。
mkdir build
cmake ..
msbuild.exe tester.sln
./Debug/tester.exe
编辑:似乎只有在从命令行运行时才会失败。在visual studio调试器的visual studio中,它成功了。仍然没有足够的视觉工作室知识来确定发生了什么。我只想创建一个可以从命令行运行并隐藏其输入的简单exe。
编辑:清洁并在调用后asap获取GetLastError。仍然没有变化。
答案 0 :(得分:4)
此错误6表示“无效句柄”,表示您传递给GetConsoleMode
的句柄不是控制台句柄。即STDIN已从某处重定向。
为确保您拥有当前控制台的句柄,无论是否从其他位置重定向STDIN,请使用具有特殊名称CreateFile
的{{1}},这将打开控制台(如果存在)。< / p>
示例:
"CONIN$"