从0开始设计_基于STM32F1的RC522读写卡

1.介绍
看网上很多RC522的教程都是基于读卡ID的,这个对于很多应用来说其实没有什么用,最近刚好有个项目需要读写卡,而RC522又是非常常用的且不容易缺货的芯片,所以准备用RC522来进行读写卡。
2.设备准备
首先准备一个开发板和一个RC522模块,开发板这里我选择正点原子的精英板(STM32F103ZET6),具体如下板子如下图1所示。

接下来就是接线,我选择的是SPI2,对应的接线如下:
RST   -->  PC4
MISO -->  PB14
MOSI -->  PB15
SCK   -->  PB13
SDA   -->  PB0
上面是硬件名称的相应接口,对于SPI来说SDA就是SPI的CS(片选)线,记得RC522模块的供电采用3.3V,可别接成5V了。
3.工程配置
首先打开外部时钟,配置如下图2所示。

根据外部晶振配置对应的外部晶振频率,设置为最大的72MB。

配置SPI2,首先配置位数,频率,以及模式,片选采用软件方式。

接下来配置引脚,由于片选已经采用软件的方式,所以只需要配置MISO,MOSI和SCK了。

RST和CS直接采用GPIO的配置。

最后配置一下UART即可,选择115200波特率,引脚默认。

设置完成之后,所有引脚如图8所示。

4.程序编写
首先需要导入RC522的库,只有两个文件分别是【RC522.c】和【RC522.h】。
接下来修改RC522.c中的硬件接口,将SPI读写修改成如下代码。

#include "RC522.h"

//三目运算符true取前面那个
#define RS522_RST(N) HAL_GPIO_WritePin(RC522_RST_GPIO_Port, RC522_RST_Pin, N==1?GPIO_PIN_SET:GPIO_PIN_RESET)
#define RS522_NSS(N) HAL_GPIO_WritePin(RC522_CS_GPIO_Port, RC522_CS_Pin, N==1?GPIO_PIN_SET:GPIO_PIN_RESET)
#define osDelay HAL_Delay
/**************************************************************************************
* 函数名称:MFRC_Init
* 功能描述:MFRC初始化
* 入口参数:无
* 出口参数:无
* 返 回 值:无
* 说    明:MFRC的SPI接口速率为0~10Mbps
***************************************************************************************/
void MFRC_Init(void)
{
    RS522_NSS(1);
    RS522_RST(1);
}

/**************************************************************************************
* 函数名称: SPI_RW_Byte
* 功能描述: 模拟SPI读写一个字节
* 入口参数: -byte:要发送的数据
* 出口参数: -byte:接收到的数据
***************************************************************************************/
static uint8_t ret;       //些函数是HAL与标准库不同和地方,【读写函数】
uint8_t SPI2_RW_Byte(uint8_t byte)
{
    HAL_SPI_TransmitReceive(&hspi2, &byte, &ret, 1, 10);//把byte写入,并读出一个值 存入ret
    return   ret;                 //入口是byte的地址,读取时用的也是ret的地址;1:一次只写入一个值 10:timeout     
}   

/**************************************************************************************
* 函数名称:MFRC_WriteReg
* 功能描述:写一个寄存器
* 入口参数:-addr:待写的寄存器地址
*           -data:待写的寄存器数据
* 出口参数:无
* 返 回 值:无
* 说    明:无
***************************************************************************************/
void MFRC_WriteReg(uint8_t addr, uint8_t data)
{
    uint8_t AddrByte;
    AddrByte = (addr << 1 ) & 0x7E; //求出地址字节
    RS522_NSS(0);                   //NSS拉低
    SPI2_RW_Byte(AddrByte);         //写地址字节
    SPI2_RW_Byte(data);             //写数据
    RS522_NSS(1);                   //NSS拉高
}

/**************************************************************************************
* 函数名称:MFRC_ReadReg
* 功能描述:读一个寄存器
* 入口参数:-addr:待读的寄存器地址
* 出口参数:无
* 返 回 值:-data:读到寄存器的数据
* 说    明:无
***************************************************************************************/
uint8_t MFRC_ReadReg(uint8_t addr)
{
    uint8_t AddrByte, data;
    AddrByte = ((addr << 1 ) & 0x7E ) | 0x80;   //求出地址字节
    RS522_NSS(0);                               //NSS拉低
    SPI2_RW_Byte(AddrByte);                     //写地址字节
    data = SPI2_RW_Byte(0x00);                  //读数据
    RS522_NSS(1);                               //NSS拉高
    return data;
}

其他接口保持不变,我们来看一下RC522提供的接口和指令有哪些。

#ifndef _RC522_H
#define _RC522_H

//头文件
//************************************************
#include "gpio.h"//要一些引脚上的宏定义
#include "spi.h"//硬件SPI的定义
#include "printf.h"
#include "main.h"//Laber User上的宏定义
//************************************************

//MFRC522驱动程序
//************************************************

/*MFRC522寄存器定义*/
//PAGE0
#define MFRC_RFU00                      0x00   
#define MFRC_CommandReg                 0x01   
#define MFRC_ComIEnReg                     0x02   
#define MFRC_DivlEnReg                     0x03   
#define MFRC_ComIrqReg                     0x04   
#define MFRC_DivIrqReg                     0x05
#define MFRC_ErrorReg                      0x06   
#define MFRC_Status1Reg                    0x07   
#define MFRC_Status2Reg                    0x08   
#define MFRC_FIFODataReg                   0x09
#define MFRC_FIFOLevelReg                  0x0A
#define MFRC_WaterLevelReg                 0x0B
#define MFRC_ControlReg                    0x0C
#define MFRC_BitFramingReg                 0x0D
#define MFRC_CollReg                       0x0E
#define MFRC_RFU0F                         0x0F
//PAGE1     
#define MFRC_RFU10                         0x10
#define MFRC_ModeReg                       0x11
#define MFRC_TxModeReg                     0x12
#define MFRC_RxModeReg                     0x13
#define MFRC_TxControlReg                  0x14
#define MFRC_TxAutoReg                     0x15 //中文手册有误
#define MFRC_TxSelReg                      0x16
#define MFRC_RxSelReg                      0x17
#define MFRC_RxThresholdReg                0x18
#define MFRC_DemodReg                      0x19
#define MFRC_RFU1A                         0x1A
#define MFRC_RFU1B                         0x1B
#define MFRC_MifareReg                     0x1C
#define MFRC_RFU1D                         0x1D
#define MFRC_RFU1E                         0x1E
#define MFRC_SerialSpeedReg                0x1F
//PAGE2   
#define MFRC_RFU20                         0x20  
#define MFRC_CRCResultRegM                 0x21
#define MFRC_CRCResultRegL                 0x22
#define MFRC_RFU23                         0x23
#define MFRC_ModWidthReg                   0x24
#define MFRC_RFU25                         0x25
#define MFRC_RFCfgReg                      0x26
#define MFRC_GsNReg                        0x27
#define MFRC_CWGsCfgReg                    0x28
#define MFRC_ModGsCfgReg                   0x29
#define MFRC_TModeReg                      0x2A
#define MFRC_TPrescalerReg                 0x2B
#define MFRC_TReloadRegH                   0x2C
#define MFRC_TReloadRegL                   0x2D
#define MFRC_TCounterValueRegH             0x2E
#define MFRC_TCounterValueRegL             0x2F
//PAGE3      
#define MFRC_RFU30                         0x30
#define MFRC_TestSel1Reg                   0x31
#define MFRC_TestSel2Reg                   0x32
#define MFRC_TestPinEnReg                  0x33
#define MFRC_TestPinValueReg               0x34
#define MFRC_TestBusReg                    0x35
#define MFRC_AutoTestReg                   0x36
#define MFRC_VersionReg                    0x37
#define MFRC_AnalogTestReg                 0x38
#define MFRC_TestDAC1Reg                   0x39  
#define MFRC_TestDAC2Reg                   0x3A   
#define MFRC_TestADCReg                    0x3B   
#define MFRC_RFU3C                         0x3C   
#define MFRC_RFU3D                         0x3D   
#define MFRC_RFU3E                         0x3E   
#define MFRC_RFU3F                         0x3F

/*MFRC522的FIFO长度定义*/
#define MFRC_FIFO_LENGTH                       64

/*MFRC522传输的帧长定义*/
#define MFRC_MAXRLEN                18               

/*MFRC522命令集,中文手册P59*/
#define MFRC_IDLE                              0x00        //取消当前命令的执行
#define MFRC_CALCCRC                           0x03    //激活CRC计算
#define MFRC_TRANSMIT                          0x04    //发送FIFO缓冲区内容
#define MFRC_NOCMDCHANGE            0x07        //无命令改变
#define MFRC_RECEIVE                           0x08    //激活接收器接收数据
#define MFRC_TRANSCEIVE                        0x0C    //发送并接收数据
#define MFRC_AUTHENT                           0x0E    //执行Mifare认证(验证密钥)
#define MFRC_RESETPHASE                        0x0F    //复位MFRC522

/*MFRC522通讯时返回的错误代码*/
#define MFRC_OK                         (char)(0)
#define MFRC_NOTAGERR                    (char)(-1)
#define MFRC_ERR                        (char)(-2)

/*MFRC522函数声明*/
void MFRC_Init(void);
void MFRC_WriteReg(uint8_t addr, uint8_t data);
uint8_t MFRC_ReadReg(uint8_t addr);
void MFRC_SetBitMask(uint8_t addr, uint8_t mask);
void MFRC_ClrBitMask(uint8_t addr, uint8_t mask);
void MFRC_CalulateCRC(uint8_t *pInData, uint8_t len, uint8_t *pOutData);
char MFRC_CmdFrame(uint8_t cmd, uint8_t *pInData, uint8_t InLenByte, uint8_t *pOutData, uint16_t *pOutLenBit);
//********************************************************************

//MFRC552与MF1卡通讯接口程序
//*********************************************************************
/*Mifare1卡片命令字*/
#define PICC_REQIDL                   0x26                       //寻天线区内未进入休眠状态的卡
#define PICC_REQALL                   0x52                       //寻天线区内全部卡
#define PICC_ANTICOLL1                0x93                       //防冲撞
#define PICC_ANTICOLL2                0x95                       //防冲撞
#define PICC_AUTHENT1A                0x60                       //验证A密钥
#define PICC_AUTHENT1B                0x61                       //验证B密钥
#define PICC_READ                     0x30                       //读块
#define PICC_WRITE                    0xA0                       //写块
#define PICC_DECREMENT                0xC0                       //减值(扣除)
#define PICC_INCREMENT                0xC1                       //增值(充值)
#define PICC_TRANSFER                 0xB0                       //转存(传送)
#define PICC_RESTORE                  0xC2                       //恢复(重储)
#define PICC_HALT                     0x50                       //休眠

/*PCD通讯时返回的错误代码*/
#define PCD_OK                         (char)0                                //成功
#define PCD_NOTAGERR            (char)(-1)                        //无卡
#define PCD_ERR                        (char)(-2)                        //出错

/*PCD函数声明*/
void PCD_Init(void);//读写器初始化
void PCD_Reset(void);
void PCD_AntennaOn(void);
void PCD_AntennaOff(void);
char PCD_Request(uint8_t RequestMode, uint8_t *pCardType);  //寻卡,并返回卡的类型
char PCD_Anticoll(uint8_t *pSnr);                           //防冲突,返回卡号
char PCD_Select(uint8_t *pSnr);                             //选卡
char PCD_AuthState(uint8_t AuthMode, uint8_t BlockAddr, uint8_t *pKey, uint8_t *pSnr); //验证密码(密码A和密码B)   
char PCD_WriteBlock(uint8_t BlockAddr, uint8_t *pData);   //写数据
char PCD_ReadBlock(uint8_t BlockAddr, uint8_t *pData);    //读数据
char PCD_Value(uint8_t mode, uint8_t BlockAddr, uint8_t *pValue);   
char PCD_BakValue(uint8_t sourceBlockAddr, uint8_t goalBlockAddr);                                 
char PCD_Halt(void);
//******************************************************************************************

#endif

不过接下来我们需要测试一下,SPI是否正常,接上LOTO示波器OSCA02,最近出门在外,不方便带示波器,所以带了一个LOTO的便携示波器,不过他刚好有逻辑分析仪的功能,刚好测试一下它的性能,接线图如下,需要将ChA口接到时钟线上,这样才能执行触发功能。

然后将代码进入调试,进入读寄存器函数,在进入前打个断点,然后开启LOTO示波器的触发功能,然后运行到读取结束,可以看到读取到了【0x83】这个值。

再来看看逻辑分析仪读取到的值,可以看到也是【0x83】,说明这个逻辑分析仪性能还行。

SPI功能测试完了,接下来就要进行读写卡了。首先科普一下读写卡的整个过程【寻卡-》放冲撞-》选卡-》解密卡-》读/写卡】。
按照上面的流程,调用相关的函数,整体代码如下。

/* USER CODE BEGIN Header */
/**
  ******************************************************************************
  * [url=home.php?mod=space&uid=288409]@file[/url]           : main.c
  * [url=home.php?mod=space&uid=247401]@brief[/url]          : Main program body
  ******************************************************************************
  * @attention
  *
  * <h2><center> Copyright (c) 2021 STMicroelectronics.
  * All rights reserved.</center></h2>
  *
  * This software component is licensed by ST under BSD 3-Clause license,
  * the "License"; You may not use this file except in compliance with the
  * License. You may obtain a copy of the License at:
  *                        opensource.org/licenses/BSD-3-Clause
  *
  ******************************************************************************
  */
/* USER CODE END Header */
/* Includes ------------------------------------------------------------------*/
#include "main.h"
#include "spi.h"
#include "usart.h"
#include "gpio.h"

/* Private includes ----------------------------------------------------------*/
/* USER CODE BEGIN Includes */
#include "delay.h"
#include "printf.h"
#include "rc522.h"
#include "string.h"
/* USER CODE END Includes */

/* Private typedef -----------------------------------------------------------*/
/* USER CODE BEGIN PTD */

/* USER CODE END PTD */

/* Private define ------------------------------------------------------------*/
/* USER CODE BEGIN PD */
/* USER CODE END PD */

/* Private macro -------------------------------------------------------------*/
/* USER CODE BEGIN PM */

/* USER CODE END PM */

/* Private variables ---------------------------------------------------------*/

/* USER CODE BEGIN PV */

/* USER CODE END PV */

/* Private function prototypes -----------------------------------------------*/
void SystemClock_Config(void);
/* USER CODE BEGIN PFP */

/* USER CODE END PFP */

/* Private user code ---------------------------------------------------------*/
/* USER CODE BEGIN 0 */
uint8_t key_A[6] = {0xFF,0xFF,0xFF,0xFF,0xFF,0xFF};
char *WriteData = {"1234567890ABCDEF"};
char ReadData[16] = {0};
/* USER CODE END 0 */

/**
  * @brief  The application entry point.
  * @retval int
  */
int main(void)
{
  /* USER CODE BEGIN 1 */
  char pcd_err = 0;
  /* USER CODE END 1 */

  /* MCU Configuration--------------------------------------------------------*/

  /* Reset of all peripherals, Initializes the Flash interface and the Systick. */
  HAL_Init();

  /* USER CODE BEGIN Init */

  /* USER CODE END Init */

  /* Configure the system clock */
  SystemClock_Config();

  /* USER CODE BEGIN SysInit */

  /* USER CODE END SysInit */

  /* Initialize all configured peripherals */
  MX_GPIO_Init();
  MX_USART1_UART_Init();
  MX_SPI2_Init();
  /* USER CODE BEGIN 2 */
  //初始化
  //*************************
  PCD_Init();//RC522初始化
  //*************************
  //全局变量
  //*************************
  uint8_t RxBuffer[4];
  char Card_ID[8];
  //*************************
  /* USER CODE END 2 */

  /* Infinite loop */
  /* USER CODE BEGIN WHILE */
  while (1)
  {
    /* USER CODE END WHILE */

    /* USER CODE BEGIN 3 */
    pcd_err = PCD_Request(PICC_REQIDL, RxBuffer);//返回值为0,代表寻卡成功;并把卡类型存入RxBuffer中
    if(pcd_err == PCD_OK)
    {
        uint16_t cardType = (RxBuffer[0] << 8) | RxBuffer[1];
        
        printf("卡类型:0x%04X\r\n", cardType);
        
        pcd_err = PCD_Anticoll(RxBuffer);   //防冲撞,完成这部就可以简单地 读取卡号,本次不涉及更高层次应用
        if(pcd_err == PCD_OK)
        {
            sprintf(Card_ID,"%x%x%x%x",RxBuffer[0],RxBuffer[1],RxBuffer[2],RxBuffer[3]);
            printf("ID=%s\r\n",Card_ID);
        }
        
        pcd_err = PCD_Select((uint8_t *)RxBuffer);       //选卡
        if(pcd_err == PCD_OK)
        {
            printf("Select Card OK\r\n");
        }
        else
        {
            printf("Select Card Error\r\n");
        }
        
        pcd_err = PCD_AuthState(PICC_AUTHENT1A, 5, key_A, RxBuffer);    //解密
        if(pcd_err == PCD_OK)
        {
            printf("Auth Card OK\r\n");
        }
        else
        {
            printf("Auth Card Error\r\n");
        }
        
        pcd_err = PCD_WriteBlock(6, (uint8_t *)WriteData);  //写卡
        if(pcd_err == PCD_OK)
        {
            printf("写卡成功\r\n");
        }
        else
        {
            printf("写卡失败:%d\r\n",pcd_err);
        }
        
        HAL_Delay(1);
        pcd_err = PCD_ReadBlock(6, (uint8_t *)ReadData);    //读卡
        if(pcd_err == PCD_OK)
        {
            printf("读卡成功:%s\r\n", ReadData);
        }
        else
        {
            printf("读卡失败:%d\r\n",pcd_err);
        }
        
        PCD_Halt();
        
        memset(RxBuffer, 0, sizeof(RxBuffer));//清空字符串,这里要清除RxBuffer才行
        HAL_Delay(1000);
    }
    HAL_Delay(100);
  }
  /* USER CODE END 3 */
}

/**
  * @brief System Clock Configuration
  * @retval None
  */
void SystemClock_Config(void)
{
  RCC_OscInitTypeDef RCC_OscInitStruct = {0};
  RCC_ClkInitTypeDef RCC_ClkInitStruct = {0};

  /** Initializes the RCC Oscillators according to the specified parameters
  * in the RCC_OscInitTypeDef structure.
  */
  RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE;
  RCC_OscInitStruct.HSEState = RCC_HSE_ON;
  RCC_OscInitStruct.HSEPredivValue = RCC_HSE_PREDIV_DIV1;
  RCC_OscInitStruct.HSIState = RCC_HSI_ON;
  RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON;
  RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSE;
  RCC_OscInitStruct.PLL.PLLMUL = RCC_PLL_MUL9;
  if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK)
  {
    Error_Handler();
  }
  /** Initializes the CPU, AHB and APB buses clocks
  */
  RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK|RCC_CLOCKTYPE_SYSCLK
                              |RCC_CLOCKTYPE_PCLK1|RCC_CLOCKTYPE_PCLK2;
  RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK;
  RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1;
  RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV2;
  RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1;

  if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_2) != HAL_OK)
  {
    Error_Handler();
  }
}

/* USER CODE BEGIN 4 */

/* USER CODE END 4 */

/**
  * @brief  This function is executed in case of error occurrence.
  * @retval None
  */
void Error_Handler(void)
{
  /* USER CODE BEGIN Error_Handler_Debug */
  /* User can add his own implementation to report the HAL error return state */
  __disable_irq();
  printf("error\r\n");
  while (1)
  {
  }
  /* USER CODE END Error_Handler_Debug */
}

#ifdef  USE_FULL_ASSERT
/**
  * @brief  Reports the name of the source file and the source line number
  *         where the assert_param error has occurred.
  * @param  file: pointer to the source file name
  * @param  line: assert_param error line source number
  * @retval None
  */
void assert_failed(uint8_t *file, uint32_t line)
{
  /* USER CODE BEGIN 6 */
  /* User can add his own implementation to report the file name and line number,
     ex: printf("Wrong parameters value: file %s on line %d\r\n", file, line) */
  /* USER CODE END 6 */
}
#endif /* USE_FULL_ASSERT */

/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/

代码是先进**,然后进**,再进**。关于解密,卡默认的密码是【FFFFFFFFFFFF】,一共是6个【0xFF】。
最终的输出结果如下图12所示,写入【“1234567890ABCDEF”】内容,读出的也是【“1234567890ABCDEF”】内容。

5.总结
SPI配置下来还是比较简单的,这个工程最主要的还是得读懂RC522的工作流程,如果能对IC卡进行读写,项目的基本功能就实现了,后续只要调用相关的接口接可以了,这次的分享就到这里了!

从0开始设计_基于STM32F1的RC522读写卡的更多相关文章

  1. 深度学习与计算机视觉(11)_基于deep learning的快速图像检索系统

    深度学习与计算机视觉(11)_基于deep learning的快速图像检索系统 作者:寒小阳 时间:2016年3月. 出处:http://blog.csdn.net/han_xiaoyang/arti ...

  2. Jsp标签字典开发_基于Spring+Hibernate

    目录 1. Jsp标签字典开发_基于Spring+Hibernate 1.1. 简述 1.2. 定义DictItem实体 1.3. 定义字典的@interface 1.4. 定义字典缓存类 1.5. ...

  3. 基于STM32F1与NRF24L01模块的SPI简单通信

    一.前言 1.简介: 本文是基于STM32F1,将数据发送至NRF模块的寄存器,并将数据重新读取,通过串口发送出来的简单SPI单通信. 2.SPI简介: 调过STM8的都已经对SPI有所了解,调法都一 ...

  4. 基于STM32F1 的BASIC解码实验 vb basic 液晶显示执行过程及结果

    基于STM32F1 的BASIC解码实验 1.basic程序以文件形式存储 2.程序文件存储在sd卡 3.解释结果显示在液晶屏上 主函数部分 int main(void){ u16 i,j; dela ...

  5. GPS部标平台的架构设计(十)-基于Asp.NET MVC构建GPS部标平台

    在当前很多的GPS平台当中,有很多是基于asp.NET+siverlight开发的遗留项目,代码混乱而又难以维护,各种耦合和关联,要命的是界面也没见到比Javascript做的控件有多好看,随着需求的 ...

  6. 菜鸟学自动化测试(八)----selenium 2.0环境搭建(基于maven)

    菜鸟学自动化测试(八)----selenium 2.0环境搭建(基于maven) 2012-02-04 13:11 by 虫师, 11419 阅读, 5 评论, 收藏, 编辑 之前我就讲过一种方试来搭 ...

  7. YY语音从4.0版本开始是基于Qt的开发过程,以及碰到的问题

    作者:姚冬链接:http://www.zhihu.com/question/21359230/answer/20127715来源:知乎著作权归作者所有,转载请联系作者获得授权. YY语音从4.0版本开 ...

  8. (转)OAuth 2.0的设计思路

    OAuth是一个关于授权(authorization)的开放网络标准,在全世界得到广泛应用,目前的版本是2.0版. 本文对OAuth 2.0的设计思路和运行流程,做一个简明通俗的解释,主要参考材料为R ...

  9. Magicodes.IE 3.0重磅设计畅谈

    总体设计 Magicodes.IE导入导出通用库,支持Dto导入导出.模板导出.花式导出以及动态导出,支持Excel.Csv.Word.Pdf和Html. IE在去年年底重构一次之后,经过这么长时间的 ...

  10. 支持 gRPC 长链接,深度解读 Nacos 2.0 架构设计及新模型

    支持 gRPC 长链接,深度解读 Nacos 2.0 架构设计及新模型 原创 杨翊(席翁) 阿里巴巴云原生 2020-12-28    

随机推荐

  1. ES6学习 第四章 字符串的新增方法

    前言 本章介绍字符串对象的新增方法.不常用的方法不做重点笔记. 本章原文链接:字符串的新增方法 includes().startsWith().endsWith() 确定一个字符串是否包含在另一个字符 ...

  2. webgl 系列

    webgl 背景 工作所需... 目录 初识 WebGL 绘制一个点 三角形 变换矩阵和动画 渐变三角形 绘制猫 着色器语言

  3. let与const

    let与const ES2015(ES6)新增加了两个重要的JavaScript关键字: let和const. 块级作用域 代码块内如果存在let或者const,代码块会对这些命令声明的变量从块的开始 ...

  4. centos7源码方式安装zabbix-4.0

    1.关闭防火墙 systemctl stop firewalld.service #临时关闭firewall systemctl disable firewalld.service #禁止firewa ...

  5. Java I/O 教程(七) DataOutputStream和DataInputStream

    Java DataOutputStream Class Java DataOutputStream class 可以以机器无关方式往指定输出流写入Java原始数据类型,例如int, double, l ...

  6. Qt实用技巧:QCustomPlot做北斗GPS显示绝对位置运动轨迹和相对位置运动轨迹图的时,使图按照输入点顺序连曲线

    需求   使用QCustomPlot绘制多个目标的北斗运行轨迹图,包括累计绝对位置图和记录时刻的相对位置图.  当前绘制存在问题:    交付客户前,公司内部自测流程发现的问题.  实际预期效果为:  ...

  7. 【LeetCode二叉树#03】翻转二叉树的几种方法

    翻转二叉树 力扣题目链接(opens new window) 翻转一棵二叉树. 这道题目背后有一个让程序员心酸的故事,听说 Homebrew的作者Max Howell,就是因为没在白板上写出翻转二叉树 ...

  8. 【Azure 存储服务】使用PowerShell脚本创建存储账号(Storage Account)的共享访问签名(SASToken) : New-AzStorageContainerSASToken

    问题描述 使用PowerShell脚本如何来创建存储账号(Storage Account)的共享访问签名呢?查询到可以使用 New-AzStorageContainerSASToken 命令来生成Az ...

  9. NET项目&DLL反编译&MSSQL监控&VS搜索&注入&上传

    知识点 1.NET普通源码&编译源码 2.DLL反编译&后缀文件&指向 3.代码审计-SQL注入&文件上传 ASPX文件 -> CS ASPX.CS DLL反编译 ...

  10. 探索浏览器录屏Web API 接口的应用前景与限制

    一.浏览器录屏Web API 接口的优点: 简化录屏流程:浏览器录屏Web API 接口可以直接在网页中调用,无需安装额外的插件或软件,简化了录屏的流程. 实时录制与传输:Web API 接口可以实时 ...