普中51单片机:矩阵按键扫描与应用详解(五)

在这里插入图片描述

文章目录

  • 引言
  • 电路图
  • 开发板IO连接
  • 矩阵键盘的工作原理
    • 行列扫描
    • 逐行/逐列扫描
  • LCD1602代码库
  • 代码演示——暴力扫描
  • 代码演示——数码管(行列式)
  • 代码演示——线翻转法
  • 代码演示——LCD1602密码锁

引言

矩阵按键是一种通过行列交叉连接的按键阵列,可以有效地减少单片机I/O口的使用。常见的4x4矩阵键盘只需要8个I/O口即可读取16个按键的状态。采用逐行或逐列的“扫描”,就可以读出任何位置按键的状态。

电路图

在这里插入图片描述

开发板IO连接

请添加图片描述
根据图片可以看出,矩阵按键的连接在P1端口,下面是它的原理图。
在这里插入图片描述

注意:牵扯到数码管相连接的实物图,这里不做展示,可查看之前章节的数码管讲解:数码管讲解演示

矩阵键盘的工作原理

矩阵键盘通过行列扫描的方式来检测按键的状态。假设我们有一个4x4的矩阵键盘,它由4条行线和4条列线组成,总共可以检测16个按键。每个按键位于行线和列线的交叉点上。

行列扫描

行列扫描的基本步骤如下:

  1. 初始化:将所有行线设置为高电平,所有列线设置为低电平。
  2. 扫描行:逐行将行线设置为低电平,检测列线的状态。如果某一列线变为低电平,说明该行的某个按键被按下。(扫描列则反之)
  3. 确定列:将检测到的列线设置为高电平,逐列扫描,确定具体的按键位置。

逐行/逐列扫描

逐行/逐列扫描的本质与行列扫描类似,但适用于矩阵键盘接到了任意的I/O口。具体步骤如下:

  • 逐行扫描:将某一行设置为低电平,其余行和列设置为高电平,读取列线数据。
  • 逐列扫描:将某一列设置为低电平,其余行和列设置为高电平,读取行线数据。

LCD1602代码库

注意:LCD1602不过多讲解,后续会提供专门的章节。因为矩阵按键代码牵扯到LCD1602,根据源码可以看如何使用即可,也可以使用上一章的内容,使用数码管进行显示!

  • LCD1602.h
#ifndef __LCD1602_H__
#define __LCD1602_H__

//用户调用函数
void LCD_Init();
void LCD_ShowChar(unsigned char Line,unsigned char Column,char Char);
void LCD_ShowString(unsigned char Line,unsigned char Column,char *String);
void LCD_ShowNum(unsigned char Line,unsigned char Column,unsigned int Number,unsigned char Length);
void LCD_ShowSignedNum(unsigned char Line,unsigned char Column,int Number,unsigned char Length);
void LCD_ShowHexNum(unsigned char Line,unsigned char Column,unsigned int Number,unsigned char Length);
void LCD_ShowBinNum(unsigned char Line,unsigned char Column,unsigned int Number,unsigned char Length);

#endif
  • LCD1602.c
#include <REGX52.H>

//引脚配置:
sbit LCD_RS=P2^6;
sbit LCD_RW=P2^5;
sbit LCD_EN=P2^7;
#define LCD_DataPort P0

//函数定义:
/**
  * @brief  LCD1602延时函数,12MHz调用可延时1ms
  * @param  无
  * @retval 无
  */
void LCD_Delay()
{
	unsigned char i, j;

	i = 2;
	j = 239;
	do
	{
		while (--j);
	} while (--i);
}

/**
  * @brief  LCD1602写命令
  * @param  Command 要写入的命令
  * @retval 无
  */
void LCD_WriteCommand(unsigned char Command)
{
	LCD_RS=0;
	LCD_RW=0;
	LCD_DataPort=Command;
	LCD_EN=1;
	LCD_Delay();
	LCD_EN=0;
	LCD_Delay();
}

/**
  * @brief  LCD1602写数据
  * @param  Data 要写入的数据
  * @retval 无
  */
void LCD_WriteData(unsigned char Data)
{
	LCD_RS=1;
	LCD_RW=0;
	LCD_DataPort=Data;
	LCD_EN=1;
	LCD_Delay();
	LCD_EN=0;
	LCD_Delay();
}

/**
  * @brief  LCD1602设置光标位置
  * @param  Line 行位置,范围:1~2
  * @param  Column 列位置,范围:1~16
  * @retval 无
  */
void LCD_SetCursor(unsigned char Line,unsigned char Column)
{
	if(Line==1)
	{
		LCD_WriteCommand(0x80|(Column-1));
	}
	else if(Line==2)
	{
		LCD_WriteCommand(0x80|(Column-1+0x40));
	}
}

/**
  * @brief  LCD1602初始化函数
  * @param  无
  * @retval 无
  */
void LCD_Init()
{
	LCD_WriteCommand(0x38);//八位数据接口,两行显示,5*7点阵
	LCD_WriteCommand(0x0c);//显示开,光标关,闪烁关
	LCD_WriteCommand(0x06);//数据读写操作后,光标自动加一,画面不动
	LCD_WriteCommand(0x01);//光标复位,清屏
}

/**
  * @brief  在LCD1602指定位置上显示一个字符
  * @param  Line 行位置,范围:1~2
  * @param  Column 列位置,范围:1~16
  * @param  Char 要显示的字符
  * @retval 无
  */
void LCD_ShowChar(unsigned char Line,unsigned char Column,char Char)
{
	LCD_SetCursor(Line,Column);
	LCD_WriteData(Char);
}

/**
  * @brief  在LCD1602指定位置开始显示所给字符串
  * @param  Line 起始行位置,范围:1~2
  * @param  Column 起始列位置,范围:1~16
  * @param  String 要显示的字符串
  * @retval 无
  */
void LCD_ShowString(unsigned char Line,unsigned char Column,char *String)
{
	unsigned char i;
	LCD_SetCursor(Line,Column);
	for(i=0;String[i]!='\0';i++)
	{
		LCD_WriteData(String[i]);
	}
}

/**
  * @brief  返回值=X的Y次方
  */
int LCD_Pow(int X,int Y)
{
	unsigned char i;
	int Result=1;
	for(i=0;i<Y;i++)
	{
		Result*=X;
	}
	return Result;
}

/**
  * @brief  在LCD1602指定位置开始显示所给数字
  * @param  Line 起始行位置,范围:1~2
  * @param  Column 起始列位置,范围:1~16
  * @param  Number 要显示的数字,范围:0~65535
  * @param  Length 要显示数字的长度,范围:1~5
  * @retval 无
  */
void LCD_ShowNum(unsigned char Line,unsigned char Column,unsigned int Number,unsigned char Length)
{
	unsigned char i;
	LCD_SetCursor(Line,Column);
	for(i=Length;i>0;i--)
	{
		LCD_WriteData(Number/LCD_Pow(10,i-1)%10+'0');
	}
}

/**
  * @brief  在LCD1602指定位置开始以有符号十进制显示所给数字
  * @param  Line 起始行位置,范围:1~2
  * @param  Column 起始列位置,范围:1~16
  * @param  Number 要显示的数字,范围:-32768~32767
  * @param  Length 要显示数字的长度,范围:1~5
  * @retval 无
  */
void LCD_ShowSignedNum(unsigned char Line,unsigned char Column,int Number,unsigned char Length)
{
	unsigned char i;
	unsigned int Number1;
	LCD_SetCursor(Line,Column);
	if(Number>=0)
	{
		LCD_WriteData('+');
		Number1=Number;
	}
	else
	{
		LCD_WriteData('-');
		Number1=-Number;
	}
	for(i=Length;i>0;i--)
	{
		LCD_WriteData(Number1/LCD_Pow(10,i-1)%10+'0');
	}
}

/**
  * @brief  在LCD1602指定位置开始以十六进制显示所给数字
  * @param  Line 起始行位置,范围:1~2
  * @param  Column 起始列位置,范围:1~16
  * @param  Number 要显示的数字,范围:0~0xFFFF
  * @param  Length 要显示数字的长度,范围:1~4
  * @retval 无
  */
void LCD_ShowHexNum(unsigned char Line,unsigned char Column,unsigned int Number,unsigned char Length)
{
	unsigned char i,SingleNumber;
	LCD_SetCursor(Line,Column);
	for(i=Length;i>0;i--)
	{
		SingleNumber=Number/LCD_Pow(16,i-1)%16;
		if(SingleNumber<10)
		{
			LCD_WriteData(SingleNumber+'0');
		}
		else
		{
			LCD_WriteData(SingleNumber-10+'A');
		}
	}
}

/**
  * @brief  在LCD1602指定位置开始以二进制显示所给数字
  * @param  Line 起始行位置,范围:1~2
  * @param  Column 起始列位置,范围:1~16
  * @param  Number 要显示的数字,范围:0~1111 1111 1111 1111
  * @param  Length 要显示数字的长度,范围:1~16
  * @retval 无
  */
void LCD_ShowBinNum(unsigned char Line,unsigned char Column,unsigned int Number,unsigned char Length)
{
	unsigned char i;
	LCD_SetCursor(Line,Column);
	for(i=Length;i>0;i--)
	{
		LCD_WriteData(Number/LCD_Pow(2,i-1)%2+'0');
	}
}

代码演示——暴力扫描

这个程序初始化LCD显示屏并不断扫描4x4矩阵键盘。当按键被按下时,相应的键号会显示在LCD1602上。延迟函数确保按键消抖,以避免多次检测到单次按键。

扫描4x4矩阵键盘。它依次将每一列设置为低电平,并检查每一行是否有低电平信号,表示按键被按下。返回相应的键号。

#include <REGX52.H>
#include "LCD1602.h"

void DelayXms(unsigned int xms)	//@12.000MHz
{
	unsigned char data i, j;
	
	while(xms)
	{
		i = 2;
		j = 239;
		do
		{
			while (--j);
		} while (--i);
		xms--;
	}
}

unsigned char MatrixKey()
{
	unsigned char keyNumber = 0;

	P1 = 0xff;
	P1_3 = 0;//第一列低电平
	if(P1_7 == 0)
	{
		DelayXms(1);
		while(P1_7 == 0);
		DelayXms(1);
		keyNumber = 1;
	}
	if(P1_6 == 0)
	{
		DelayXms(1);
		while(P1_6 == 0);
		DelayXms(1);
		keyNumber = 5;
	}
	if(P1_5 == 0)
	{
		DelayXms(1);
		while(P1_5 == 0);
		DelayXms(1);
		keyNumber = 9;
	}
	if(P1_4 == 0)
	{
		DelayXms(1);
		while(P1_4 == 0);
		DelayXms(1);
		keyNumber = 13;
	}

	P1 = 0xff;
	P1_2 = 0;//第二列低电平
	if(P1_7 == 0)
	{
		DelayXms(1);
		while(P1_7 == 0);
		DelayXms(1);
		keyNumber = 2;
	}
	if(P1_6 == 0)
	{
		DelayXms(1);
		while(P1_6 == 0);
		DelayXms(1);
		keyNumber = 6;
	}
	if(P1_5 == 0)
	{
		DelayXms(1);
		while(P1_5 == 0);
		DelayXms(1);
		keyNumber = 10;
	}
	if(P1_4 == 0)
	{
		DelayXms(1);
		while(P1_4 == 0);
		DelayXms(1);
		keyNumber = 14;
	}

	P1 = 0xff;
	P1_1 = 0;//第三列低电平
	if(P1_7 == 0)
	{
		DelayXms(1);
		while(P1_7 == 0);
		DelayXms(1);
		keyNumber = 3;
	}
	if(P1_6 == 0)
	{
		DelayXms(1);
		while(P1_6 == 0);
		DelayXms(1);
		keyNumber = 7;
	}
	if(P1_5 == 0)
	{
		DelayXms(1);
		while(P1_5 == 0);
		DelayXms(1);
		keyNumber = 11;
	}
	if(P1_4 == 0)
	{
		DelayXms(1);
		while(P1_4 == 0);
		DelayXms(1);
		keyNumber = 15;
	}

	P1 = 0xff;
	P1_0 = 0;//第四列低电平
	if(P1_7 == 0)
	{
		DelayXms(1);
		while(P1_7 == 0);
		DelayXms(1);
		keyNumber = 4;
	}
	if(P1_6 == 0)
	{
		DelayXms(1);
		while(P1_6 == 0);
		DelayXms(1);
		keyNumber = 8;
	}
	if(P1_5 == 0)
	{
		DelayXms(1);
		while(P1_5 == 0);
		DelayXms(1);
		keyNumber = 12;
	}
	if(P1_4 == 0)
	{
		DelayXms(1);
		while(P1_4 == 0);
		DelayXms(1);
		keyNumber = 16;
	}

	return keyNumber;
} 

void main()
{
	LCD_Init();
	LCD_ShowString(1,1,"helloword");
	while(1)
	{
		unsigned char temp = MatrixKey();
		if(temp)
		{
			LCD_ShowNum(2,1,temp,2);
		}
	}
}

代码演示——数码管(行列式)

程序初始化数码管并不断扫描4x4矩阵键盘。当按键被按下时,相应的键号会显示在数码管上。延迟函数确保按键消抖,以避免多次检测到单次按键。(9之后的数字显示的是字母A~E),数码管如何连接可参考此博客:数码管讲解演示

#include <REGX52.H>

//共阴极数码管显示 0~F 的段码数据
unsigned char gsmg_code[17]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,
0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71};

void DelayXms(unsigned int xms)	//@12.000MHz
{
	unsigned char data i, j;
	
	while(xms)
	{
		i = 2;
		j = 239;
		do
		{
			while (--j);
		} while (--i);
		xms--;
	}
}

unsigned char MatrixRanksScan()
{
	unsigned char keyNum = 0;
	P1 = 0xf7;//第一列低电平
	if(P1 != 0xf7)//表示第一列有按键按下
	{
		DelayXms(1);//消抖处理
		switch(P1)
		{
			case 0x77:
				keyNum = 1;
				break;
			case 0xb7:
				keyNum = 5;
				break;
			case 0xd7:
				keyNum = 9;
				break;
			case 0xe7:
				keyNum = 13;
				break;
		}
	}
	while(P1 != 0xf7);

	P1 = 0xfb;//第二列低电平
	if(P1 != 0xfb)//表示第一列有按键按下
	{
		DelayXms(1);//消抖处理
		switch(P1)
		{
			case 0x7b:
				keyNum = 2;
				break;
			case 0xbb:
				keyNum = 6;
				break;
			case 0xdb:
				keyNum = 10;
				break;
			case 0xeb:
				keyNum = 14;
				break;
		}
	}
	while(P1 != 0xfb);

	P1 = 0xfd;//第三列低电平
	if(P1 != 0xfd)//表示第一列有按键按下
	{
		DelayXms(1);//消抖处理
		switch(P1)
		{
			case 0x7d:
				keyNum = 3;
				break;
			case 0xbd:
				keyNum = 7;
				break;
			case 0xdd:
				keyNum = 11;
				break;
			case 0xeb:
				keyNum = 15;
				break;
		}
	}
	while(P1 != 0xfd);

	P1 = 0xfe;//第四列低电平
	if(P1 != 0xfe)//表示第一列有按键按下
	{
		DelayXms(1);//消抖处理
		switch(P1)
		{
			case 0x7e:
				keyNum = 4;
				break;
			case 0xbe:
				keyNum = 8;
				break;
			case 0xde:
				keyNum = 12;
				break;
			case 0xee:
				keyNum = 16;
				break;
		}
	}
	while(P1 != 0xfe);
	return keyNum;
}

void main()
{
	unsigned char temp = 0;
	P0 = ~gsmg_code[temp];
	while(1)
	{
		temp = MatrixRanksScan();
		if(temp)
		{
			P0 = ~gsmg_code[temp];
		}
	}
}

代码演示——线翻转法

线翻转法是一种用于矩阵键盘扫描的技术,特别适用于单片机控制的场景。它通过交替设置行和列的电平来检测按键的按下位置。扫描4x4矩阵键盘。它首先将列设置为低电平,并检查是否有按键按下。如果有按键按下,则进一步检查行,确定具体的按键位置,并返回相应的键值。

#include <REGX52.H>

//共阴极数码管显示 0~F 的段码数据
unsigned char gsmg_code[17]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,
0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71};

void DelayXms(unsigned int xms)	//@12.000MHz
{
	unsigned char data i, j;
	
	while(xms)
	{
		i = 2;
		j = 239;
		do
		{
			while (--j);
		} while (--i);
		xms--;
	}
}

unsigned char MatrixFlipScan()
{
	unsigned char key_value = 0;	

	P1 = 0x0f;
	if(P1 != 0x0f)//列被按下
	{
		DelayXms(1);//消抖处理
		
		if(P1 != 0x0f)
		{
			//测试列
			switch(P1)
			{
				case 0x07://第一列有按键按下
					key_value = 1;
					break;
				case 0x0b://第二列有按键按下
					key_value = 2;
					break;
				case 0x0d://第三列有按键按下
					key_value = 3;
					break;
				case 0x0e://第四列有按键按下
					key_value = 4;
					break;	
			}
			//测试行
			P1 = 0xf0;
			switch(P1)
			{
				case 0x70://第一行有按键按下
					key_value = key_value;
					break;
				case 0xb0://第二行有按键按下
					key_value = key_value+4;
					break;
				case 0xd0://第三行有按键按下
					key_value = key_value+8;
					break;
				case 0xe0://第四行有按键按下
					key_value = key_value+12;
					break;	
			}
		}
	}
	return key_value;
}

void main()
{
	unsigned char temp = 0;
	P0 = ~gsmg_code[temp];
	while(1)
	{
		temp = MatrixFlipScan();
		if(temp)
		{
			P0 = ~gsmg_code[temp];
		}
	}
}

代码演示——LCD1602密码锁

这个程序初始化LCD显示屏并不断扫描4x4矩阵键盘。当按键被按下时,相应的键号会显示在LCD上。用户可以输入密码并进行验证,正确的密码会显示“OK PASS”,错误的密码会显示“ERR”。密码输入和验证,如果按下S1S10(键号110),则输入密码。如果按下S11(键号11),则确认密码是否正确。如果按下S12(键号12),则取消输入,重置密码。

#include <REGX52.H>
#include "LCD1602.h"

void DelayXms(unsigned int xms)	//@12.000MHz
{
	unsigned char data i, j;
	
	while(xms)
	{
		i = 2;
		j = 239;
		do
		{
			while (--j);
		} while (--i);
		xms--;
	}
}

unsigned char MatrixKey()
{
	unsigned char keyNumber = 0;

	P1 = 0xff;
	P1_3 = 0;//第一列低电平
	if(P1_7 == 0)
	{
		DelayXms(1);
		while(P1_7 == 0);
		DelayXms(1);
		keyNumber = 1;
	}
	if(P1_6 == 0)
	{
		DelayXms(1);
		while(P1_6 == 0);
		DelayXms(1);
		keyNumber = 5;
	}
	if(P1_5 == 0)
	{
		DelayXms(1);
		while(P1_5 == 0);
		DelayXms(1);
		keyNumber = 9;
	}
	if(P1_4 == 0)
	{
		DelayXms(1);
		while(P1_4 == 0);
		DelayXms(1);
		keyNumber = 13;
	}

	P1 = 0xff;
	P1_2 = 0;//第二列低电平
	if(P1_7 == 0)
	{
		DelayXms(1);
		while(P1_7 == 0);
		DelayXms(1);
		keyNumber = 2;
	}
	if(P1_6 == 0)
	{
		DelayXms(1);
		while(P1_6 == 0);
		DelayXms(1);
		keyNumber = 6;
	}
	if(P1_5 == 0)
	{
		DelayXms(1);
		while(P1_5 == 0);
		DelayXms(1);
		keyNumber = 10;
	}
	if(P1_4 == 0)
	{
		DelayXms(1);
		while(P1_4 == 0);
		DelayXms(1);
		keyNumber = 14;
	}

	P1 = 0xff;
	P1_1 = 0;//第三列低电平
	if(P1_7 == 0)
	{
		DelayXms(1);
		while(P1_7 == 0);
		DelayXms(1);
		keyNumber = 3;
	}
	if(P1_6 == 0)
	{
		DelayXms(1);
		while(P1_6 == 0);
		DelayXms(1);
		keyNumber = 7;
	}
	if(P1_5 == 0)
	{
		DelayXms(1);
		while(P1_5 == 0);
		DelayXms(1);
		keyNumber = 11;
	}
	if(P1_4 == 0)
	{
		DelayXms(1);
		while(P1_4 == 0);
		DelayXms(1);
		keyNumber = 15;
	}

	P1 = 0xff;
	P1_0 = 0;//第四列低电平
	if(P1_7 == 0)
	{
		DelayXms(1);
		while(P1_7 == 0);
		DelayXms(1);
		keyNumber = 4;
	}
	if(P1_6 == 0)
	{
		DelayXms(1);
		while(P1_6 == 0);
		DelayXms(1);
		keyNumber = 8;
	}
	if(P1_5 == 0)
	{
		DelayXms(1);
		while(P1_5 == 0);
		DelayXms(1);
		keyNumber = 12;
	}
	if(P1_4 == 0)
	{
		DelayXms(1);
		while(P1_4 == 0);
		DelayXms(1);
		keyNumber = 16;
	}

	return keyNumber;
}

void main()
{
	unsigned int password = 0;
	unsigned int countkey = 0;
	LCD_Init();
	LCD_ShowString(1,1,"Password:");

	while(1)
	{
		unsigned char temp = MatrixKey();
		if(temp)
		{
			if(temp<=10)//如果S1~S10按键按下,进行输入密码
			{
				if(countkey < 4)
				{
					password*=10;
					password+=temp%10;//获取一位密码
					countkey++;		
				}
				LCD_ShowNum(2,1,password,4);//更新密码
			}
			if(temp == 11)//S11确认
			{
				if(password == 2345)//2345定义为正确密码
				{
					LCD_ShowString(1,14,"O K");
					LCD_ShowString(2,13,"PASS");
				}
				else
				{
					LCD_ShowString(1,14,"ERR");
					password = 0;//初始密码
					countkey = 0;//初始次数
					LCD_ShowNum(2,1,password,4);//更新密码
				}
			}
			if(temp == 12)//S12取消键
			{
				password = 0;//初始密码
				countkey = 0;//初始次数
				LCD_ShowNum(2,1,password,4);//更新密码	
			}
		}
	}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/784846.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

萝卜快跑的狠活

萝卜快跑作为百度旗下的自动驾驶出行服务平台&#xff0c;在科技应用上展现了多项领先的技术。以下是萝卜快跑采用的一些主要科技“狠活”&#xff1a; 自动驾驶技术&#xff1a; 萝卜快跑主要使用了百度Apollo的L4级自动驾驶技术&#xff0c;该技术能够应对海量的城市道路场景…

Vue的常见指令

目录 1.v-bind 2. class绑定 3.style绑定 4.v-if/v-show 1.v-bind v-bind指令用于绑定属性 可以简写成 “ &#xff1a;” 它的作用就是我们可以动态的定义属性的值&#xff0c;比如常见的<img src "1.jpg"> 我们如果想要修改图片就需要获取到DOM对象&am…

Nginx:关于实现跨域代理

运维专题 Nginx&#xff1a;关于实现跨域代理 - 文章信息 - Author: 李俊才 (jcLee95) Visit me at CSDN: https://jclee95.blog.csdn.netMy WebSite&#xff1a;http://thispage.tech/Email: 291148484163.com. Shenzhen ChinaAddress of this article:https://blog.csdn.ne…

科普文:分布式系统的架构设计模式

一、分布式架构基本概念 分布式架构是一种计算机系统设计方法&#xff0c;它将一个复杂的系统划分为多个自治的组件或节点&#xff0c;并通过网络进行通信和协作。每个组件或节点在功能上可以相互独立&#xff0c;但又能够通过消息传递或共享数据来实现协同工作。分布式架构主要…

为什么独立站需要高质量的GPB外链?

独立站需要高质量的GPB外链&#xff0c;主要是因为它们能显著提升网站的可信度和可见性。高质量的外链相当于得到其他权威网站的认可和推荐&#xff0c;这会让搜索引擎认为你的内容有价值&#xff0c;从而提升你的搜索排名。试想一下&#xff0c;当其他有影响力的网站愿意链接到…

设计模式7-装饰模式

设计模式7-装饰模式 写在前面动机模式定义结构代码推导原始代码解决问题分析 选择装饰模式的理由1. 职责分离&#xff08;Single Responsibility Principle&#xff09;2. 动态扩展功能3. 避免类爆炸4. 开闭原则&#xff08;Open/Closed Principle&#xff09;5. 更好的组合复用…

如何忽略部分文件或者文件夹在git提交项目时

嗨&#xff0c;我是兰若&#xff0c;最近发现有些小伙伴在提交代码时&#xff0c;总是把不该提交的文件&#xff0c;比如说本地批跑的缓存文件给提交到了git上面&#xff0c;导致别人在拉取代码的时候&#xff0c;也会把这部分文件拉取到自己本地&#xff0c;从而导致和本地的缓…

深度学习(笔记内容)

1.国内镜像网站 pip使用清华源镜像源 pip install <库> -i https://pypi.tuna.tsinghua.edu.cn/simple/ pip使用豆瓣的镜像源 pip install <库> -i https://pypi.douban.com/simple/ pip使用中国科技大学的镜像源 pip install <库> -i https://pypi.mirro…

PyCharm如何安装requirements.txt中的依赖包

问题&#xff1a;下载别人的源码&#xff0c;如何安装代码中requirement.txt中的依赖包。 解决方案&#xff1a; &#xff08;1&#xff09;打开PyCharm下面的Terminal&#xff0c;先为代码创建单独的虚拟环境并进入到虚拟环境中&#xff08;每个项目单独的环境&#xff0c;这…

HINet: Half Instance Normalization Network for Image Restoration

论文&#xff1a;HINet: Half Instance Normalization Network for Image Restoration Abstract&#xff1a; 在本文中&#xff0c;我们探讨了实例归一化在低级视觉任务中的作用。 具体来说&#xff0c;我们提出了一个新颖的块&#xff1a;半实例归一化块&#xff08;HIN 块&…

Oracle 19c 统一审计表清理

zabbix 收到SYSAUX表空间告警超过90%告警&#xff0c;最后面给出的清理方法只适合ORACLE 统一审计表的清理&#xff0c;传统审计表的清理SYS.AUD$不适合&#xff0c;请注意。 SQL> Col tablespace_name for a30 Col used_pct for a10 Set line 120 pages 120 select total.…

Linux系统下的用户管理模式

Linux系统下的用户管理模式 本文以属于Linux系统基本概念&#xff0c;如果以查找教程教程&#xff0c;解决问题为主&#xff0c;只需要查看本文后半部分。 如需要系统性学习请查看本文前半部分。 文章目录 Linux系统下的用户管理模式1. Linux下用户的概念2. 创建不同类型的用户…

C++继承(一文说懂)

目录 一&#xff1a; &#x1f525;继承的概念及定义1.1 继承的概念1.2 继承定义1.2.1 定义格式1.2.2 继承关系和访问限定符1.2.3 继承基类成员访问方式的变化 二&#xff1a;&#x1f525;基类和派生类对象赋值转换三&#xff1a;&#x1f525;继承中的作用域四&#xff1a;&a…

SpringSecurity中文文档(Servlet Authorization Architecture )

Authorization 在确定了用户将如何进行身份验证之后&#xff0c;还需要配置应用程序的授权规则。 Spring Security 中的高级授权功能是其受欢迎的最有说服力的原因之一。无论您选择如何进行身份验证(无论是使用 Spring Security 提供的机制和提供者&#xff0c;还是与容器或其…

基于单片机的空调控制器的设计

摘 要 &#xff1a; 以单片机为核心的空调控制器因其体积小 、 成本低 、 功能强 、 简便易行而得到广泛应用 。 本设计通过 &#xff21;&#xff34;&#xff18;&#xff19;&#xff33;&#xff15;&#xff12; 控制&#xff24;&#xff33;&#xff11;&#xff18;&a…

基于YOLOv9的脑肿瘤区域检测

数据集 脑肿瘤区域检测&#xff0c;我们直接采用kaggle公开数据集&#xff0c;Br35H 数据中已对医学图像中脑肿瘤位置进行标注 数据集我已经按照YOLO格式配置好&#xff0c;数据内容如下 数据集中共包含700张图像&#xff0c;其中训练集500张&#xff0c;验证集200张 模型训…

全网最适合入门的面向对象编程教程:11 类和对象的Python实现-子类调用父类方法-模拟串口传感器和主机

全网最适合入门的面向对象编程教程&#xff1a;11 类和对象的 Python 实现-子类调用父类方法-模拟串口传感器和主机 摘要&#xff1a; 本节课&#xff0c;我们主要讲解了在 Python 类的继承中子类如何进行初始化、调用父类的属性和方法&#xff0c;同时讲解了模拟串口传感器和…

【问题记录】Nodeclub运行make install报错npm ERR! code ELIFECYCLE

问题展示 按照官网给出的教程进行到make install这一步卡住了&#xff0c;显示了如下报错。 解决方法 将node.js版本变更为能够识别代码的版本&#xff0c;我将版本修改成了16.14.0以后成功运行。 nvm install 16.14.0

Java设计模式---(创建型模式)工厂、单例、建造者、原型

目录 前言一、工厂模式&#xff08;Factory&#xff09;1.1 工厂方法模式&#xff08;Factory Method&#xff09;1.1.1 普通工厂方法模式1.1.2 多个工厂方法模式1.1.3 静态工厂方法模式 1.2 抽象工厂模式&#xff08;Abstract Factory&#xff09; 二、单例模式&#xff08;Si…

从零开始做题:好怪哦

题目 给出一个压缩文件 解题 方法1 01Edit打开&#xff0c;发现是个反着的压缩包&#xff08;末尾倒着的PK头&#xff09; import os# 目标目录路径 # target_directory /home/ai001/alpaca-lora# 切换到目标目录 # os.chdir(target_directory)# 打印当前工作目录以确认…