windows/linux 模拟鼠标键盘输入

windows 使用 SendInput (用户空间 API)

cpp 复制代码
#include <Windows.h>


// UINT SendInput(
//   UINT nInputs,             // 输入结构体数量
//   LPINPUT pInputs,          // 指向输入结构体数组的指针
//   int cbSize                // 每个结构体的大小(单位:字节)
// );



void SimulateMouseClick(int x, int y) {
    // 设置鼠标位置(绝对坐标)
    SetCursorPos(x, y);

    // 创建鼠标事件结构体
    INPUT input = {0}
    input.type = INPUT_MOUSE;
    input.mi.dwFlags = MOUSEEVENTF_LEFTDOWN;  // 鼠标按下
    SendInput(1, &input, sizeof(INPUT));

    Sleep(50);  // 等待 50ms

    input.mi.dwFlags = MOUSEEVENTF_LEFTUP;    // 鼠标释放
    SendInput(1, &input, sizeof(INPUT));
}

void SimulateKeyPress(WORD keyCode) {
    INPUT input = {0};
    input.type = INPUT_KEYBOARD;
    input.ki.wVk = keyCode;        // 虚拟键码(如 VK_RETURN)
    input.ki.dwFlags = 0;          // 按下
    SendInput(1, &input, sizeof(INPUT));

    Sleep(50);  // 等待 50ms

    input.ki.dwFlags = KEYEVENTF_KEYUP;  // 释放
    SendInput(1, &input, sizeof(INPUT));
}


// 模拟 Ctrl + C(复制)操作
void ControlC() {
    INPUT inputs[4] = {0};

    // Ctrl down
    inputs[0].type = INPUT_KEYBOARD;
    inputs[0].ki.wVk = VK_CONTROL;

    // C down
    inputs[1].type = INPUT_KEYBOARD;
    inputs[1].ki.wVk = 'C';

    // C up
    inputs[2].type = INPUT_KEYBOARD;
    inputs[2].ki.wVk = 'C';
    inputs[2].ki.dwFlags = KEYEVENTF_KEYUP;

    // Ctrl up
    inputs[3].type = INPUT_KEYBOARD;
    inputs[3].ki.wVk = VK_CONTROL;
    inputs[3].ki.dwFlags = KEYEVENTF_KEYUP;

    SendInput(4, inputs, sizeof(INPUT));

}


int main() {
    SimulateMouseClick(500, 300);
    Sleep(100);
    SimulateKeyPress(VK_RETURN);
    return 0;
}

linux 使用 X11 库 (用户空间 API)

cpp 复制代码
#include <X11/Xlib.h>
#include <X11/extensions/XTest.h>
#include <unistd.h>
#include <X11/keysym.h>

void SimulateKeyPress(Display* display, KeySym keysym) {
    KeyCode keycode = XKeysymToKeycode(display, keysym);
    XTestFakeKeyEvent(display, keycode, True, 0);   // True 按下
    XTestFakeKeyEvent(display, keycode, False, 0);  // False 释放
    XFlush(display);                                // 强制刷新事件队列,把事件立即发送给 X Server。
}

void SimulateMouseClick(Display* display, int button) {
    XTestFakeButtonEvent(display, button, True, 0);  // True 按下
    XTestFakeButtonEvent(display, button, False, 0); // False 释放
    XFlush(display);
}

int main() {
    Display* display = XOpenDisplay(nullptr);
    if (!display) return -1;

    SimulateMouseClick(display, 1);             // 1:鼠标左键,2:中键,3:右键
    sleep(1);
    SimulateKeyPress(display, XK_Return);       // 模拟回车键

    XCloseDisplay(display);
    return 0;
}

以上方式都是调用了用户空间的 API 来实现。

接下来介绍驱动层模拟的方式

输入子系统(Input Subsystem)结构

cpp 复制代码
[用户层] → /dev/uinput → 内核 input 子系统 → 事件发送至 X Server/Wayland
                                     ↓
                            应用程序收到输入
cpp 复制代码
[用户层] → DeviceIoControl → HID 驱动层 → 输入堆栈(Kbdclass/Mouclass) → Win32 消息

Linux 驱动层:使用 /dev/uinput 创建虚拟输入设备

Linux 提供了 uinput 接口,可在用户空间模拟底层输入事件,但从内核角度 看,是通过驱动层完成,在用户空间创建一个虚拟输入设备。通过 ioctl() 设置设备能力,然后通过写 input_event 结构体模拟键盘事件(比如按下 A 键),最后发送同步事件使其生效。这种方式运行在驱动层之上,绕过常规用户空间 API,常用于自动化测试或底层输入模拟。

/dev 文件夹的作用

  • /dev/tty:终端设备

  • /dev/console:系统控制台

  • /dev/input/eventX:键盘鼠标等输入设备

  • /dev/uinput:用户空间模拟输入设备

实现步骤:

  1. 打开 /dev/uinput

  2. 配置一个虚拟设备(设置支持的事件、按键)

  3. 写入 input_event 结构体发送事件

  4. 内核将其当作真实输入设备处理

伪代码如下:

cpp 复制代码
int fd = open("/dev/uinput", O_WRONLY | O_NONBLOCK);

// 设置设备支持按键
ioctl(fd, UI_SET_EVBIT, EV_KEY);
ioctl(fd, UI_SET_KEYBIT, KEY_A);

// 配置虚拟设备信息,在 /proc/bus/input/devices 或 evtest 工具中能看到这个"虚拟键盘"。
struct uinput_user_dev uidev = {};
strcpy(uidev.name, "virtual-keyboard");
write(fd, &uidev, sizeof(uidev));
ioctl(fd, UI_DEV_CREATE);

// 模拟按下 'A'
struct input_event ev = {};
ev.type = EV_KEY;    // 键盘事件
ev.code = KEY_A;     // 对应 'A' 键
ev.value = 1;        // 按下
write(fd, &ev, sizeof(ev));

// 释放 'A'
ev.value = 0;        // 0 = 松开
write(fd, &ev, sizeof(ev));

// 同步事件,告诉内核:"我一组输入事件已经发完了,请立即处理它们"。
ev.type = EV_SYN;
ev.code = SYN_REPORT;
ev.value = 0;
write(fd, &ev, sizeof(ev));

Windows 驱动层:写 HID 模拟驱动或使用驱动框架

在 Windows 中,驱动层模拟键盘鼠标通常通过以下方式实现:

方式 1:HID 模拟驱动

  • 编写一个虚拟 HID 设备(Human Interface Device)

  • 使用 KMDF/UMDF(内核/用户模式驱动框架)

  • 模拟输入数据包送入 HID Class 驱动堆栈

  • 类似于:软件伪造一个"插入的 USB 键盘"

相关推荐
melonbo2 分钟前
MHDD使用说明 - 硬盘检测
计算机外设
Geehy极海半导体4 天前
APM32芯得 EP.29 | 基于APM32F103的USB键盘与虚拟串口复合设备配置详解
计算机外设·usb·usb配置
人工干智能5 天前
pygame的帧处理中,涉及键盘的有`pg.event.get()`与`pg.key.get_pressed()` ,二者有什么区别与联系?
python·游戏·计算机外设·pygame
R-G-B5 天前
【P18 3-10】OpenCV Python—— 鼠标控制,鼠标回调函数(鼠标移动、按下、。。。),鼠标绘制基本图形(直线、圆、矩形)
python·opencv·计算机外设·鼠标回调函数·鼠标控制·鼠标移动·鼠标绘制图形
寻道模式5 天前
【运维心得】三步10分钟拆装笔记本键盘
运维·计算机外设·笔记本
OBOO鸥柏商用液晶显示厂家7 天前
OBOO鸥柏丨75寸/86平板企业办公会议触控一体机核心国产化品牌招投标参数
计算机外设·电脑·大屏端·信息发布系统·会议一体机
python-行者7 天前
akamai鼠标轨迹
爬虫·python·计算机外设·akamai
大Mod_abfun9 天前
多显示器窗口分布规律探索(包括WorkerW的区域)
计算机外设·显示器布局·桌面背景层·workerw
猎板PCB 邹亮13 天前
猎板PCB:专业键盘PCB板解决方案供应商
计算机外设·键盘·pcb工艺
点灯小铭14 天前
基于STM32单片机的无线鼠标设计
stm32·单片机·计算机外设·毕业设计·课程设计