单片机-STM32部分:12、I2C

news2025/7/17 20:17:37

飞书文档https://x509p6c8to.feishu.cn/wiki/MsB7wLebki07eUkAZ1ec12W3nsh

一、简介

IIC协议,又称I2C协议,是由PHILP公司在80年代开发的两线式串行总线,用于连接微控制器及其外围设备,IIC属于半双工同步通信方式。

IIC是一种同步的串行通信总线协议它可以在多个设备之间传输数据。IIC总线由两根线组成:数据线(SDA)和时钟线(SCL)。它使用主从模式,其中一个设备作为主设备控制总线并向其他设备发出命令。IIC协议可以支持高速数据传输和多设备通信,但它的距离限制较短。
UART是一种异步的串行通信协议,它用于在两个设备之间传输数据。UART协议使用两根线:TX(发送)和RX(接收)。UART没有时钟线,数据传输的时序是通过发送和接收设备之间的协议约定实现的。UART协议通常用于短距离通信,例如在计算机和串口设备之间进行通信。
因此,IIC和UART协议在通信的方式、数据传输速度和距离限制等方面存在差异,根据具体的应用场景和需求选择合适的协议更为重要。

多主控(multimastering)

其中任何能够进行发送和接收的设备都可以成为主总线,一个主控能够控制信号的传输和时钟频率。当然,在任何时间点上只能有一个主控。

特征:简单性和有效性

两根线,在标准模式下,I2C总线的最大长度为5米,最大速率为100 kbit/s。在快速模式下,I2C总线的最大长度为1米,最大速率为400 kbit/s。在高速模式下,I2C总线的最大长度为0.4米,最大速率为3.4 Mbit/s。需要注意的是,总线长度的实际限制还取决于总线上的电容负载和电缆质量等因素。

IIC完成的通讯过程如下:

IIC完整的通讯过程

  1. 1、总线是空闲状态,SCL=1,SDA =1;
  2. 2、要开始传输数据了,此时SCL还是高电平,SCL=1,主机将SDA从1变成0;
  3. 3、跟哪个从机通讯,把从机的地址发出去。一般地址是8个bit(也有16个bit的),这8个bit其实真实的地址是7个bit,最后1个bit是用来表示读或者写的。1表示读,0表示写;这个过程相当于主机往SDA上发了8个bit的数据(地址也是数据啊);
  4. 4、主机发地址的过程,相当于在找从机,从机是要给应答信号的,就是ACK,你老板喊你,你也得先回答声A吧;
  5. 5、应答之后,就是要传输数据了,如果第3步中发的地址是写操作,那就由主机来控制SDA的电平变化,如果第3步中发的地址是读操作,那就由从机来控制SDA的电平变化;
  6. 6、每次8bit的数据传输完成,都要有个应答信号,谁接收数据,谁来应答
  7. 7、完事之后,在SCL高电平时,主机把SDA从低电平拉高,表示结束。

STM32中的I2C

STM32 芯片有多个 I2C 外设,它们的 I2C 通讯信号引出到不同的 GPIO 引脚上,使用时必须配置到这些指定的引脚。

SMBus(系统管理总线—System Management Bus)

SMBus总线和I2C是比较类似的,所以STM32兼容了这两种设计,一般场景比较少用SMBus,I2C则是非常多外设使用的接口,我们本节课以I2C为主,看看如何使用STM32的I2C功能。

选择为I2C功能后,会自动选择对应的IO作为I2C的IO,这里I2C是支持重映射的,我们可以根据需要手动修改。

然后,我们可以设置I2C的主从模式,这里我们设置主机模式,模式为Standard Mode,速率为100000Hz。

Master features 主模式特性

Master 为主机模式相关参数,如果是驱动触摸屏、传感器、EEPROM等外设,只需配置这里的参数。

Slave 为从机模式相关参数,如果是开发触摸屏,传感器本身,则需要配置从机参数。


 

Standard Mode:标准模式

Fast Mode:高速模式

这两种模式支持的通讯速率不同,在标准模式中,最大只能设置100KHz,作为主机时,速率大小要看从机支持的最大速率,一般来说100KHz可以满足上述的触摸屏、传感器、EEPROM的驱动。

Slave features 从模式特性

作为主机使用时,这里无需修改

Clock No Stretch Mode: 时钟没有扩展模式
IIC时钟拉伸(Clock stretching):clock stretching通过将SCL线拉低来暂停一个传输.直到释放SCL线为高电平,传输才继续进行.clock stretching是可选的,实际上大多数从设备不包括SCL驱动,所以它们不能stretch时钟.
Primary Address Length selection: 从设备地址长度 设置从设备的地址是7bit还是10bit 大部分为7bit
Dual Address Acknowledged: 双地址确认
Primary slave address:  从设备初始地址

然后就可以生成MDK工程,这里主要用的函数有四个:

 HAL_I2C_Master_Transmit(I2C_HandleTypeDef *hi2c, uint16_t DevAddress,
                         uint8_t *pData, uint16_t Size, uint32_t Timeout);
功能:IIC写数据
参数:
    *hi2c 设置使用的是那个IIC 例:&hi2c1
    DevAddress 写入的地址 设置写入数据的地址 例 0xA0
    *pData 需要写入的数据
    Size 要发送的字节数
    Timeout 最大传输时间,超过传输时间将自动退出传输函数

HAL_I2C_Master_Receive(I2C_HandleTypeDef *hi2c, uint16_t DevAddress,
                        uint8_t *pData, uint16_t Size, uint32_t Timeout);
功能:IIC读一个字节
参数:
    *hi2c: 设置使用的是那个IIC 例:&hi2c1
    DevAddress: 写入的地址 设置写入数据的地址 例 0xA0
    *pDat:a 存储读取到的数据
    Size: 发送的字节数
    Timeout: 最大读取时间,超过时间将自动退出读取函数


HAL_I2C_Mem_Write(I2C_HandleTypeDef *hi2c, uint16_t DevAddress, uint16_t MemAddress,
                  uint16_t MemAddSize, uint8_t *pData, uint16_t Size, uint32_t Timeout);
- 功能:IIC写多个数据,该函数适用于IIC外设里面还有子地址寄存器的地址,比如E2PROM,除了
       设备地址,每个存储字节都有其对应的地址。
- 参数:
  *hi2c      设置使用的是那个IIC 例:&hi2c1
  DevAddress 写入的地址 设置写入数据的地址 例:0xA0
  MemAddress 从机寄存器地址,每写入一个字节数据,地址就会自动+1
  MemAddSize 从机寄存器地址字节长度 8位/16位
             写入数据的字节类型 8位/16位
             I2C_MEMADD_SIZE_8BIT
             I2C_MEMADD_SIZE_16BIT
  *pData     需要写入的数据的起始地址
  Size       传输数据的大小,需要发送的字节数
  Timeout    最大传输时间,超过传输时间将自动退出传输函数
- 例如:HAL_I2C_Mem_Write(&hi2c1,ADDR,i,I2C_MEMADD_SIZE_8BIT,&(I2C_Buffer_Write[i]),8,1000);

HAL_I2C_Mem_Read(I2C_HandleTypeDef *hi2c, uint16_t DevAddress,
                 uint16_t MemAddress, uint16_t MemAddSize,
                 uint8_t *pData, uint16_t Size, uint32_t Timeout)
- 功能:IIC读多个数据,该函数适用于IIC外设里面还有子地址寄存器的地址,比如E2PROM,除了
       设备地址,每个存储字节都有其对应的地址。
- 参数:
  *hi2c      设置使用的是那个IIC 例:&hi2c1
  DevAddress 读设备的地址 设置读数据的地址 例:0xA0
  MemAddress 从机寄存器地址,每读出一个字节数据,地址就会自动+1
  MemAddSize 从机寄存器地址字节长度 8位/16位
             读出数据的字节类型 8位/16位
             I2C_MEMADD_SIZE_8BIT
             I2C_MEMADD_SIZE_16BIT
  *pData     需要读出的数据的起始地址
  Size       传输数据的大小,需要读出的字节数
  Timeout    最大传输时间,超过传输时间将自动退出传输函数


如果只往某个外设中写数据,则用Master_Transmit。 
如果是外设里面还有子地址,例如我们的E2PROM,有设备地址,还有每个数据的寄存器存储地址。则用Mem_Write。
Mem_Write是2个地址,Master_Transmit只有从机地址。
HAL_I2C_Mem_Write(&hi2c1,
0xA0, 0, I2C_MEMADD_SIZE_8BIT,WriteBuffer,BufferSize, 0xff);
HAL_I2C_Mem_Read(&hi2c1,
0xA1, 0, I2C_MEMADD_SIZE_8BIT,ReadBuffer,BufferSize, 0xff);
 

EEPROM

EEPROM (Electrically Erasable Programmable read only memory)是指带电可擦可编程存储器。是一种掉电后数据不丢失的存储芯片。

https://item.szlcsc.com/320744.html

BL24C02是一个2Kbit的EEPROM, 内部含有256个字节可以存储数据,总共有32页,每页8Byte。

设备读写地址说明

其中设备地址如下,A2 A1 A0对应芯片硬件接的电平


 

如果我们把A2 A1 A0都接到GND,这时候,

写数据时,设备地址字节应该是0b1010 0000=0xA0

读数据时,设备地址字节应该是0b1010 0001=0xA1

参考飞书文档

字节写

每次写入一个Byte数据

  • 先发送起始信号
  • 发送从设备地址+写入标志数据(Wbit=0),等待应答
  • 发送写入地址,等待应答
  • 发送数据,等待应答
  • 发送结束信号。

页写

每次可以写入一页(8Byte)的数据

读字节

连续读

连续读操作可通过立即读或选择性读操作启动。在 24C02 发送完一个 8 位字节数据后,主器件产生一个应答信号来响应,告知 24C02  主器件要求更多的数据,对应每个主机产生的应答信号 24C02 将发送一个 8 位数据字节。当主器件不发送应答信号而发送停止位时结束此操作。

STM32CUBEMX开启I2C1,对应PB6 PB7

然后打开USART1用于打印日志,方便查看

注意,要勾选MicroLIB哦,否则printf打印不了数据

然后添加代码如下:

main.c

/* USER CODE BEGIN Includes */
#include <stdio.h>
/* USER CODE END Includes */

/* USER CODE BEGIN 0 */
#define ADDR_24LCxx_Write 0xA0
#define ADDR_24LCxx_Read 0xA1
#define BufferSize 8
uint8_t WriteBuffer[BufferSize] = {1,2,3,4,5,6,7,8};
uint8_t ReadBuffer[BufferSize] = {0};
/* USER CODE END 0 */

  while (1)
  {
    /* USER CODE END WHILE */

    /* USER CODE BEGIN 3 */
    printf("start to test i2c eeprom\n");
    if(HAL_I2C_Mem_Write(&hi2c1, ADDR_24LCxx_Write, 0, I2C_MEMADD_SIZE_8BIT, WriteBuffer, sizeof(WriteBuffer), 0xff) == HAL_OK)
    {
        printf("EEPROM 24C02 Write Test OK \r\n");
    }
    HAL_Delay(10);          
    /* read date from EEPROM */
    HAL_I2C_Mem_Read(&hi2c1, ADDR_24LCxx_Read, 0, I2C_MEMADD_SIZE_8BIT, ReadBuffer, sizeof(ReadBuffer), 0xff);
    for(int i = 0; i < sizeof(ReadBuffer); i++)
    {
       printf("0x%02X  ",ReadBuffer[i]);
    }
    HAL_Delay(1000);
  }
  /* USER CODE END 3 */
 
 

/* USER CODE BEGIN 4 */
int fputc(int ch, FILE *f)
{
    HAL_UART_Transmit(&huart1 , (uint8_t *)&ch, 1, 0xFFFF);
    return ch;
}

打印如下

温湿度传感器

温湿度传感器:CJ-GXHT3L

GXHT3L-DIS 是中科银河芯开发的新一代单芯片集成温湿度一 体传感器。

★ I2C 接口,通信速度高达 1MHz

★ 两个用户可选择的地址

★ GXHT3L 典型精度为±4%RH 和±0.5°C

★ GXHT30 典型精度为±3%RH 和±0.3°C

★ GXHT31 典型精度为±2%RH 和±0.3°C

★ 单芯片集成温湿传感器

★ 高可靠性和长期稳定性

★ 测量 0-100%范围相对湿度

★ 测量-45-130℃范围内温度

https://item.szlcsc.com/3199174.html

关于设备地址与ADDR管脚说明:

这里要注意的是,0x44指的是I2C地址的高7位,第八位为读写标志位。


 

0x44 = 0b0100 0100,把最高位去掉 = 0b100 0100

写数据时,设备地址字节应该是0b1000 1000=0x88

读数据时,设备地址字节应该是0b1000 1001=0x89

高重复率和周期转换频率,例如0x2130中,21代表每秒转换一次,30代表高重复率。

设置进入连续转换模式的命令

* USER CODE BEGIN Includes */
#include <stdio.h>
#define    GXHT3L_ADDR_WRITE    0x44<<1         //10001000
#define    GXHT3L_ADDR_READ     (0x44<<1)+1     //10001001

typedef enum
{
    /* 软件复位命令 */

    SOFT_RESET_CMD = 0x30A2,   
    /* 加热使能/禁能命令 */
    PREHEAT_ENABLE_CMD = 0x306D,
    PREHEAT_DISENABLE_CMD = 0x3066,
    /* 芯片状态命令 */
    DEVICE_STATUS_CMD = 0xF32D,
   
    /*
    单次测量模式
    命名格式:Repeatability_CS_CMD
    CS:Clock stretching
    */
    HIGH_ENABLED_CMD    = 0x2C06,
    MEDIUM_ENABLED_CMD  = 0x2C0D,
    LOW_ENABLED_CMD     = 0x2C10,
    HIGH_DISABLED_CMD   = 0x2400,
    MEDIUM_DISABLED_CMD = 0x240B,
    LOW_DISABLED_CMD    = 0x2416,

    /*
    周期测量模式
    命名格式:Repeatability_MPS_CMD
    MPS:measurement per second
    */
    HIGH_0_5_CMD   = 0x2032,
    MEDIUM_0_5_CMD = 0x2024,
    LOW_0_5_CMD    = 0x202F,
    HIGH_1_CMD     = 0x2130,
    MEDIUM_1_CMD   = 0x2126,
    LOW_1_CMD      = 0x212D,
    HIGH_2_CMD     = 0x2236,
    MEDIUM_2_CMD   = 0x2220,
    LOW_2_CMD      = 0x222B,
    HIGH_4_CMD     = 0x2334,
    MEDIUM_4_CMD   = 0x2322,
    LOW_4_CMD      = 0x2329,
    HIGH_10_CMD    = 0x2737,
    MEDIUM_10_CMD  = 0x2721,
    LOW_10_CMD     = 0x272A,
    /* 周期测量模式读取数据命令 */
    READOUT_FOR_PERIODIC_MODE = 0xE000,
} GXHT3L_CMD;
/* USER CODE END Includes */


/* USER CODE BEGIN 0 */
/**
 * @brief    向GXHT3L发送一条指令(16bit)
 * @param    cmd —— GXHT3L指令(在GXHT3L_MODE中枚举定义)
 * @retval    成功返回HAL_OK
*/
static uint8_t GXHT3L_Send_Cmd(GXHT3L_CMD cmd)
{
    uint8_t cmd_buffer[2];
    cmd_buffer[0] = cmd >> 8;
    cmd_buffer[1] = cmd;
    return HAL_I2C_Master_Transmit(&hi2c2, GXHT3L_ADDR_WRITE, (uint8_t*) cmd_buffer, 2, 0xFFFF);
}

/**
 * @brief    复位GXHT3L
 * @param    none
 * @retval    none
*/
static void GXHT3L_Reset(void)
{
    GXHT3L_Send_Cmd(SOFT_RESET_CMD);
    HAL_Delay(20);
}

void GXHT3L_Preheat_Disable(void)
{
    GXHT3L_Send_Cmd(PREHEAT_DISENABLE_CMD);
    HAL_Delay(20);
}

uint8_t GXHT3L_Read_Status(uint8_t* dat)
{
    GXHT3L_Send_Cmd(DEVICE_STATUS_CMD);
    return HAL_I2C_Master_Receive(&hi2c2, GXHT3L_ADDR_READ, dat, 3, 0xFFFF);
}

/**
 * @brief    初始化GXHT3L
 * @param    none
 * @retval    成功返回HAL_OK
 * @note    周期测量模式
*/
uint8_t GXHT3L_Init(void)
{
    return GXHT3L_Send_Cmd(MEDIUM_2_CMD);
}

/**
 * @brief    从GXHT3L读取一次数据
 * @param    dat —— 存储读取数据的地址(6个字节数组)
 * @retval    成功 —— 返回HAL_OK
*/
uint8_t GXHT3L_Read_Dat(uint8_t* dat)
{
    GXHT3L_Send_Cmd(READOUT_FOR_PERIODIC_MODE);
    return HAL_I2C_Master_Receive(&hi2c2, GXHT3L_ADDR_READ, dat, 6, 0xFFFF);
}

#define CRC8_POLYNOMIAL 0x31

uint8_t CheckCrc8(uint8_t* const message, uint8_t initial_value)
{
    uint8_t  remainder;        //余数
    uint8_t  i = 0, j = 0;  //循环变量

    /* 初始化 */
    remainder = initial_value;
    for(j = 0; j < 2;j++)
    {
        remainder ^= message[j];
        /* 从最高位开始依次计算  */
        for (i = 0; i < 8; i++)
        {
            if (remainder & 0x80)
            {
                remainder = (remainder << 1)^CRC8_POLYNOMIAL;
            }
            else
            {
                remainder = (remainder << 1);
            }
        }
    }
    /* 返回计算的CRC码 */
    return remainder;
}

/**
 * @brief    将GXHT3L接收的6个字节数据进行CRC校验,并转换为温度值和湿度值
 * @param    dat  —— 存储接收数据的地址(6个字节数组)
 * @retval    校验成功  —— 返回0
 *             校验失败  —— 返回1,并设置温度值和湿度值为0
*/
uint8_t GXHT3L_Dat_To_Float(uint8_t* const dat, float* temperature, float* humidity)
{
    uint16_t recv_temperature = 0;
    uint16_t recv_humidity = 0;

    /* 校验温度数据和湿度数据是否接收正确 */
    if(CheckCrc8(dat, 0xFF) != dat[2] || CheckCrc8(&dat[3], 0xFF) != dat[5])
        return 1;

    /* 转换温度数据 */
    recv_temperature = ((uint16_t)dat[0]<<8)|dat[1];
    *temperature = -45 + 175*((float)recv_temperature/65535);

    /* 转换湿度数据 */
    recv_humidity = ((uint16_t)dat[3]<<8)|dat[4];
    *humidity = 100 * ((float)recv_humidity / 65535);

    return 0;
}



  /* USER CODE BEGIN 1 */
  uint8_t recv_dat[6] = {0};
  uint8_t recv_status[3] = {0};
  float temperature = 0.0;
  float humidity = 0.0;
  /* USER CODE END 1 */

  /* USER CODE BEGIN 2 */
  GXHT3L_Reset();
  if(GXHT3L_Init() == HAL_OK)
      printf("GXHT3L init ok.\n");
  else
      printf("GXHT3L init fail.\n");
  if(GXHT3L_Read_Status(recv_status) == HAL_OK){
      printf("GXHT3L Read Status ok. Status = 0x%x%x\n",recv_status[0],recv_status[1]);
  }else
      printf("GXHT3L Read Status fail.\n");
  /* USER CODE END 2 */


  while (1)
  {
    /* USER CODE BEGIN 3 */
    HAL_Delay(1000);
    if(GXHT3L_Read_Dat(recv_dat) == HAL_OK)
    {
       if(GXHT3L_Dat_To_Float(recv_dat, &temperature, &humidity)==0)
       {
            printf("temperature = %f, humidity = %f\n", temperature, humidity);
       }
       else
       {
          printf("crc check fail.\n");
       }
    }
    else
    {
       printf("read data from GXHT3L fail.\n");
    }
  }
  /* USER CODE END 3 */



/* USER CODE BEGIN 4 */
int fputc(int ch, FILE *f)
{
    HAL_UART_Transmit(&huart1 , (uint8_t *)&ch, 1, 0xFFFF);
    return ch;
}
/* USER CODE END 4 */


最终工程,可以参考淘宝旺旺发送的源码部分哦

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

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

相关文章

Payload的定义及核心概念

在IT领域&#xff0c;Payload&#xff08;有效载荷&#xff09; 指数据传输或操作中承载实际功能或信息的主体部分&#xff0c;与协议头、元数据等辅助内容区分。其核心特点是完成特定目标&#xff0c;例如传递关键数据、执行代码逻辑或实现攻击行为。 主要应用场景及技术解析 …

计算机网络笔记(二十四)——4.6互联网的路由选择协议

4.6.1有关路由选择协议的几个基本概念 路由选择协议是计算机网络中维护和生成路由表的核心机制。 1. 路由选择的核心目标 转发&#xff08;Forwarding&#xff09;&#xff1a;路由器基于本地转发表&#xff0c;将分组从输入链路转移到输出链路&#xff08;单台路由器的本地…

论文阅读与写作:《从探索到突破:解密科研和论文写作的思维密码》

文章目录 一、如何做科研1.科研的步骤2.课题选择3.快速入门一个新领域&#xff1a;读论文&#xff0c;先读综述(1)自己看论文的时候&#xff0c;每篇论文花3-5分钟记录一下自己的idea和一些瞬间的想法(2)高质量文献&#xff1a;顶会顶刊(3)如何检索 4.注重团队协作与学术交流5.…

致远OA绩效考核管理应用【附百度网盘下载链接,官方售价8K】

产品概述 绩效考核管理预置三种考核方式&#xff1a;工作事务考核、关键绩效考核、360度考评&#xff0c;满足不同企业考核需求&#xff0c;从考核等级定义、考核方案设置、考核分发到员工考核&#xff0c;再到考核结果汇总并审批&#xff0c;对绩效考核全过程进行闭环管理&…

DeepSeek执行流程加速指南:跨框架转换与编译优化的核心策略全解析

&#x1f49d;&#x1f49d;&#x1f49d;欢迎来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐:kwan 的首页,持续学…

manuskript开源程序是面向作家的开源工具

一、软件介绍 文末提供程序和源码下载 manuskript开源程序是面向作家的开源工具&#xff0c;Manuskript 可在 GNU/Linux、Mac OS X 和 Windows 上运行。 二、Features 特征 Manuskript provides a rich environment to help writers create their first draft and then furt…

游戏引擎学习第281天:在房间之间为摄像机添加动画效果

回顾并为今天的内容定下基调 这次我们要继续深入处理实体系统。在前一阶段对实体系统做了一些很酷的改动&#xff0c;但现在到了要认真面对和完善它的时候。 今天的主要目标是修复并优化摄像机在房间之间移动时的逻辑。在上一次的实现中&#xff0c;我们重新启用了基于房间的…

Kaamel隐私合规洞察:Temu在韩被罚事件分析

Kaamel隐私合规与数据安全团队分析报告 韩国个人信息保护委员会&#xff08;PIPC&#xff09;对中国电子商务巨头Temu处以巨额罚款&#xff0c;原因是其严重违反了用户数据保护法律 。核心违规行为包括未经适当披露或用户同意非法跨境传输数据、未能指定当地代表、账户注销流程…

计算机视觉----基于锚点的车道线检测、从Line-CNN到CLRNet到CLRKDNet 本文所提算法Line-CNN 后续会更新以下全部算法

本文所提算法如下&#xff1a; 叙述按时间顺序 你也可以把本文当作快速阅读这几篇文献的一个途径 所有重要的部分我都已经标注并弄懂其原理 方便自己也是方便大家 Line-CNN&#xff1a;基于线提议单元的端到端交通线检测 摘要 交通线检测是一项基础且具有挑战性的任务。以往的…

25.5.15

没有比水题更令人开心的事情了 典型的并查集题目&#xff0c;并查集分为并和查&#xff0c;并就是把有关系的父亲根结点设为同一个&#xff0c;查就是在成功构造后对其进行查询 查通过递归实现 if (x f[x])return x; return f[x] find(f[x]); 由于并查集的特点&#xff0…

5.重建大师数据管理模块介绍

摘要&#xff1a;本文主要介绍重建大师数据管理模块&#xff0c;包含&#xff1a;照片、点云数据可视化管理工具。 数据管理界面主要包含工具栏、可视化界面和照片组列表三部分。 图 数据管理界面 1.工具栏 工具栏包含以下功能按钮&#xff0c;包含添加照片、视频、点云、控制…

MATLAB安装常见问题及解决方案详解(含代码示例)

MATLAB作为科学计算和工程分析的核心工具&#xff0c;其安装过程可能因操作系统版本、硬件配置或网络环境等因素而出现各种问题。本文基于MATLAB官方文档和社区经验&#xff0c;系统总结了安装过程中常见的问题&#xff0c;并提供详细的解决方案和代码示例&#xff0c;帮助用户…

微信小程序智能商城系统(uniapp+Springboot后端+vue管理端)

一、系统介绍 本智能商城系统是基于当今主流技术栈开发的一款多端商城解决方案&#xff0c;主要包括微信小程序前端、SpringBoot 后端服务以及 Vue 管理后台三大部分。系统融合了线上商城的核心功能&#xff0c;支持商品浏览、下单、支付、订单管理等操作&#xff0c;适用于中小…

【Spark分析HBase数据】Spark读取并分析HBase数据

Spark读取并分析HBase数据 一、摘要二、实现过程三、小结 一、摘要 Apache Spark 是一个快速、通用的大数据处理引擎&#xff0c;提供了丰富的 API 用于数据处理和分析。HBase 是一个分布式、可扩展的 NoSQL 数据库&#xff0c;适合存储海量结构化和半结构化数据。Spark 与 HB…

leetcode2934. 最大化数组末位元素的最少操作次数-medium

1 题目&#xff1a;最大化数组末位元素的最少操作次数 官方标定难度&#xff1a;中 给你两个下标从 0 开始的整数数组 nums1 和 nums2 &#xff0c;这两个数组的长度都是 n 。 你可以执行一系列 操作&#xff08;可能不执行&#xff09;。 在每次操作中&#xff0c;你可以选…

环境配置与MySQL简介

目录 1 环境配置 2 MySQL简介 1 环境配置 本专栏使用CentOS7进行讲解。首先我们查看系统中是否已经安装了MySQL&#xff0c;可以使用rpm -qa 命令查看系统安装包/压缩包 列表 这只是看我们是否下载过对应安装包&#xff0c;不一定就安装了。如果我们需要重新下载&#xff0c;…

适用于 iOS 的 开源Ultralytics YOLO:应用程序和 Swift 软件包,用于在您自己的 iOS 应用程序中运行 YOLO

​一、软件介绍 文末提供程序和源码下载 该项目利用 Ultralytics 最先进的 YOLO11 模型将您的 iOS 设备转变为用于对象检测的强大实时推理工具。直接从 App Store 下载该应用程序&#xff0c;或浏览我们的指南&#xff0c;将 YOLO 功能集成到您自己的 Swift 应用程序中。 二、…

Java零基础学习Day12——集合ArrayList

一、基本使用 1. 集合与数组 集合只存引用数据类型&#xff1b;长度可变 数组可存基本数据类型、引用数据类型&#xff1b;长度固定 2. 基本格式 ArrayList<String> list new ArrayList<>(); 3. 方法 增、删 import java.util.ArrayList; public class St…

[论文阅读]Formalizing and Benchmarking Prompt Injection Attacks and Defenses

Formalizing and Benchmarking Prompt Injection Attacks and Defenses Formalizing and Benchmarking Prompt Injection Attacks and Defenses | USENIX 33rd USENIX Security Symposium (USENIX Security 24) 提出了一个框架来形式化提示注入攻击&#xff0c;对提示注入攻击…

JavaScript性能优化实战,从理论到落地的全面指南

在前端开发领域&#xff0c;JavaScript的性能优化是提升用户体验的核心环节。随着Web应用复杂度的提升&#xff0c;开发者面临的性能瓶颈也日益多样化。本文将从理论分析、代码实践和工具使用三个维度&#xff0c;系统性地讲解JavaScript性能优化的实战技巧&#xff0c;并通过大…