【stm32】ADC的介绍与使用

news2024/10/7 7:15:58

ADC的介绍与使用

  • 1、ADC介绍
  • 2、逐次逼近型ADC
  • 3、ADC电路
  • 4、ADC基本结构
    • 程序代码编写:
    • ADC 通道和引脚复用的关系
  • 5、转换模式
    • (1)单次转换,非扫描模式
      • 转换流程:
      • 程序编写:
    • (2)连续转换,非扫描模式
      • 程序编写:
    • (3)单次转换,扫描模式
      • 程序编写:
    • (4)连续转换,扫描模式
  • 6、数据对齐
  • 7、转换时间
  • 8、ADC校准
  • 9、ADC使用的几种硬件电路

1、ADC介绍

  • ADC(Analog-Digital Converter)模拟-数字转换器

  • ADC可以将引脚上连续变化的模拟电压转换为内存中存储的数字变量,建立模拟电路到数字电路的桥梁

    • STM32主要是数字电路,数字电路只有高低电平,没有几V电压的概念,所以如果想读取电压值,就需要借助ADC模数转换器来实现,ADC读取引脚上的模拟电压,转换为一个数据,存在寄存器里,再把这个数据读取到变量中,就可以进行显示、判断、记录等等操作
    • ADC可以将模拟信号转换为数字信号,是模拟电路到数字电路的桥梁
  • 12位逐次逼近型ADC,1us转换时间

    两个重要参数:

    • 分辨率:一般用多少位来表示,12位AD值,它的表示范围就是0 - (2^12-1),量化结果的范围是0~4095,位数越高,量化结果越精细,对应分辨率越高;
    • 转换时间 :1us表示从AD转换开始到产生结果,需要花1us的时间,对应AD转换的频率就是1MHZ,1MHZ为STM32 ADC的最快转换频率
  • 输入电压范围:03.3V,转换结果范围:04095

    • 0V对应AD值为0,3.3V对应AD值4095,电压值与AD值呈线性关系
  • 18个输入通道,可测量16个外部和2个内部信号源

    • 16个外部信号源:16个GPIO口,引脚可直接接模拟信号测电压(指的是芯片有16个,由于C8T6引脚较少,只有10个);
    • 2个内部信号源:内部温度传感器(可测量CPU的温度)和内部参考电压(1.2V左右基准电压,不随外部供电电压变化而变化,若芯片的供电不是3.3V,测量外部引脚的电压便会不正确,此时可以读取基准电压进行校准,即能得到正确的电压值)。
  • 规则组和注入组两个转换单元

  • 模拟看门狗自动监测输入电压范围

    • 模拟看门狗监测指定的某些通道,当AD值高于它设定的上阈值或者低于下阈值时,看门狗申请中断,此时可以在中断函数里执行相应的操作
  • STM32F103C8T6 ADC资源:ADC1、ADC2,10个外部输入通道

    • 10个外部输入通道:最多只能测量10个外部引脚的模拟信号

2、逐次逼近型ADC

在这里插入图片描述

  • IN0~IN7:8路输入通道,通过通道选择开关,选中一路,到输出点进行转换;
  • 地址锁存和译码:选中的通道,把通道号放到ADDA、ADDB、ADDC这三个脚上,之后给一个锁存信号(ALE),对应的通路开关即可打开;
  • 电压比较器:判断两个输入信号电压(待测电压和DAC电压)的大小关系,如果DAC输出的电压较大,调小DAC数据,如果DAC输出的电压较小,增大DAC数据,直到DAC输出的电压和外部通道输入的电压近似相等,就可得到DAC输入的数据,DAC输入的数据又是外部电压的编码数据;
  • DAC(数模转换器):输入一个数据,输出数据对应的电压;
  • EOC(End OfConvent):转换结束信号;
  • START:给一个输入脉冲,开始转换;
  • VREF+、VREF-:DAC/ADC的参考电压

3、ADC电路

在这里插入图片描述

  • IN0-15—>GPIO端口—>模拟多路开关—>输出到模数转换器(逐次比较)—>转换结果存入到数据寄存器中—>读取寄存器可知ADC转换结果
  • 普通ADC:多路开关一般都是只选中一个,即选中某一个通道、开始转换、等待转换完成,取出结果
  • 高级ADC:可同时选中多个

4、ADC基本结构

在这里插入图片描述

程序代码编写:

步骤:
1、开启RCC时钟,包括ADC和GPIO时钟,ADCCLK的分频器配置
2、配置GPIO,把需要用的GPIO配置成模拟输入的模式
3、配置多路开关,把左边的通道接入到右边的规则组列表里
4、配置ADC转换器,使用库函数结构体配置
5、开关控制,调用ADC_Cmd函数,开启ADC

ADC 通道和引脚复用的关系

在这里插入图片描述

  • 双ADC模式:ADC1和ADC2一起工作,可以配合组成同步模式、交叉模式等模式;
  • 交叉模式:ADC1和ADC2交叉地对一个通道进行采样,可以进一步提高采样率;

5、转换模式

(1)单次转换,非扫描模式

只转换一次,每次只有一个通道,读取EOC标志位的值为1时,读取结束,此时可读取ADC的值

转换流程:

1、软件触发转换
2、等待转换完成
3、读取ADC数据寄存器

程序编写:

// 单通道AD转换
// AD.c
#include "stm32f10x.h"                  // Device header
// 单次转换,非扫描模式    
void AD_Init(void)
{
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
    
    RCC_ADCCLKConfig(RCC_PCLK2_Div6);  // 72M/6 = 12M
    // 初始化PA0为模拟输入的引脚
    GPIO_InitTypeDef GPIO_InitStructure;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;  // 模拟输入模式 在AIN模式下,GPIO口是无效的
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOA, &GPIO_InitStructure);
    //     在规则组菜单列表的第一个位置,写入通道0 , 55.5 + 12.5 = 68个ADC周期
    ADC_RegularChannelConfig(ADC1, ADC_Channel_0, 1, ADC_SampleTime_55Cycles5);// 选择规则组输入通道
    
    ADC_InitTypeDef ADC_InitStructure;
    ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;         // ADC 模式---独立模式
    ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;     // 数据对齐---右对齐
    ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;// 外部触发转换选择(触发控制的触发源)---软件触发
    ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;     // 连续转换模式---单次转换
    ADC_InitStructure.ADC_ScanConvMode = DISABLE;            // 扫描转换模式---非扫描模式
    ADC_InitStructure.ADC_NbrOfChannel = 1;                    // 通道数目
    ADC_Init(ADC1, &ADC_InitStructure);
    
    ADC_Cmd(ADC1, ENABLE);   // 开启ADC
    // ADC校准
    ADC_ResetCalibration(ADC1);     // 复位校准
    while (ADC_GetResetCalibrationStatus(ADC1) == SET);    // 等待复位校准完成
    ADC_StartCalibration(ADC1);        // 开始校准
    while (ADC_GetCalibrationStatus(ADC1) == SET);         // 等待校准完成
}

// 启动转换,获取结果
uint16_t AD_GetValue(void)
{
    ADC_SoftwareStartConvCmd(ADC1, ENABLE);  // 软件触发转换
    // 等待时间5.6us, (1/12M)*68个ADC周期 = 5.6us
    while (ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC) == RESET);// 等待转换完成
    return ADC_GetConversionValue(ADC1);  // 读取ADC数据寄存器
}

// main.c
#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "OLED.h"
#include "AD.h"

uint16_t ADValue;
float Voltage;
/*
    AD采样值来回抖动解决方法:
    1、迟滞比较方法:
    如开关灯,设置两个阈值,低于下阈值时,开灯,高于上阈值时,关灯
    2、滤波方法:均值滤波,
    读10个或20个值,取平均值,作为滤波的AD值,
*/
// 
int main(void)
{
    OLED_Init();
    AD_Init();
    
    OLED_ShowString(1, 1, "ADValue:");
    OLED_ShowString(2, 1, "Volatge:0.00V");
    
    while (1)
    {
        ADValue = AD_GetValue();
        Voltage = (float)ADValue / 4095 * 3.3;  //     AD值转换成电压
        
        OLED_ShowNum(1, 9, ADValue, 4);
        OLED_ShowNum(2, 9, Voltage, 1);
        OLED_ShowNum(2, 11, (uint16_t)(Voltage * 100) % 100, 2); // 显示小数部分
        
        Delay_ms(100);
    }
}

(2)连续转换,非扫描模式

起始触发一次,后续可持续转换,需要ADC的值时直接读取即可

程序编写:

// 单通道AD转换
// AD.c
#include "stm32f10x.h"                  // Device header
// 连续转换,非扫描模式    
void AD_Init(void)
{
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
    
    RCC_ADCCLKConfig(RCC_PCLK2_Div6);  // 72M/6 = 12M
    // 初始化PA0为模拟输入的引脚
    GPIO_InitTypeDef GPIO_InitStructure;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;  // 模拟输入模式 在AIN模式下,GPIO口是无效的
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOA, &GPIO_InitStructure);
    //     在规则组菜单列表的第一个位置,写入通道0 , 55.5 + 12.5 = 68个ADC周期
    ADC_RegularChannelConfig(ADC1, ADC_Channel_0, 1, ADC_SampleTime_55Cycles5);// 选择规则组输入通道
    
    ADC_InitTypeDef ADC_InitStructure;
    ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;         // ADC 模式---独立模式
    ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;     // 数据对齐---右对齐
    ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;// 外部触发转换选择(触发控制的触发源)---软件触发
    ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;     // 连续转换模式---单次转换
    ADC_InitStructure.ADC_ScanConvMode = DISABLE;            // 扫描转换模式---非扫描模式
    ADC_InitStructure.ADC_NbrOfChannel = 1;                    // 通道数目
    ADC_Init(ADC1, &ADC_InitStructure);
    
    ADC_Cmd(ADC1, ENABLE);   // 开启ADC
    // ADC校准
    ADC_ResetCalibration(ADC1);     // 复位校准
    while (ADC_GetResetCalibrationStatus(ADC1) == SET);    // 等待复位校准完成
    ADC_StartCalibration(ADC1);        // 开始校准
    while (ADC_GetCalibrationStatus(ADC1) == SET);         // 等待校准完成

    ADC_SoftwareStartConvCmd(ADC1, ENABLE);  // 软件触发转换 --开始时触发一次即可,后续内部ADC会持续转换
}

// 启动转换,获取结果
uint16_t AD_GetValue(void)
{

    return ADC_GetConversionValue(ADC1);  // 读取ADC数据寄存器
}
// main.c
#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "OLED.h"
#include "AD.h"

uint16_t ADValue;
float Voltage;
/*
    AD采样值来回抖动解决方法:
    1、迟滞比较方法:
    如开关灯,设置两个阈值,低于下阈值时,开灯,高于上阈值时,关灯
    2、滤波方法:均值滤波,
    读10个或20个值,取平均值,作为滤波的AD值,
*/
// 
int main(void)
{
    OLED_Init();
    AD_Init();
    
    OLED_ShowString(1, 1, "ADValue:");
    OLED_ShowString(2, 1, "Volatge:0.00V");
    
    while (1)
    {
        ADValue = AD_GetValue();
        Voltage = (float)ADValue / 4095 * 3.3;  //     AD值转换成电压
        
        OLED_ShowNum(1, 9, ADValue, 4);
        OLED_ShowNum(2, 9, Voltage, 1);
        OLED_ShowNum(2, 11, (uint16_t)(Voltage * 100) % 100, 2); // 小数部分
        
        Delay_ms(100);
    }
}


(3)单次转换,扫描模式

触发一次,转换结束就停止,下次转换需要再触发才能开始,可一次转换7个通道,7个通道转换完成后,产生EOC信号,转换结束
ps: 使用单次转换,非扫描模式实现多通道转换:在每次触发转换之前,先指定一下通道,再启动转换

程序编写:

// AD.c
// 多通道AD转换
#include "stm32f10x.h"                  // Device header

void AD_Init(void)
{
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
    
    RCC_ADCCLKConfig(RCC_PCLK2_Div6);
    
    GPIO_InitTypeDef GPIO_InitStructure;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1 | GPIO_Pin_2 | GPIO_Pin_3;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOA, &GPIO_InitStructure);
        
    ADC_InitTypeDef ADC_InitStructure;
    ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;
    ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
    ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;
    ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;
    ADC_InitStructure.ADC_ScanConvMode = DISABLE;
    ADC_InitStructure.ADC_NbrOfChannel = 1;
    ADC_Init(ADC1, &ADC_InitStructure);
    
    ADC_Cmd(ADC1, ENABLE);
    
    ADC_ResetCalibration(ADC1);
    while (ADC_GetResetCalibrationStatus(ADC1) == SET);
    ADC_StartCalibration(ADC1);
    while (ADC_GetCalibrationStatus(ADC1) == SET);
}

// param:ADC_Channel 指定转换的通道
uint16_t AD_GetValue(uint8_t ADC_Channel)
{
    ADC_RegularChannelConfig(ADC1, ADC_Channel, 1, ADC_SampleTime_55Cycles5); // 改变ADC通道
    ADC_SoftwareStartConvCmd(ADC1, ENABLE);
    while (ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC) == RESET);
    return ADC_GetConversionValue(ADC1);
}
// main.c
#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "OLED.h"
#include "AD.h"

uint16_t AD0, AD1, AD2, AD3;

int main(void)
{
    OLED_Init();
    AD_Init();
    
    OLED_ShowString(1, 1, "AD0:");
    OLED_ShowString(2, 1, "AD1:");
    OLED_ShowString(3, 1, "AD2:");
    OLED_ShowString(4, 1, "AD3:");
    
    while (1)
    {
        // 获取AD值之前先改变通道
        AD0 = AD_GetValue(ADC_Channel_0);    // 获取通道0的AD数据
        AD1 = AD_GetValue(ADC_Channel_1);    // 获取通道1的AD数据
        AD2 = AD_GetValue(ADC_Channel_2);    // 获取通道2的AD数据
        AD3 = AD_GetValue(ADC_Channel_3);    // 获取通道3的AD数据
        
        OLED_ShowNum(1, 5, AD0, 4);
        OLED_ShowNum(2, 5, AD1, 4);
        OLED_ShowNum(3, 5, AD2, 4);
        OLED_ShowNum(4, 5, AD3, 4);
        
        Delay_ms(100);
    }
}

(4)连续转换,扫描模式

有多个通道,一次转换完成继续下一次转换且可以转换多个通道

ps:配合DMA使用

6、数据对齐

  • 数据右对齐:(常用)
    • 12位数据向右靠,高位多出的位数补0
      在这里插入图片描述
  • 数据左对齐:
    • 12位数据向左靠,低位多出的位数补0

在这里插入图片描述

7、转换时间

  • AD转换的步骤:采样,保持,量化,编码
    • 采样,保持:AD转换中,量化,编码需要小部分时间,在这段时间中输入的电压在不断变化,不能具体到输入电压的值,需要在量化编码前,设置采样开关,打开采样开关,采集外部电压,用一个小容量的电容存储这个电压,存储完成之后,断开采样开关,再进行后面的AD转换,这样在量化编码的期间,电压始终保持不变,才能精确地定位未知电压的位置,这就是采样保持电路,采样保持的过程,需要闭合采样开关,过一段时间再断开,会产生一个采样时间。
    • 量化,编码:ADC逐次比较的过程
  • STM32 ADC的总转换时间为:
    T(conv) = 采样时间 + 12.5个ADC周期
    • 采样时间:采样保持花费的时间,在程序中配置(采样时间越大,越能避免一些毛刺信号的干扰,转换时间也会相应延长)
    • 12.5个ADC周期:是量化编码花费的时间(因为是12位的ADC,所以需要花费12个周期),ADC周期是从RCC分频过来的ADCCLK(最大14MHz)
  • 例如:当ADC(clk)=14MHz,采样时间为1.5个ADC周期
    T(conv) = 1.5 + 12.5 = 14个ADC周期 =1μs
    (1/14M)*14个ADC采样周期 = 1μs

8、ADC校准

  • ADC有一个内置自校准模式。校准可大幅减小因内部电容器组的变化而造成的准精度误差。校准期间,在每个电容器上都会计算出一个误差修正码(数字值),这个码用于消除在随后的转换中每个电容器上产生的误差
  • 建议在每次上电后执行一次校准
  • 启动校准前, ADC必须处于关电状态超过至少两个ADC时钟周期

校准过程为固定的,只需要在ADC初始化的最后,加几条代码即可

9、ADC使用的几种硬件电路

在这里插入图片描述1、电位器产生可调电压的电路:一个电位器产生一个可调的电压,电位器的两个固定端,一端接3.3V,另一端接GND,中间的滑动端可输出一个0~3.3V可调的电压输出,滑动端接ADC的输入通道,向上滑动,电压增大,反之减小
2、传感器输出电压电路:N1可以是光敏电阻、热敏电阻、红外接收管、麦克风等,都可以等效为一个可变电阻,此时电阻值无法直接测量,可以通过和一个固定电阻串联分压,来得到一个反应电阻值电压的电路,传感器(N1)阻值变小时,下拉作用变强,输出端(PA1)电压就下降,传感器(N1)阻值变大时,下拉作用变弱,输出端(PA1)受上拉电阻的作用,电压就会升高,固定电阻(R1)一般可以选择和传感器阻值相近的电阻,这样可以得到一个位于中间电压区域比较好的输出,图2传感器和固定电阻的位置也可以交换,此时输出电压的极性相反,这就是这个分压方法来输出传感器阻值的电路。
3、电压转换电路:例如要测一个0-5V的VIN电压,但ADC只能接收0-3.3V的电压,可以使用如上的电压转换电路,使用电阻进行分压,R1+R2 = 50k,根据分压公式 ,中间端的电压=VIN / 50k x 33k,最后得到的电压范围就是0~3.3V,即可进入ADC转换。

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

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

相关文章

Android一个APP里面最少有几个线程

Android一个APP里面最少有几个线程 参考 https://www.jianshu.com/p/92bff8d6282f https://www.jianshu.com/p/8a820d93c6aa 线程查看 Android一个进程里面最少包含5个线程,分别为: main线程(主线程)FinalizerDaemon线程 终结者守护线程…

pg 视图

1.概念 2.创建 3.修改 3.删除 4.递归

【量子计算】开辟全新计算范式

1. 🚀 引言1.1 🚀 量子计算的现状与发展趋势1.2 📜 量子位、量子门和量子电路的基本概念1.3 🏆 量子计算在科学研究中的作用 2. 🔍 量子计算的演变与创新2.1 🌟 量子计算的发展历程2.2 🧠 量子算…

Dit架构 diffusion范式分类+应用

1.ping 网址 2.ssh nscc/l20 3.crtl,打开vscode的setting 4.win 10修改ssh配置文件及其密钥权限为600 - 晴云孤魂 - 博客园 整体来看: 使用transformer作为其主干网络,代替了原先的UNet 在latent space进行训练,通过transformer处理潜…

Leetcode——数组:二分搜索法704.二分查找相似题目

知识点&#xff1a; 二分搜索的区间&#xff0c;一般分为左闭右闭或左闭右开 left0 易错点1 while(易错点2) {middle(leftright)/2if(nums[middle]>target){right易错点3}else if(nums[middle]<target){leftmiddle1}else if(nums[middle]target){return target} } retu…

NRF52832

当我下载完NRF的芯片包后出现了一个缺失pack的问题 解决方法&#xff1a; CMSIS缺失警告_error #541: arm::cmsis:core:4.3.0 component is m-CSDN博客 引入新的型号的芯片问题总结&#xff1a; 一般引入新的型号的芯片都会涉及到.pack和芯片包的问题 外设调用&#xff1a; …

论文阅读:LLaVA-OneVision: Easy Visual Task Transfer

论文地址&#xff1a;https://arxiv.org/pdf/2408.03326 公开时间&#xff1a;2024年9月14日 项目地址&#xff1a;https://llava-vl.github.io/blog/llava-onevision LLaVA-OneVision是一个开放的大型多模态模型&#xff08;LMMs&#xff09;&#xff0c;它是通过整合在LLaVA…

【星闪开发连载】SLE_UUID_Server和SLE_UUID_Client程序分析

引言 星闪测试已经搁置了很久了&#xff0c;主要是8-9月份太忙了。今天重新捡回来。想看看官方提供的星闪例子&#xff0c;在application\samples\bt\sle\sle_uuid_client\和sle_uuid_server目录。这两个例子是一对&#xff0c;一个用来广播服务&#xff0c;另一个去连接。对应…

Mybatis-plus的基础用法

文章目录 1. 核心功能1.1 配置与编写规则1.2 条件构造器1.3 自定义SQL1.4 IService接口1.4.1 Lambda方法1.4.2 批量新增 1.5 分页查询 2. 拓展功能2.1 代码生成器2.2 DB静态工具2.3 逻辑删除2.4 枚举处理器 参考 1. 核心功能 1.1 配置与编写规则 Maven依赖&#xff1a; <…

Autosar EcuM配置-初始化及下电执行函数-基于ETAS软件

文章目录 前言EcuMCommonConfigurationEcuMDriverInitListOneEcuMModuleIDEcuMModuleParameterEcuMModuleServiceEcuMRbDriverInitCoreIdEcuMDriverInitListZeroEcuMGeneralEcuMRbOnGoOffOneCalloutEcuMRbCalloutFunctionEcuMRbModuleIDEcuMRbServiceIsNonAutosarEcuMRbOnGoOff…

java控制台输出乱码

第一步&#xff1a;修改 IntelliJ IDEA 全局编码、项目编码、属性文件编码 File->Settings… Editor->File Encodings(不要忘记点击Apply然后OK) 第二步&#xff1a;修改当前 Web 项目 Tomcat Server 的虚拟机输出选项 Run->Edit Configurations… 复制代码->…

DBMS-3.2 SQL(2)——DML的SELECT(含WHERE、聚集函数、GROUP BY、HAVING之间的关系)

本文章的素材与知识来自李国良老师和王珊老师。 数据操纵语言DML&#xff08;Data Manipulation Language&#xff09; SELECT 一.SELECT的语法与构成 1.语法 2.构成 二.投影 投影操作可以选择表中的若干列&#xff0c;主要体现在SELECT子句后的列表达式。 1.列表达式 2.…

【书生浦语实战】茴香豆企业级知识库问答工具-搭建Dify问答助手

快速结论 1、用茴香豆快速搭建Dify问答助手&#xff0c;自带拒答、rerank、切片长度判断、阈值调节功能&#xff0c;回答还能带出图片&#xff0c;顶呱呱&#x1f44d; 2、茴香豆git仓地址&#xff1a;https://github.com/internlm/huixiangdou 遇到问题去翻这里会更多解释&…

如何利用免费音频剪辑软件制作出精彩音频

现在有许多免费的音频剪辑软件可供选择&#xff0c;它们为广大用户提供了丰富的功能和便捷的操作体验&#xff0c;让音频编辑变得更加轻松和有趣。接下来&#xff0c;让我们一起走进这些免费音频剪辑软件的世界&#xff0c;探索它们的独特魅力和强大功能。 1.福昕音频剪辑 链…

李宏毅深度学习-图神经网络GNN

图卷积的开源代码网站DGL 好用的还是 GAT, GIN&#xff08;指出最好的卷积 就是 hi 邻居特征&#xff08;而且只能用 sum&#xff09;&#xff09; Introduction GNN 可以理解为是由 Graph&#xff08;图) Nerual Networks 组合而成的&#xff0c;图结构应该都在数据结构与…

No.12 笔记 | 网络基础:ARP DNS TCP/IP与OSI模型

一、计算机网络&#xff1a;安全的基石 1. 网络的本质&#xff1a;数字世界的神经系统 定义&#xff1a;计算机的互联互通&#xff0c;实现资源共享和信息交换组成要素&#xff1a;发送者、接收者、介质、数据、协议&#xff08;五大要素&#xff09; 2. 网络架构&#xff1…

STM32PWM应用

一、输出比较(OC) OC&#xff08;Output Compare&#xff09;输出比较 可以通过比较CNT与CCR寄存器值的关系&#xff0c;来对输出电平进行置1、置0或翻转的操作&#xff0c;用于输出一定频率和占空比的PWM波形每个高级定时器和通用定时器都拥有4个输出比较通道高级定时器的前…

三菱FX3U定位控制接线示例(脉冲控制伺服)

一、FX3u系列基本单元(DC24V输入) 二、FX3u系列基本单元(晶体管输出) 脉冲输出用端子Y000、 Y001、 Y002为高速响应输出。 三、FX3UPLC链接MR-J4-A伺服连接实例 1、为了安全起见&#xff0c;不仅仅在可编程控制器侧&#xff0c;在伺服放大器侧也请设计正转限位和反转限位的限位…

查缺补漏----拥塞窗口

注意事项1&#xff1a; 首先区分超时和收到3个冗余ACK&#xff0c;显然超时更加严重。如下图所示&#xff1a; 如果网络出现超时&#xff0c;那么发送窗口cwnd变为1&#xff0c;门限值ssthresh变为原来拥塞窗口的一半 如果收到3个冗余ACK&#xff0c;发送窗口cwnd门限值ssth…

Java重修笔记 第六十一天 坦克大战(十一)IO 流 - 节点流和处理流、BufferedReader 和 BufferedWriter

节点流和处理流的区别和联系 1. 节点流是底层流/低级流&#xff0c;直接和数据源联系 2. 处理流是经过包装过后的节点流&#xff0c;也叫包装流&#xff0c;既可以消除不同节点流的实现差异&#xff0c;也可以提供更方便的方法来完成输入输出 3. 处理流&#xff08;包装流&…