普冉PY32系列(十一) 基于PY32F002A的6+1通道遥控小车II - 控制篇

news2024/9/22 15:37:23

目录

  • 普冉PY32系列(一) PY32F0系列32位Cortex M0+ MCU简介
  • 普冉PY32系列(二) Ubuntu GCC Toolchain和VSCode开发环境
  • 普冉PY32系列(三) PY32F002A资源实测 - 这个型号不简单
  • 普冉PY32系列(四) PY32F002A/003/030的时钟设置
  • 普冉PY32系列(五) 使用JLink RTT代替串口输出日志
  • 普冉PY32系列(六) 通过I2C接口驱动PCF8574扩展的1602LCD
  • 普冉PY32系列(七) SOP8,SOP10,SOP16封装的PY32F002A/PY32F003管脚复用
  • 普冉PY32系列(八) GPIO模拟和硬件SPI方式驱动无线收发芯片XN297LBW
  • 普冉PY32系列(九) GPIO模拟和硬件SPI方式驱动无线收发芯片XL2400
  • 普冉PY32系列(十) 基于PY32F002A的6+1通道遥控小车I - 综述篇
  • 普冉PY32系列(十一) 基于PY32F002A的6+1通道遥控小车II - 控制篇
  • 普冉PY32系列(十二) 基于PY32F002A的6+1通道遥控小车III - 驱动篇

基于PY32F002A的6+1通道遥控小车II - 控制篇

这篇继续介绍6+1通道遥控小车的控制端, 关于遥控手柄的硬件和软件设计的说明

PCB实物

正面

在嘉立创下单了PCB, 收到的是这个样子的.

  • PCB中二极管的位置稍微偏上, 存在与螺丝短接的风险, 在新的PCB设计中已经将其下移.
  • 无线模块的天线没有覆漆, 在LCEDA中不知道怎么修改. PCB做出来是焊盘的效果(上锡了), 不影响使用.

背面

因为空间限制, PY32F002A和74HC595/165都放到了背面

分割后的各个模块

遥控面板成品

遥控面板的焊接过程运气不错, 从贴片到接插件都是一次成功, 没有返工.

正面

  • 空间限制, 只比一张名片稍微大点, 布局比较局促.
  • LCD因为是裸片没有托板, 和背光板一起是用热熔胶直接固定在PCB上的.

LCD试车, 显示没问题

背面

  • 正面基本上全是接插件, 如果PY32F002A放到这面, 将来万一烧坏更换非常麻烦, 所以贴片元件都放到了背面
  • 电源接口用的是XH2.54
  • LCD背光担心电流过大, 补焊串了一颗1KR的电阻

LCD控制界面

这是最终的LCD控制界面

  • 上面两道横杆代表旋钮的模拟量
  • 中间和下方的四道横杆代表摇杆的模拟量
  • 两边的6个数字代表了模拟量的数值, 都是8bit, 从0 - 255
  • 下方的8个方格代表了8个开关量, 高亮(黑)代表按键按下(低电压), 正常(白)代表按键松开(高电压)

软件设计

整体结构

因为只考虑发送, 所以控制端的流程较为简单, 做一个大循环肯定可行, 采集数据 -> 发送数据 -> 采集数据 -> 发送数据. 如果要提升大循环的效率, 因为LCD显示和无线发送共用SPI, 需要保留在大循环, ADC可以用定时器触发做成DMA, 节省出ADC的时间.

最终使用的执行流程是

  • 使用一个uint8_t pad_state[8]存储6+1通道的数据
  • ADC使用定时器触发, 通过DMA存储转换结果到6个双字节内存地址, ADC DMA转换完成后
    • 将结果转为8bit, 存入 pad_state,
    • 收集74HC165的按键状态, 合成一个byte 也存入pad_state
    • 计算CRC并存至 pad_state 最后一个字节
  • 外层大循环读取 pad_state
    • 更新LCD显示
    • 通过无线发送数据

主循环

int main(void)
{
  // ...

  /* Infinite loop */
  while(1)
  {
    // 更新LCD显示
    DRV_Display_Update(pad_state);
    // 发送
    wireless_tx++;
    if (XL2400_Tx(pad_state, XL2400_PLOAD_WIDTH) == 0x20)
    {
      wireless_tx_succ++;
    }
    // 每 255 次发送, 打印一次成功次数, 用于标识成功率
    if (wireless_tx == 0xFF)
    {
      wireless_state[10] = wireless_tx_succ;
      DEBUG_PRINTF("TX_SUCC: %02X\r\n", wireless_tx_succ);
      wireless_tx = 0;
      wireless_tx_succ = 0;
    }
    // 延迟可以调节
    LL_mDelay(20);
  }
}

DMA中断

void DMA1_Channel1_IRQHandler(void)
{
  uint8_t crc = 0;
  if (LL_DMA_IsActiveFlag_TC1(DMA1) == 1)
  {
    LL_DMA_ClearFlag_TC1(DMA1);
    // 转换DMA读数为uint8_t并存入pad_state
    for (uint8_t i = 0; i < 6; i++)
    {
      pad_state[i] = (uint8_t)(*(adc_dma_data + i) >> 4);
      crc += pad_state[i];
    }
    // 从 74HC165 读取按键状态
    pad_state[6] = HC165_Read();
    // 存入CRC结果
    pad_state[7] = crc + pad_state[6];
  }
}

无线通讯

无线部分使用的是硬件SPI驱动的 XL2400, 代码可以参考
https://github.com/IOsetting/py32f0-template/tree/main/Examples/PY32F0xx/LL/SPI/XL2400_Wireless

传输的数据格式为固定长度8字节

#define XL2400_PLOAD_WIDTH       8   // Payload width

其中字节[0, 5]为6个ADC采集的数值结果, 字节[6]为74HC165采集的按键结果, 字节[7]为CRC校验.

收发的地址是固定的(将来需要改进)

const uint8_t TX_ADDRESS[5] = {0x11,0x33,0x33,0x33,0x11};
const uint8_t RX_ADDRESS[5] = {0x33,0x55,0x33,0x44,0x33};

输入采集

ADC采集

DMA初始化

void MSP_DMA_Config(void)
{
  LL_AHB1_GRP1_EnableClock(LL_AHB1_GRP1_PERIPH_DMA1);
  LL_APB1_GRP2_EnableClock(LL_APB1_GRP2_PERIPH_SYSCFG);

  // Remap ADC to LL_DMA_CHANNEL_1
  LL_SYSCFG_SetDMARemap_CH1(LL_SYSCFG_DMA_MAP_ADC);
  // Transfer from peripheral to memory
  LL_DMA_SetDataTransferDirection(DMA1, LL_DMA_CHANNEL_1, LL_DMA_DIRECTION_PERIPH_TO_MEMORY);
  // Set priority
  LL_DMA_SetChannelPriorityLevel(DMA1, LL_DMA_CHANNEL_1, LL_DMA_PRIORITY_HIGH);
  // Circular mode
  LL_DMA_SetMode(DMA1, LL_DMA_CHANNEL_1, LL_DMA_MODE_CIRCULAR);
  // Peripheral address no increment
  LL_DMA_SetPeriphIncMode(DMA1, LL_DMA_CHANNEL_1, LL_DMA_PERIPH_NOINCREMENT);
  // Memory address increment
  LL_DMA_SetMemoryIncMode(DMA1, LL_DMA_CHANNEL_1, LL_DMA_MEMORY_INCREMENT);
  // Peripheral data alignment : 16bit
  LL_DMA_SetPeriphSize(DMA1, LL_DMA_CHANNEL_1, LL_DMA_PDATAALIGN_HALFWORD);
  // Memory data alignment : 16bit
  LL_DMA_SetMemorySize(DMA1, LL_DMA_CHANNEL_1, LL_DMA_MDATAALIGN_HALFWORD);
  // Data length
  LL_DMA_SetDataLength(DMA1, LL_DMA_CHANNEL_1, 6);
  // Sorce and target address
  LL_DMA_ConfigAddresses(DMA1, LL_DMA_CHANNEL_1, (uint32_t)&ADC1->DR, (uint32_t)adc_dma_data, LL_DMA_GetDataTransferDirection(DMA1, LL_DMA_CHANNEL_1));
  // Enable DMA channel 1
  LL_DMA_EnableChannel(DMA1, LL_DMA_CHANNEL_1);
  // Enable transfer-complete interrupt
  LL_DMA_EnableIT_TC(DMA1, LL_DMA_CHANNEL_1);

  NVIC_SetPriority(DMA1_Channel1_IRQn, 0);
  NVIC_EnableIRQ(DMA1_Channel1_IRQn);
}

ADC初始化

void MSP_ADC_Init(void)
{
  __IO uint32_t backup_setting_adc_dma_transfer = 0;

  LL_APB1_GRP2_EnableClock(LL_APB1_GRP2_PERIPH_ADC1);

  LL_ADC_Reset(ADC1);
  // Calibrate start
  if (LL_ADC_IsEnabled(ADC1) == 0)
  {
    /* Backup current settings */
    backup_setting_adc_dma_transfer = LL_ADC_REG_GetDMATransfer(ADC1);
    /* Turn off DMA when calibrating */
    LL_ADC_REG_SetDMATransfer(ADC1, LL_ADC_REG_DMA_TRANSFER_NONE);
    LL_ADC_StartCalibration(ADC1);

    while (LL_ADC_IsCalibrationOnGoing(ADC1) != 0);

    /* Delay 1ms(>= 4 ADC clocks) before re-enable ADC */
    LL_mDelay(1);
    /* Apply saved settings */
    LL_ADC_REG_SetDMATransfer(ADC1, backup_setting_adc_dma_transfer);
  }
  // Calibrate end

  /* PA0 ~ PA5 as ADC input */
  LL_GPIO_SetPinMode(GPIOA, LL_GPIO_PIN_0, LL_GPIO_MODE_ANALOG);
  LL_GPIO_SetPinMode(GPIOA, LL_GPIO_PIN_1, LL_GPIO_MODE_ANALOG);
  LL_GPIO_SetPinMode(GPIOA, LL_GPIO_PIN_2, LL_GPIO_MODE_ANALOG);
  LL_GPIO_SetPinMode(GPIOA, LL_GPIO_PIN_3, LL_GPIO_MODE_ANALOG);
  LL_GPIO_SetPinMode(GPIOA, LL_GPIO_PIN_4, LL_GPIO_MODE_ANALOG);
  LL_GPIO_SetPinMode(GPIOA, LL_GPIO_PIN_5, LL_GPIO_MODE_ANALOG);
  /* Set ADC channel and clock source when ADEN=0, set other configurations when ADSTART=0 */
  LL_ADC_SetCommonPathInternalCh(__LL_ADC_COMMON_INSTANCE(ADC1), LL_ADC_PATH_INTERNAL_NONE);

  LL_ADC_SetClock(ADC1, LL_ADC_CLOCK_SYNC_PCLK_DIV2);
  LL_ADC_SetResolution(ADC1, LL_ADC_RESOLUTION_12B);
  LL_ADC_SetDataAlignment(ADC1, LL_ADC_DATA_ALIGN_RIGHT);
  LL_ADC_SetLowPowerMode(ADC1, LL_ADC_LP_MODE_NONE);
  LL_ADC_SetSamplingTimeCommonChannels(ADC1, LL_ADC_SAMPLINGTIME_41CYCLES_5);

  /* Set TIM1 as trigger source */
  LL_ADC_REG_SetTriggerSource(ADC1, LL_ADC_REG_TRIG_EXT_TIM1_TRGO);
  LL_ADC_REG_SetTriggerEdge(ADC1, LL_ADC_REG_TRIG_EXT_RISING);
  /* Single conversion mode (CONT = 0, DISCEN = 0), performs a single sequence of conversions, converting all the channels once */
  LL_ADC_REG_SetContinuousMode(ADC1, LL_ADC_REG_CONV_SINGLE);

  LL_ADC_REG_SetDMATransfer(ADC1, LL_ADC_REG_DMA_TRANSFER_UNLIMITED);
  LL_ADC_REG_SetOverrun(ADC1, LL_ADC_REG_OVR_DATA_OVERWRITTEN);
  /* Enable: each conversions in the sequence need to be triggerred separately */
  LL_ADC_REG_SetSequencerDiscont(ADC1, LL_ADC_REG_SEQ_DISCONT_DISABLE);
  /* Set channel 0/1/2/3/4/5 */
  LL_ADC_REG_SetSequencerChannels(ADC1, LL_ADC_CHANNEL_0 | LL_ADC_CHANNEL_1 | LL_ADC_CHANNEL_2 | LL_ADC_CHANNEL_3 | LL_ADC_CHANNEL_4 | LL_ADC_CHANNEL_5);

  LL_ADC_Enable(ADC1);

  // Start ADC regular conversion
  LL_ADC_REG_StartConversion(ADC1);
}

用于触发ADC的TIM1定时器初始化

void MSP_TIM1_Init(void)
{
  LL_TIM_InitTypeDef TIM1CountInit = {0};

  // RCC_APBENR2_TIM1EN == LL_APB1_GRP2_PERIPH_TIM1 
  LL_APB1_GRP2_EnableClock(RCC_APBENR2_TIM1EN);
  
  TIM1CountInit.ClockDivision       = LL_TIM_CLOCKDIVISION_DIV1;
  TIM1CountInit.CounterMode         = LL_TIM_COUNTERMODE_UP;
  // 系统时钟48MHz, 预分频8K, 预分频后定时器时钟为6KHz
  TIM1CountInit.Prescaler           = (SystemCoreClock / 6000) - 1;
  // 每600次计数一个周期, 每秒10个周期, 可以减小数值提高频率
  TIM1CountInit.Autoreload          = 600 - 1;
  TIM1CountInit.RepetitionCounter   = 0;
  LL_TIM_Init(TIM1, &TIM1CountInit);
  /* Triggered by update */
  LL_TIM_SetTriggerOutput(TIM1, LL_TIM_TRGO_UPDATE);
  LL_TIM_EnableCounter(TIM1);
}

开关量采集

74HC165的状态读取

uint8_t HC165_Read(void)
{
    uint8_t i, data = 0;

    HC165_LD_LOW;  // Pull down LD to load parallel inputs
    HC165_LD_HIGH; // Pull up to inhibit parallel loading

    for (i = 0; i < 8; i++)
    {
        data = data << 1;
        HC165_SCK_LOW;
        HC165_NOP; // NOP to ensure reading correct value
        if (HC165_DATA_READ)
        {
            data |= 0x01;
        }
        HC165_SCK_HIGH;
    }
    return data;
}

74HC165的示例代码, 可以参考 https://github.com/IOsetting/py32f0-template/tree/main/Examples/PY32F0xx/LL/GPIO/74HC165_8bit_Parallel_In_Serial_Out

LCD显示

PY32F002A驱动ST7567的示例代码可以参考 Examples/PY32F0xx/LL/SPI/ST7567_128x64LCD, 但是这个示例, 包括GitHub上可以搜到的其它示例, 都是使用 128 x 8 的内存作为显示缓存, 通过读写这块缓存再将缓存内容写入 ST7567 实现的显示内容更新. 这种方式可以实现非常灵活的显示, 缺点就是需要占用1KB的内存. 对于STM32F103这类有16KB或20KB内存的控制器, 1KB内存不算什么, 但是 PY32F002A 只有4KB内存, 1KB就值得考虑一下了. 因为遥控部分的数显, 显示格式相对固定, page之间可以相互独立, 没有相互交叠的部分, 启动后只需要显示滑动条和读数, 因此完全可以采用直接输出的方式.

换成直接输出后就变成这样的显示函数了, 定制LCD显示是比较费时费事的一步.

移动光标到坐标

void ST7567_SetCursor(uint8_t page, uint8_t column)
{
    ST7567_WriteCommand(ST7567_SET_PAGE_ADDRESS | (page & ST7567_SET_PAGE_ADDRESS_MASK));
    ST7567_WriteCommand(ST7567_SET_COLUMN_ADDRESS_MSB | ((column + ST7567_X_OFFSET) >> 4));
    ST7567_WriteCommand(ST7567_SET_COLUMN_ADDRESS_LSB | ((column + ST7567_X_OFFSET) & 0x0F));
}

指定宽度和偏移量, 填入固定内容

static void DRV_DrawRepeat(uint8_t symbol, uint8_t width, uint8_t offset, uint8_t colorInvert)
{
  symbol = symbol << offset;
  symbol = colorInvert? ~symbol : symbol;
  ST7567_TransmitRepeat(symbol, width);
}

画出横条

static void DRV_DrawHorizBar(uint8_t page, uint8_t column, uint8_t size)
{
  ST7567_SetCursor(page, column);
  DRV_DrawRepeat(0x7E, 1, 0, 0);
  DRV_DrawRepeat(0x42, size, 0, 0);
  DRV_DrawRepeat(0x7E, 1, 0, 0);
}

在横条中画出高亮滑块

static void DRV_DrawHorizBarCursor(uint8_t page, uint8_t column, uint8_t value, uint8_t barWidth, uint8_t cursorWidth, uint8_t direction)
{
  value = direction? value : 255 - value;
  ST7567_SetCursor(page, column + 1);
  DRV_DrawRepeat(0x42, barWidth, 0, 0);
  ST7567_SetCursor(page, column + 1 + (value * (barWidth - cursorWidth) / 255));
  DRV_DrawRepeat(0x7E, cursorWidth, 0, 0);
}

画出竖条和竖条光标的方法更复杂, 这里就不贴代码了.

在main函数的while循环中, 每次会更新LCD显示

void DRV_Display_Update(uint8_t *state)
{
  // 更新按键显示
  DRV_DrawKeyState(*(state + 6));
  // 更新4个横条的显示
  DRV_DrawHorizBarCursor(0, 10, *(state + 4), 50, 4, 0);
  DRV_DrawHorizBarCursor(0, 65, *(state + 5), 50, 4, 1);
  DRV_DrawHorizBarCursor(7,  0, *(state + 1), 60, 4, 0);
  DRV_DrawHorizBarCursor(7, 65, *(state + 2), 60, 4, 1);
  // 更新2个竖条显示, 因为竖条处于多个page, 每次更新显示都需要全部重绘
  DRV_DrawVertiBar(0, 1, 52);
  DRV_DrawVertiBarCursor(0, 1, 52, *(state + 0), 4, 0);
  DRV_DrawVertiBar(121, 1, 52);
  DRV_DrawVertiBarCursor(121, 1, 52, *(state + 3), 4, 1);
  // 输出6个模拟通道的数值(0 ~ 255)
  DRV_DrawNumber(1, 10, *(state + 4));
  DRV_DrawNumber(1, 100, *(state + 5));

  DRV_DrawNumber(4, 10, *(state + 0));
  DRV_DrawNumber(4, 100, *(state + 3));

  DRV_DrawNumber(5, 10, *(state + 1));
  DRV_DrawNumber(5, 100, *(state + 2));
}

用直接写入的方式, 在不开JLink RTT 的情况下, 整机内存只需要不到400个字节, 资源节约效果明显.

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

电脑磁盘怎么设置密码?磁盘加密软件哪个好?

电脑磁盘经常被用于存放各种重要数据&#xff0c;而为了避免数据泄露&#xff0c;我们需要为磁盘设置密码&#xff0c;以防止未授权人员使用磁盘。那么&#xff0c;电脑磁盘怎么设置密码呢&#xff1f;下面我们就一起来了解一下。 如何设置磁盘密码&#xff1f; 想要为磁盘设置…

腾讯云服务器99元一年?假的,阿里云是99元

腾讯云服务器99元一年是真的吗&#xff1f;假的&#xff0c;不用99元&#xff0c;只要88元即可购买一台2核2G3M带宽的轻量应用服务器&#xff0c;99元太多了&#xff0c;88元就够了&#xff0c;腾讯云百科活动 txybk.com/go/txy 活动打开如下图&#xff1a; 腾讯云服务器价格 腾…

Redis主从复制,哨兵和Cluster集群

主从复制&#xff1a; 主从复制是高可用Redis的基础&#xff0c;哨兵和集群都是在主从复制基础上实现高可用的。主从复制主要实现了数据的多机备份&#xff08;和同步&#xff09;&#xff0c;以及对于读操作的负载均衡和简单的故障恢复。 缺陷&#xff1a;故障恢复无法自动化…

Java核心知识点整理大全8-笔记

Java核心知识点整理大全7-笔记-CSDN博客文章浏览阅读1.2k次&#xff0c;点赞27次&#xff0c;收藏26次。但是如果锁的竞争激烈&#xff0c;或者持有锁的线程需要长时间占用锁执行同步块&#xff0c;这时候就不适合 使用自旋锁了&#xff0c;因为自旋锁在获取锁前一直都是占用 c…

什么是凸函数

假设函数是定义在某个向量空间的凸子集上的实值函数&#xff0c;并且&#xff0c;如果对于中的任何两个向量和&#xff0c;都满足&#xff1a; 则称为上的凸函数

Threejs_11 补间动画的实现

啥是补间动画呢&#xff1f;其实就是我们在threejs中移动一个物体的时候&#xff0c;不让他是瞬时移动&#xff0c;让他跟css动画的transition一样&#xff0c;有个过度效果&#xff0c;就是补间动画。补间动画如何设置呢&#xff1f; 补间动画实现 1.引入补间动画库 在我们…

BW4HANA 从头到脚 概念详解 ---- 持续更新中

1. 理解BW4HANA是干嘛的 好歹干了这么久的活了&#xff0c;从当初的啥也不懂到现在感觉啥都知道点&#xff0c;虽然知道的有限&#xff0c;但是也不是小白。渐渐的也知道了SAP开发的一些逻辑。本来咱是想当个BW的大牛的。但是现在感觉这条船要沉了是怎么回事。个人才稍微摸到点…

Windows常用cmd网络命令详解

中午好&#xff0c;我的网工朋友。 上回给你们梳理了一些有趣的cmd命令&#xff0c;很多朋友希望再拓展一下&#xff0c;这不就来了&#xff1f; 今天从windows切入&#xff0c;给你分享一些常用cmd网络命令&#xff0c;如果能熟悉上手&#xff0c;很多功能都可以快速实现&am…

今日定音,博通以610亿美元成功收购VMware | 百能云芯

博通&#xff08;Broadcom&#xff09;日前宣布&#xff0c;已获得中国监管机构的批准&#xff0c;将于今日完成对云计算公司VMware的收购交易。这意味着&#xff0c;610亿美元的收购案正式收关。 据悉&#xff0c;中国市场监管总局在11月21日晚发布了有关附加限制性条件批准博…

基于STM32的电子时钟(论文+源码)

1. 系统设计 电子时钟是一种广泛使用的工具&#xff0c;其可以帮助人们准确掌握时间&#xff0c;本课题基于STM32的电子时钟系统的设计&#xff0c;在功能上设计如下&#xff1a; 具有电子时钟的基本功能&#xff0c;显示年月日&#xff0c;时分秒等基本信息&#xff1b;可以…

代码文档浏览器 Dash mac中文版软件特色

Dash mac是一个基于 Python 的 web 应用程序框架&#xff0c;它可以帮助开发者快速构建数据可视化应用。Dash 的工作原理是将 Python 代码转换成 HTML、CSS 和 JavaScript&#xff0c;从而在浏览器中呈现交互式的数据可视化界面。Dash 提供了一系列组件&#xff0c;包括图表、表…

电脑盘符错乱,C盘变成D盘怎么办?

在一些特殊情况下&#xff0c;磁盘盘符会出现错乱&#xff0c;C盘可能会变成D盘。那么&#xff0c;这该怎么办呢&#xff1f;下面我们就来了解一下。 通过磁盘管理更改盘符 磁盘管理是Windows自带的工具&#xff0c;它位于“计算机管理”的控制台中。管理硬盘及其所包含的卷或…

动态规划求 x 轴上相距最远的两个相邻点 java 代码实现

如图为某一状态下 x 轴上的情况&#xff0c;此时 E、F相距最远&#xff0c;现在加入一个点H&#xff0c;如果H位于点A的左边的话&#xff0c;只需要比较 A、H 的距离 和 E、F 的距离&#xff1b;如果点H位于点G的右边&#xff0c;则值需要比较 G、H 的距离 和 E、F 的距离&…

C++静态链接库的生成以及使用

目录 一.前言二.生成静态链接库三.使用静态链接库四.其他 一.前言 这篇文章简单讨论一下VS如何生成和使用C静态链接库&#xff0c;示例使用VS2022环境。 二.生成静态链接库 先创建C项目-静态库 然后将默认生成的.h和.cpp文件清理干净&#xff0c;当然你也可以选择保留。 然…

VUE+element可以为空不为空时只能为(正整数和0)的验证

rule{ 变量: [ { required: true, validator: validateparamPosition, trigger: blur }] } ​​​​​​​ ​​​​​​​ ​​​​​​​ var validateparamPosition (rule, value, callback) > { if (!value) { //先判断空可以过 ca…

html网站-关于发展历程的案例

一、案例一 1.效果图&#xff1a; 2.代码&#xff1a; 所用到的文件自行在官网下载&#xff0c;也可在git上拉取。 <!DOCTYPE html> <html><head><meta http-equiv"Content-Type" content"text/html; charsetutf-8" /><meta…

Logstash同步MySQL数据到ES

简介 1.1 什么是Logstash&#xff1f; Logstash作为一个具备实时流水线功能的开源数据收集引擎&#xff0c;拥有强大的能力。它能够从不同来源收集数据&#xff0c;并将其动态地汇聚&#xff0c;进而根据我们定义的规范进行转换或者输出到我们定义的目标地址。 1.2 Logstash的…

小雪来袭,安全无忧

今日#小雪# 智安网络提前为你的云保驾护航 让安全成为你的最佳伙伴#智安网络# ​​​

【C++上层应用】3. 动态内存

文章目录 【 1. new和delete运算符 】1.1 new 分配内存1.2 delete 释放内存1.3 实例 【 2. 数组的动态内存分配 】2.1 一维数组2.2 二维数组2.3 三维数组 【 3. 对象的动态内存分配 】 C 程序中的内存分为两个部分&#xff1a; 栈&#xff1a;在 函数内部 声明的所有变量都将占…

Servlet实现一个简单的表白墙网站

文章目录 前言效果展示事前准备HTML、CSS、JavaScript分别负责哪些HTML和CSS构架出页面的基本结构和样式JavaScript 实现行为和交互实现服务器端的业务代码整理pom.xmlweb.xmlmessageWall.htmlMessageServlet.java 前言 前面我们学习了 Java 中知名的 HTTP 服务器 tomcat 的安…