Linux驱动之input输入子系统

2023-11-06

目录

前言:

介绍:

input_dev结构体:

输入子系统的使用流程:

实例测试 :


前言:

输入子系统用于实现Linux系统输入设备(鼠标 键盘 触摸屏  游戏杆)驱动的一种框架。Linux内核将其中的固定部分放入内核,驱动开发时只需要实现其中的不固定部分(主要还是和硬件相关的部分),这和platform设备驱动总线很相似。

介绍:

输入子系统对应的设备文件是固定的名称。例如 /dev/input/event0,event0是触摸屏的设备文件。别的输入设备文件可能是event1或event2等。

输入子系统的主设备号固定是13,向混杂设备的主设备号固定是10一样。

输入子系统从下到上分为输入驱动层、输入核心层、输入事件处理层,最终给用户空间提供可访问的设备节点。

  • Input driver :主要将硬件信息和上报的数据信息提供给核心层,中断设置。

  • Input core :为驱动层提供输入设备注册和操作接口,通知事件层对输入事件进行处理。注册设备类和cdev。

  • Event handler :为用户层提供统一的访问接口,并处理驱动层提交的数据处理。设备文件的创建。

input核心层会向Linux内核注册一个字符设备,drivers/input/input.c文件就是input输入子系统的核心层。核心层向内核注册了一个input类(/sys/class),并且注册了主设备号INPUT_MAJOR(13)的字符设备。在使用input子系统处理输入设备时就不需要去注册字符设备了,只需要向系统注册一个input_device即可。

input_dev结构体:

使用input子系统时,需要注册input_device设备,input_device由input_dev结构体表示。

   struct input_dev {  
        const char *name;  //提供给用户的输入设备的名称  
        const char *phys;  //提供给编程者的设备节点的名称  
        const char *uniq;  //指定唯一的ID号,就像MAC地址一样
        struct input_id id;  //输入设备标识ID,用于和事件处理层进行匹配
        unsigned long evbit[NBITS(EV_MAX)];   // 记录设备支持的事件类型 
        unsigned long keybit[NBITS(KEY_MAX)]; // 记录设备支持的按键类型   
        unsigned long relbit[NBITS(REL_MAX)]; // 表示能产生哪些相对坐标事件, x,y,滚轮  
        unsigned long absbit[NBITS(ABS_MAX)]; // 表示能产生哪些绝对坐标事件, x,y  
        unsigned long mscbit[BITS_TO_LONGS(MSC_CNT)];  
        unsigned long ledbit[BITS_TO_LONGS(LED_CNT)];  
        unsigned long sndbit[BITS_TO_LONGS(SND_CNT)];  
        unsigned long ffbit[BITS_TO_LONGS(FF_CNT)];  
        unsigned long swbit[BITS_TO_LONGS(SW_CNT)];
        unsigned int hint_events_per_packet;
        unsigned int keycodemax;
        unsigned int keycodesize;
        void *keycode;
        ...  
   }
  • input_dev 结构体中的evbit 表示输入事件类型, 每个输入事件类型都有不同的值,当我们这个输入子系统包含哪些输入事件,也就是能向用户上报哪些输入事件,就需要在evbit数组中使能置1。
#define EV_SYN   0x00            // 同步事件(上报)
#define EV_KEY   0x01            // 按键事件 
#define EV_REL   0x02            // 相对坐标事件 
#define EV_ABS   0x03            // 绝对坐标事件 
#define EV_MSC   0x04           
#define EV_SW    0x05            
#define EV_LED   0x11            
#define EV_SND   0x12          
#define EV_REP   0x14           
#define EV_FF    0x15           
#define EV_PWR   0x16           
#define EV_FF_STATUS 0x17     
#define EV_MAX   0x1f
#define EV_CNT   (EV_MAX+1)

上面是evbit所支持的事件类型,其中EV_SYN表示同步事件是用来上报事件的。当没有输入事件应用层read进程是阻塞的,收到EV_SYN才会唤醒向应用空间上报事件,简单来说就是当我们发生了一个按键事件或者相对坐标事件,内核会把事件放在缓冲区中,当发生了同步事件才会把阻塞的进程(read)唤醒并向应用空间上报发生的事件。绝对坐标事件是以整个屏幕的左上角为起点来计算坐标,但在滑动窗口中,是以窗口的左上角为起点来计算坐标称为相对坐标。

  • input_dev 结构体中的keybit 表示按键事件类型,每个按键都有不同的值,当我们这个输入子系统包含哪些按键码,也就是能向用户上报哪些按键码,就需要在keybit数组中使能置1。
#define KEY_RESERVED 0
#define KEY_ESC 1
#define KEY_1 2
#define KEY_2 3
#define KEY_3 4
#define KEY_4 5
#define KEY_5 6
#define KEY_6 7
#define KEY_7 8
#define KEY_8 9
#define KEY_9 10
#define KEY_0 11
  • input_dev 结构体中的absbit 表示绝对坐标事件类型,有不同的坐标类型,我们需要定义包含哪些坐标类型,在absbit数组中使能置1。
#define ABS_x          0x00
#define ABS_Y          0x01
#define A_z            0×02
#define ABS_RX         0x03
#define ABS_RY         0x04
#define ABS_RZ         0x05
#define ABS_THROTTLE   0x06
#define ABS_RUDDER     0x07
#define ABS_WHEEL      0×0B
#define ABS_GAS        0x09
#define ABS_BRAKE      0x0a
#define ABS_HATOx      0x10
#define ABS_HATOY      0x11
#define ABS_HAT1X      0x12
#define ABS_HAT1Y      0x13

使用set_bit()函数对以上的成员中某些位进行置1。

void set_bit(int nr, unsigned long *addr)

参数:

        nr: 要设置的那一位,从0开始

        addr: 开始计数的地址

例如key = 0x10; set_bit(0,&key); 那key的第0位被置1,key = 0x11。

输入子系统的使用流程:

需要的头文件:

#include <linux/input.h>

先申请,再注册,先注销,再释放。 

//申请 input_dev 结构体变量 
struct input_dev *input_allocate_device(void)
//返回值:申请到的input_dev

//注册 input_dev 
int input_register_device(struct input_dev *dev)
参数:

        dev:要注册的input_dev
//返回值:0,注册成功;负值,注册失败

............

//注销 input_dev 
void input_unregister_device(struct input_dev *dev)
参数:        

        dev:要注销的 input_dev/

/注销 input_dev 
void input_unregister_device(struct input_dev *dev)
参数:

        dev:要注销的 input_dev

把触发的输入事件类型及输入数据,上报给内核,使用input_event()函数上报。

void input_event(struct input_dev *dev,
                 unsigned int type,
                 unsigned int code,
                 int value)
参数:
      dev: 需要上报的 input_dev
      type: 上报的事件类型,比如 EV_KEY,EV_ABS
      code: 事件码,也就是我们注册的按键值或者坐标轴,比如KEY_A、 ABS_X 等等
      value: 事件值,例如0表示按键按下,1表示按键松开,(50,64)表示触摸点的X,Y坐标

上报事件结束以后还需要使用告诉内核,使用input_sync()或者input_event()函数。

void input_sync(struct input_dev *dev)
参数:

        dev:需要上报同步事件的 input_dev

void input_event(struct input_dev *dev, EV_SYN, 0, 0)

Linux内核使用input_event结构体来表示所有的输入事件,用户应用程序可以通过input_event来获取到具体的输入事件或相关的值。

/* input_envent 结构体定义在include/uapi/linux/input.h 文件中 */
struct input_event {
 struct timeval time;
 __u16 type;
 __u16 code;
 __s32 value;
};
//time:时间,也就是此事件发生的时间,为 timeval 结构体类型
//type:事件类型,比如EV_KEY,表示此次事件为按键事件
//code:事件码,比如在EV_KEY事件中就表示具体的按键码,KEY_0/KEY_1等按键
//value:值,比如EV_KEY事件中就表示按键值,表示按键有没有被按下

实例测试 :

编写用一个按键的输入子系统例子来测试一下吧。

btn_drv.c

#include <linux/init.h>
#include <linux/module.h>
#include <linux/interrupt.h>
#include <linux/irq.h>
#include <asm/gpio.h>
#include <mach/platform.h>
#include <linux/timer.h>
#include <linux/input.h>

struct btn_res{
	int gpio;//端口号
	char *name;//名称
	int code;//键值
};


struct btn_res btn_info[] = {
	[0] = {
		.gpio = PAD_GPIO_A+28,
		.name = "K2",
		.code = KEY_2,
	},
	[1] = {
		.gpio = PAD_GPIO_B+9,
		.name = "K6",
		.code = KEY_6,
	},
	[2] = {
		.gpio = PAD_GPIO_B+30,
		.name = "K3",
		.code = KEY_3,
	},
	[3] = {
		.gpio = PAD_GPIO_B+31,
		.name = "K4",
		.code = KEY_4,
	}
};

//初始化内核定时器
struct timer_list btn_timer;
//声明input_dev指针
struct input_dev *btn_dev = NULL; 

//超时处理函数---真实按键事件
void btn_timer_function(unsigned long data)
{
	int state;//引脚状态
	struct btn_res *pdata = (struct btn_res *)data;//引脚数据
	
	//区分哪个按键
	//区分按下松开
	state = gpio_get_value(pdata->gpio);

	//上报数据
	//按键事件
	input_event(btn_dev,EV_KEY,pdata->code,!state);
	//同步事件-----唤醒接口层阻塞的读函数
	input_event(btn_dev,EV_SYN,0,0);
}

//中断处理函数
irqreturn_t btn_handler(int irq, void *dev_id)
{
	//设置超时处理函数的参数
	btn_timer.data = (unsigned long)dev_id;
	//重置定时器--- 10ms超时
	mod_timer(&btn_timer, jiffies+msecs_to_jiffies(10));
	
	return IRQ_HANDLED;//处理成功
}

//加载函数
int btn_input_init(void)
{
	int ret,i,j;

	// 1.分配input_dev
	btn_dev = input_allocate_device();
	if(IS_ERR_OR_NULL(btn_dev)){
		printk("input_allocate_device failed!\n");
		ret = -ENOMEM;
		goto failure_input_allocate;
	}

	// 2.初始化input_dev
	//设置会发生的事件
	set_bit(EV_KEY,btn_dev->evbit);
	set_bit(EV_SYN,btn_dev->evbit);
	//会使用的键值
	for(i=0;i<ARRAY_SIZE(btn_info);i++){
		set_bit(btn_info[i].code,btn_dev->keybit);
	}

	// 3.将input_dev注册到内核
	ret = input_register_device(btn_dev);
	if(ret<0){
		printk("input_register_device failed!\n");
		goto failure_input_register;
	}
	

	/*ARRAY_SIZE求数组元素个数*/
	for(i=0;i<ARRAY_SIZE(btn_info);i++){
		//申请中断
		ret = request_irq(gpio_to_irq(btn_info[i].gpio), //中断号
						btn_handler, //中断处理函数
						IRQF_TRIGGER_RISING|IRQF_TRIGGER_FALLING, //中断标志,包括触发方式----- 上升下降沿触发
						btn_info[i].name, //中断名称
						&btn_info[i]);//传递给中断处理函数的参数
		if(ret<0){
			printk("request_irq failed!\n");
			goto failure_request_irq;
		}
	}
	
	//初始化定时器
	init_timer(&btn_timer);
	btn_timer.function = btn_timer_function;

	return 0;

failure_request_irq:
	//第i次失败了
	//释放0 --- i-1中申请的中断
	for(j=0;j<i;j++){
		free_irq(gpio_to_irq(btn_info[j].gpio), &btn_info[j]);
	}
	input_unregister_device(btn_dev);
failure_input_register:
	input_free_device(btn_dev);
failure_input_allocate:
	return ret;
}

//卸载函数
void btn_input_exit(void)
{
	int i;

	del_timer(&btn_timer);
	
	//释放所有申请的中断
	for(i=0;i<ARRAY_SIZE(btn_info);i++){
		free_irq(gpio_to_irq(btn_info[i].gpio), &btn_info[i]);
	}

	//从内核注销inpt_dev
	input_unregister_device(btn_dev);
	//释放input_dev空间
	input_free_device(btn_dev);
}

//声明为模块的入口和出口
module_init(btn_input_init);
module_exit(btn_input_exit);

MODULE_LICENSE("GPL");//GPL模块许可证
MODULE_AUTHOR("xin");//作者
MODULE_VERSION("1.0");//版本
MODULE_DESCRIPTION("button input module!");//描述信息

 btn_test.c

#include <stdio.h>
#include <stdlib.h>
#include <fcntl.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <unistd.h>
#include <sys/ioctl.h>
#include <linux/input.h>

int main()
{
	int ret;
	struct input_event key_evt;

	int fd = open("/dev/event5",O_RDWR);
	if(fd==-1){
		perror("open");
		exit(-1);
	}

	printf("open successed!fd = %d\n",fd);

	while(1){
		//读取键值
		ret = read(fd,&key_evt,sizeof(key_evt));
		if(ret<0){
			perror("read");
			break;
		}
		
			
		if(key_evt.type==EV_KEY){
			switch(key_evt.code){
				case KEY_2:
					printf("key_2 %s \n",key_evt.value?"pressed":"released");
					break;
				case KEY_3:
					printf("key_3 %s \n",key_evt.value?"pressed":"released");
					break;
				case KEY_4:
					printf("key_4 %s \n",key_evt.value?"pressed":"released");
					break;
				case KEY_6:
					printf("key_6 %s \n",key_evt.value?"pressed":"released");
					break;
				default:
					printf("unknow input!\n");
					break;
			}
		}
	}

	close(fd);
	return 0;
}

 如图所示,按键输入子系统加载到内核中/dev/input目录下多出了一个event5设备文件,且使用定时器消,抖按键没有抖动。 

参考文章:https://zhuanlan.zhihu.com/p/571498030 

好了,以上就是Linux驱动输入子系统的全部内容了,如果有什么疑问和建议欢迎在评论区中提出来喔。 

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

Linux驱动之input输入子系统 的相关文章

  • 如何在shell中输出返回码?

    我正在尝试通过调用自定义 shell 脚本sh bin sh c myscript sh gt log txt 2 gt 1 echo 该命令的输出是创建的后台进程的 PID 我想指导 bin sh保存返回码myscript sh到某个文件
  • 使用 sh 运行 bash 脚本

    我有 bash 脚本 它需要 bash 另一个人尝试运行它 sh script name sh 它失败了 因为 sh 是他的发行版中 dash 的符号链接 ls la bin sh lrwxrwxrwx 1 root root 4 Aug
  • 如何通过保持目录结构完整来同步路径中匹配模式的文件?

    我想将所有文件从服务器 A 复制到服务器 B 这些文件在不同级别的文件系统层次结构中具有相同的父目录名称 例如 var lib data sub1 sub2 commonname filetobecopied foo var lib dat
  • jq中如何分组?

    这是 json 文档 name bucket1 clusterName cluster1 name bucket2 clusterName cluster1 name bucket3 clusterName cluster2 name bu
  • 查找哪些页面不再与写入时复制共享

    假设我在 Linux 中有一个进程 我从中fork 另一个相同的过程 后forking 因为原始进程将开始写入内存 Linux写时复制机制将为进程提供与分叉进程使用的不同的唯一物理内存页 在执行的某个时刻 我如何知道原始进程的哪些页面已被写
  • 如何确保应用程序在 Linux 上持续运行

    我试图确保脚本在开发服务器上保持运行 它会整理统计数据并提供网络服务 因此它应该会持续存在 但一天中有几次 它会因未知原因而消失 当我们注意到时 我们只需再次启动它 但这很麻烦 并且某些用户没有权限 或专有技术 来启动它 作为一名程序员 我
  • 内核模式下的线程(和进程)与用户模式下的线程(和进程)有什么区别?

    我的问题 1 书中现代操作系统 它说线程和进程可以处于内核模式或用户模式 但没有明确说明它们之间有什么区别 2 为什么内核态线程和进程的切换比用户态线程和进程的切换花费更多 3 现在 我正在学习Linux 我想知道如何在LINUX系统中分别
  • 如何在 Bash 中给定超时后终止子进程?

    我有一个 bash 脚本 它启动一个子进程 该进程时不时地崩溃 实际上是挂起 而且没有明显的原因 闭源 所以我对此无能为力 因此 我希望能够在给定的时间内启动此进程 如果在给定的时间内没有成功返回 则将其终止 有没有simple and r
  • 是否可以创建一个脚本来保存和恢复权限?

    我正在使用 Linux 系统 需要对一组嵌套文件和目录进行一些权限实验 我想知道是否没有某种方法可以保存文件和目录的权限 而不保存文件本身 换句话说 我想保存权限 编辑一些文件 调整一些权限 然后将权限恢复到目录结构中 将更改的文件保留在适
  • 删除 Git 存储库,但保留所有文件

    在我使用 Linux 的过程中的某个时刻 我决定将我的主目录中的所有内容都放入源代码管理中是个好主意 我不是在问这是否是一个好主意 我是在问如何撤销它 删除存储库的原因是我最近安装了 Oh My Zsh 而且我非常喜欢它 问题是我的主目录有
  • Linux - 从第二个选项卡获取文本

    假设我们有这样的文件 一些文本11 一些文本12 一些文本13 一些文本21 一些文本22 一些文本23 文本由制表符分隔 我们知道第 1 列中的一些文本 但希望从第 2 列中获取文本 我知道我可以通过以下方式获取线路 grep somet
  • 在两次之间每分钟执行一次 Cronjob

    我需要在 crontab 中每分钟运行一个 bash 脚本8 45am and 9 50am每天的 Code 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 8 home pull sh gt ho
  • 确定我可以向文件句柄写入多少内容;将数据从一个 FH 复制到另一个 FH

    如何确定是否可以将给定数量的字节写入文件句柄 实际上是套接字 或者 如何 取消读取 我从其他文件句柄读取的数据 我想要类似的东西 n how much can I write w handle n read r handle buf n a
  • 如何在c linux中收听特定接口上的广播?

    我目前可以通过执行以下操作来收听我编写的简单广播服务器 仅广播 hello int fd socket PF INET SOCK DGRAM 0 struct sockaddr in addr memset addr 0 sizeof ad
  • C++ Boost ASIO 简单的周期性定时器?

    我想要一个非常简单的周期性计时器每 50 毫秒调用我的代码 我可以创建一个始终休眠 50 毫秒的线程 但这很痛苦 我可以开始研究用于制作计时器的 Linux API 但它不可移植 I d like使用升压 我只是不确定这是否可能 boost
  • 在 unix 中编译 dhrystone 时出错

    我是使用基准测试和 makefile 的新手 我已经从下面的链接下载了 Dhrystone 基准测试 我正在尝试编译它 但我遇到了奇怪的错误 我尝试解决它 但没有成功 有人可以帮助我运行 dhrystone 基准测试吗 以下是我尝试编译的两
  • Linux 上的基准测试程序

    对于一项任务 我们需要使用不同的优化和参数来对我们的实现进行基准测试 有没有一种可行的方法可以在Linux命令行 我知道时间 上使用不同的参数对小程序进行基准测试 从而为我提供CSV或类似内容的时间数据 输出可能类似于 Implementa
  • 警告:请求的映像平台 (linux/amd64) 与检测到的主机平台 (linux/arm64/v8) 不匹配

    警告 请求的映像平台 linux amd64 与检测到的主机平台 linux arm64 v8 不匹配 并且未请求特定平台 docker 来自守护程序的错误响应 无法选择具有功能的设备驱动程序 gpu 我在 mac 上尝试运行此命令时遇到此
  • 如何在 GNU/Linux 上设置 Subversion (SVN) 服务器 - Ubuntu [关闭]

    Closed 这个问题需要多问focused help closed questions 目前不接受答案 我有一台运行 Ubuntu 的笔记本电脑 我想将其用作 Subversion 服务器 既让我自己在本地承诺 也让其他人远程承诺 要使其
  • 如何从 PROC 获取有关子进程的信息

    我正在尝试编写一个以几个进程作为参数的程序 然后父进程执行每个子进程并打印出一些相关的统计信息 示例 generate ls l 将生成一个程序 打印出有关 ls l 的一些统计信息 特别是其系统时间 用户时间和上下文切换次数 我不想使用

随机推荐

  • linux下使用X2Go进行远程连接

    http www unixmen com x2go an open source remote desktop solution for linux About X2Go X2Go is an Open Source remote desk
  • openwrt上“DHCP packet received on ra0 which has no address”问题查找和解决

    在商场进行商业wifi布点的过程中 经常有用户反映说wifi连不上了 或者是连上了是一个叹号 分配不了IP 电脑设置了静态IP是可以接入的 check了logread发现打印里面有这么一句 DHCP packet received on r
  • 机器人学笔记(01):1.绪论

    1 机器人名称的由来 About Robot 机器人的英文名词叫Robot Robot一词最早出现在1920年捷克作家卡雷尔 卡佩克 Karel Capek 所写的一个剧本中 这个剧本的名字叫 Rossum s Universal Robo
  • 信息学奥赛一本通(c++):2024:【例4.10】末两位数

    一 题目 2024 例4 10 末两位数时间限制 1000 ms 内存限制 65536 KB 题目描述 求n个1992的乘积的末两位数是多少 输入 输入n 输出 如题述的末两位数 输入样例 3 输出样例 88 提示 数据范围 对于所有数据
  • GBase 8c 教程(十)分片基础操作

    在建表语句中加入分片操作 可以将数据以HASH分片的方式分配至每个DN节点 操作如下 1 建表 使用distribute by进行HASH分片操作 CREATE TABLE disttab c1 int c2 int c3 varchar
  • 深度学习笔记20235018

    使用深度神经网络 几十层 几百层的神经网络 从大规模数据中学习的过程就是深度学习 将输入的数据的异同点进行提取并且层层抽象 记录在神经网络的参数里就训练出来一个可以区分这些物体的深度神经网络 面向机器的知识图谱 是一个关联的知识网络 举一反
  • PCL求取三维点云模型每点曲率

    最近在做有关实验需要计算模型曲率 但是网上找了一圈也没找到满意的资料 最后发现PCL库可以很方便的求取模型中每一个点的曲率 但是我们要想将PCL库求得的曲率数据应用到自己的项目中需要将PCL库与我们的项目进行结合 并且在PCL求出曲率后存放
  • SSM框架之SpringMVC

    文章目录 SpringMVC MVC设计模式 springmvc框架 入门程序 配置前端控制器和处理器适配器 开发Handler 映射器和适配器 非注解的处理器映射器 非注解的处理器适配器 注解的处理器映射器和适配器 重点 源码分析 spr
  • 长短时记忆网络(LSTM)负荷预测项目(matlab)

    目录 1 LSTM介绍 2 数据集准备及预处理 3 LSTM模型搭建与训练 4 预测模型测试 1 LSTM介绍 长短期记忆网络 LSTM long short term memory 是 RNN 的一种变体 其核心概念在于细胞状态以及 门
  • 微信小程序宠物用品服务商城SSM-JAVA【数据库设计、论文、源码、开题报告】

    功能介绍 操作角色为管理员和用户 商家 管理员的功能为用户管理 商家管理 宠物分类管理 宠物信息管理 商品分类管理 宠物用品管理 项目类型管理 服务项目管理 宠物日志管理 订单管理等 用户的功能为购买宠物 商品 预约服务发表日志管理订单等
  • 美团西安美食部分爬虫(修改版)(python)

    美团美食 coding UTF 8 import requests import time from bs4 import BeautifulSoup import json import csv import random with op
  • 区块链100篇之fabric的网络搭建(一)

    好久没写博客了 今天开始写关于fabric相关的博客 包括网络的搭建 链码的编写以及sdk的封装 搭建网络 网络的搭建想分为三篇来写 第一篇写网络的搭建 这里是官方fabric samples的简化版 第二篇是关于yaml文件的编写 第三篇
  • 【数据科学】肯德尔等级相关系数( Kendall's tau coefficient )

    在统计学中 Kendall等级相关系数 通常称为Kendall的tau系数 在希腊字母 之后 是用于测量两个测量量之间的序数关联的统计量 甲tau蛋白测试是一种非参数假设检验用于基于所述tau蛋白系数统计依赖性 它是衡量等级相关 数据的排序
  • java 版本企业招标投标管理系统源码+功能描述+tbms+及时准确+全程电子化

    功能描述 1 门户管理 所有用户可在门户页面查看所有的公告信息及相关的通知信息 主要板块包含 招标公告 非招标公告 系统通知 政策法规 2 立项管理 企业用户可对需要采购的项目进行立项申请 并提交审批 查看所有的立项信息 主要功能包含 招标
  • 第五章 Maven结合Junit实现单元测试

    maven的重要职责之一就是自动运行单元测试 它通过maven surefire plugin与主流的单元测试框架junit和testng集成 并且能够自动生成丰富的结果报表 maven并不是一个单元测试框架 他只是在构建执行打特定的生命周
  • RobotFramework介绍

    Robot Framework 1 入门介绍 小菠萝测试笔记 博客园 cnblogs com
  • C++——初始化列表

    初始化列表 在构造函数执行时 先执行初始化列表 实现变量的初始化 然后再执行函数内部的语句 构造函数体赋值 在创建对象时 编译器通过调用构造函数 给对象中各个成员变量一个合适的初始值 class Date public Date int y
  • css中nth-child的属性

    参数为整数 nth child 1 它表示要选择父元素中索引为该数值的子元素 此时的索引值从1开始 参数是奇数偶数 nth child odd odd表示选择奇数项的子元素 nth child even even表示选择偶数项的子元素 参数
  • Tkinter 组件详解(一):Label

    Tkinter 组件详解之Label Label 标签 组件用于在屏幕上显示文本或图像 Label 组件仅能显示单一字体的文本 但文本可以跨越多行 另外 还可以为其中的个别字符加上下划线 例如用于表示键盘快捷键 何时使用 Label 组件
  • Linux驱动之input输入子系统

    目录 前言 介绍 input dev结构体 输入子系统的使用流程 实例测试 前言 输入子系统用于实现Linux系统输入设备 鼠标 键盘 触摸屏 游戏杆 驱动的一种框架 Linux内核将其中的固定部分放入内核 驱动开发时只需要实现其中的不固定