我正在遵循另一篇文章中的示例从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;
}
}
Run Code Online (Sandbox Code Playgroud)
编辑:使用此代码复制。
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})
Run Code Online (Sandbox Code Playgroud)
主程序
#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;
}
Run Code Online (Sandbox Code Playgroud)
Visual Studio 12. 在某个目录中调用这些。我得到“无法获得控制台模式 6”,这意味着 GetLastError() 返回 6。
mkdir build
cmake ..
msbuild.exe tester.sln
./Debug/tester.exe
Run Code Online (Sandbox Code Playgroud)
编辑:这似乎只有在从命令行运行时才会失败。在 Visual Studio 调试器的 Visual Studio 中,它成功了。仍然没有足够的视觉工作室知识来确定发生了什么。我只想制作一个可以从命令行运行并隐藏其输入的简单 exe。
编辑:清洁度并在通话后尽快获得 GetLastError。还是没有变化。
此错误 6 表示“无效句柄”,表示您传递给的句柄GetConsoleMode
不是控制台句柄。即 STDIN 已从某处重定向。
为了确保您拥有当前控制台的句柄,无论 STDIN 是否从其他地方重定向,请CreateFile
与特殊名称一起使用"CONIN$"
,它会打开控制台(如果存在)。
例子:
HANDLE hConsole = CreateFile("CONIN$",
GENERIC_READ, FILE_SHARE_READ|FILE_SHARE_WRITE,
NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
Run Code Online (Sandbox Code Playgroud)
归档时间: |
|
查看次数: |
3531 次 |
最近记录: |