【嵌入式】HC32F07X CAN通讯配置和使用配置不同缓冲器以连续发送

一 背景说明

        使用小华(华大)的MCU HC32F07X实现 CAN 通讯配置和使用

二 原理分析

【1】CAN原理说明(参考文章《CAN通信详解》):

        CAN是控制器局域网络(Controller Area Network, CAN)的简称,是一种能够实现分布式实时控制的串行通信网络。

        (i)CAN通信形式:CAN 使用称为 CANH / CANL 的通信线路执行传输和接收。电位差较小的电信号称为隐性信号,其逻辑值为1。电位差较大的电信号称为显性信号,其逻辑值0。如果通信总线上发生显性和隐性冲突,则显性优先。总线空闲时保持隐性。

        (ii)CAN数据格式:CAN的数据定义了有5种帧类型:

        (ii)CAN数据帧:数据帧一般由 7 个段构成,即:

(1) 帧起始。表示数据帧开始的段。
(2) 仲裁段。表示该帧优先级的段。
(3) 控制段。表示数据的字节数及保留位的段。
(4) 数据段。数据的内容,一帧可发送 0~8 个字节的数据。
(5) CRC 段。检查帧的传输错误的段。
(6) ACK 段。表示确认正常接收的段。
(7) 帧结束。表示数据帧结束的段

        图中 D 表示显性电平, R 表示隐形电平。

        更多具体内容不再赘述,可以参考上面的文章链接或者自行搜索。

【2】HC32F07X的CAN外设

        芯片CAN外设的主要特性:

■ 完全支持 CAN2.0A/CAN2.0B 协议。
■ 向上兼容 CAN-FD 协议。
■ 支持最高通信波特率 1Mbit/s
■ 支持 1~1/256 的波特率预分频, 灵活配置波特率。
■ 10 个接收缓冲器
- FIFO 方式
- 错误或者不被接收的数据不会覆盖存储的消息
■ 1 个高优先主发送缓冲器 PTB
■ 4 个副发送缓冲器 STB
- FIFO 方式
- 优先级仲裁方式
■ 8 组独立的筛选器
- 支持 11 位标准 ID 和 29 位扩展 ID
- 可编程 ID CODE 位以及 MASK 位
■ PTB/STB 均支持支持单次发送模式
■ 支持静默模式
■ 支持回环模式
■ 支持捕捉传输的错误种类以及定位仲裁失败位置
■ 可编程的错误警告值
■ 支持 ISO11898-4 规定时间触发 CAN 以及接收时间戳

        系统框图如下:

        更多详细的内容可以参考HC32F07X芯片的DATASHEET。

三 CAN通讯硬件设计

        以下推荐了两个CAN收发的硬件电路,可以将外部的 CANH/CANL 差分信号,转换为 CAN_TX/CAN_RX 信号用以内部MCU处理。一个是官方提供的,一个是开发板提供的,都可以正常使用。

        注意CAN通信需要5V,用以给收发器供电

【1】推荐电路1

        参考官方的硬件设计指南《AN_HC32L072_HC32L073_HC32F072系列硬件开发指南_Rev1.1》,里面对于CAN通信收发的硬件电路有如下推荐电路,其中使用了NXP的 TJA1042 作为收发器

【2】推荐电路2

        使用的周立功开发板中也引出了CAN功能,其中使用了NXP的TJA1051T作为收发器

四 CAN通讯软件配置

        选用引脚 PD00(CAN_RX)、PD01(CAN_TX)、PC12(CAN_STB)实现CAN通信功能,使用时钟频率为48MHz(官方例程使用的是外部8M晶振),CAN通讯波特率为1M,标准帧ID号为1。

【1】系统时钟配置

static void App_SysClkInit(void)
{
    stc_sysctrl_clk_cfg_t stcCfg;
    stc_sysctrl_pll_cfg_t stcPLLCfg;

    Sysctrl_SetPeripheralGate(SysctrlPeripheralFlash, TRUE);    ///< 使能FLASH模块的外设时钟
    Flash_WaitCycle(FlashWaitCycle1);
    Sysctrl_SetRCHTrim(SysctrlRchFreq4MHz);             ///< PLL使用RCH作为时钟源,因此需要先设置RCH

    stcPLLCfg.enInFreq    = SysctrlPllInFreq4_6MHz;     ///< RCH 4MHz
    stcPLLCfg.enOutFreq   = SysctrlPllOutFreq36_48MHz;  ///< PLL 输出48MHz
    stcPLLCfg.enPllClkSrc = SysctrlPllRch;              ///< 输入时钟源选择RCH
    stcPLLCfg.enPllMul    = SysctrlPllMul12;            ///< 4MHz x 12 = 48MHz
    Sysctrl_SetPLLFreq(&stcPLLCfg);

    ///< 选择PLL作为HCLK时钟源;
    stcCfg.enClkSrc  = SysctrlClkPLL;
    ///< HCLK SYSCLK/2
    stcCfg.enHClkDiv = SysctrlHclkDiv1;
    ///< PCLK 为HCLK/8
    stcCfg.enPClkDiv = SysctrlPclkDiv1;
    ///< 系统时钟初始化
    Sysctrl_ClkInit(&stcCfg);
}

【2】CAN初始化GPIO

//CAN通信引脚定义
#define CAN_RX_PORT     (GpioPortD)
#define CAN_RX_PIN      (GpioPin0)
#define CAN_TX_PORT     (GpioPortD)
#define CAN_TX_PIN      (GpioPin1)
#define CAN_STB_PORT    (GpioPortC)
#define CAN_STB_PIN     (GpioPin12)

#define  APB1_CLK       48000000    //CAN 的输入时钟
#define  CAN_BAUD       1000000     //CAN 的波特率

/**************************************************************************
* 函数名称: COM_Init
* 功能描述: CAN通信初始化GPIO
**************************************************************************/
void COM_Init(void)
{
    stc_gpio_cfg_t stcGpioCfg;

    Sysctrl_SetPeripheralGate(SysctrlPeripheralGpio, TRUE);
    
    stcGpioCfg.enDir = GpioDirIn;       ///< 端口方向配置->输入
    stcGpioCfg.enDrv = GpioDrvL;        ///< 端口驱动能力配置->高驱动能力
    stcGpioCfg.enPu  = GpioPuDisable;   ///< 端口上下拉配置->无
    stcGpioCfg.enPd  = GpioPdDisable;
    stcGpioCfg.enOD  = GpioOdDisable;   ///< 端口开漏输出配置->开漏输出关闭
    stcGpioCfg.enCtrlMode = GpioAHB;    ///< 端口输入/输出值寄存器总线控制模式配置->AHB
    Gpio_Init(CAN_RX_PORT, CAN_RX_PIN, &stcGpioCfg);
    
    stcGpioCfg.enDir = GpioDirOut;
    Gpio_Init(CAN_TX_PORT, CAN_TX_PIN, &stcGpioCfg);
    Gpio_Init(CAN_STB_PORT, CAN_STB_PIN, &stcGpioCfg);
    
    ///<CAN RX\TX复用功能配置
    Gpio_SetAfMode(CAN_RX_PORT, CAN_RX_PIN, GpioAf1);
    Gpio_SetAfMode(CAN_TX_PORT, CAN_TX_PIN, GpioAf1);
    ///<STB 低-PHY有效
    Gpio_ClrIO(CAN_STB_PORT, CAN_STB_PIN);
}

【3】CAN波特率自适应配置接口

/**************************************************************************
* 函数名称: COM_BaudCfg
* 功能描述: CAN通信波特率自动配置
* 其他说明: 48MHz主频,1M波特率计算得到:SJW=2, PRESC=1-1, SEG_2=15, SEG_1=30
             若输入500K波特率,PRESC为2-1; 输入250K波特率,PRESC为4-1
**************************************************************************/
void COM_BaudCfg(stc_can_init_config_t *p_stcCanInitCfg, uint32_t src_clk, uint32_t baud)
{
    uint32_t i,value = baud,record = 1;
    uint32_t remain = 0,sum_prescaler = 0;
    while(( baud == 0 )||( src_clk == 0 ));
    sum_prescaler = src_clk / baud;
    for ( i = 73; i > 3; i-- ) {
        remain = sum_prescaler - ((sum_prescaler / i)*i);
        if( remain == 0 ) {
          record = i;
          break;
        } else {
          if (remain < value) {
              value = remain;
              record = i;
          }
        }
    }

    /* 设置重新同步跳跃宽度为2个时间单位 */
    p_stcCanInitCfg->stcCanBt.SJW = 2;
    p_stcCanInitCfg->stcCanBt.PRESC = (sum_prescaler/record) - 1;
    p_stcCanInitCfg->stcCanBt.SEG_2 = (record - 3) / 3;
    p_stcCanInitCfg->stcCanBt.SEG_1 = (record - 3) - p_stcCanInitCfg->stcCanBt.SEG_2;
}

        【注】:CAN波特率的计算和配置,也可以使用CAN波特率计算器工具来计算得到(下载地址:CAN波特率计算器下载):

【4】CAN初始化配置(其中,波特率配置为1M,滤波器配置标准帧ID为1

/**************************************************************************
* 函数名称: COM_Cfg
* 功能描述: CAN通信初始化配置
**************************************************************************/
void COM_Cfg(void)
{
    stc_can_init_config_t   stcCanInitCfg;
    stc_can_filter_t        stcFilter;

    Sysctrl_SetPeripheralGate(SysctrlPeripheralCan, TRUE);
    
    ///<CAN 波特率配置
    COM_BaudCfg(&stcCanInitCfg, APB1_CLK, CAN_BAUD);    //48MHz主频,1M波特率
    
    stcCanInitCfg.stcWarningLimit.CanErrorWarningLimitVal = 16-1;
    stcCanInitCfg.stcWarningLimit.CanWarningLimitVal = 10;
    stcCanInitCfg.enCanRxBufAll  = CanRxNormal;
    stcCanInitCfg.enCanRxBufMode = CanRxBufNotStored;
    stcCanInitCfg.enCanSTBMode   = CanSTBFifoMode;
    CAN_Init(&stcCanInitCfg);

    ///<CAN 滤波器配置
    stcFilter.enAcfFormat = CanAllFrames;
    stcFilter.enFilterSel = CanFilterSel1;
    stcFilter.u32CODE     = 0x00000001;
    stcFilter.u32MASK     = 0x1FFFFFFF;
    CAN_FilterConfig(&stcFilter, TRUE);

    CAN_IrqCmd(CanRxIrqEn, TRUE);
    EnableNvic(CAN_IRQn, IrqLevel0, TRUE);
}

【5】CAN中断服务子程序(用于CAN接收)

stc_can_rxframe_t       stcRxFrame;
stc_can_txframe_t       stcTxFrame;
uint8_t                 u8RxFlag = FALSE;

/**************************************************************************
* 函数名称: Can_IRQHandler
* 功能描述: CAN中断服务函数
**************************************************************************/
void Can_IRQHandler(void)
{
    if(TRUE == CAN_IrqFlgGet(CanRxIrqFlg))
    {
        CAN_IrqFlgClr(CanRxIrqFlg);
        CAN_IrqCmd(CanRxIrqEn, FALSE);

        CAN_Receive(&stcRxFrame);

        u8RxFlag = TRUE;
    }
}

【6】CAN发送接口(用于CAN发送)

/**************************************************************************
* 函数名称: COM_Tx
* 功能描述: CAN通信发送测试
**************************************************************************/
void COM_Tx(void)
{
    //中断接收,循环发送
    uint8_t u8Idx = 0;
    
    if(TRUE == u8RxFlag)
    {
        u8RxFlag = FALSE;

        if(1 == stcRxFrame.Cst.Control_f.RTR)
        {
            return;
        }

        //<<Can Tx
        stcTxFrame.StdID         = stcRxFrame.StdID;
        stcTxFrame.Control_f.DLC = stcRxFrame.Cst.Control_f.DLC;
        stcTxFrame.Control_f.IDE = stcRxFrame.Cst.Control_f.IDE;
        stcTxFrame.Control_f.RTR = stcRxFrame.Cst.Control_f.RTR;

        for(u8Idx=0; u8Idx<stcRxFrame.Cst.Control_f.DLC; u8Idx++)
        {
            stcTxFrame.Data[u8Idx] = stcRxFrame.Data[u8Idx];
        }

        CAN_SetFrame(&stcTxFrame);
        CAN_TransmitCmd(CanPTBTxCmd);

        CAN_IrqCmd(CanRxIrqEn, TRUE);
    }
}

【7】主函数调用

int32_t main(void)
{
    //系统时钟
    App_SysClkInit();

    //通信模块
    COM_Init();
    COM_Cfg();
    
    while(1)
    {
        //通信
        COM_Tx();
        delay1ms(10);
    }
}

五 CAN通讯测试1(中断接收,Echo发送单帧)

        进行CAN测试的方法很多,如果没有USB-CAN通信转换工具/CAN协议分析工具,可以增加CAN外部回环或者内部回环的配置,配合Log串口输出或者仿真器Debug进行测试。我这边使用了USB-CAN通信转换工具(CAN-II)/CAN协议分析工具(CANTest)直接进行收发测试:

【1】选择 CANTest 中的对应设备 USBCAN2:

【2】匹配软件中的波特率为1M,确定并启动CAN:

【3】修改帧ID为1,点击发送,可以看到上位机发送数据到测试板,经由测试板返回相同长度及内容的数据

        以上配置及测试成功。

六 CAN通讯测试2(中断接收,发送多帧)

【1】发送多帧背景说明:

        实际应用中,往往需要收到一组请求之后,发送多组数据。尝试直接在 COM_Tx 函数中紧跟着第一帧数据发送之后再发一帧,有了如下操作:

        实测上述连续发送的方法是错误的,因为时间间隔太短,前一组通过主缓冲器(PTB)发送的数据还没有完成就又收到新的发送命令,导致后发送的命令失败

【2】同一个缓冲器连续发送多帧,中间加延时的方法:

        使用同一个主缓冲器(PTB)发送,中间增加1ms延时等待上一帧发送结束在我的系统上实测这个延迟的时间要大于500us,其他环境不绝对):

/**************************************************************************
* 函数名称: COM_Tx
* 功能描述: CAN通信发送测试
* 其他说明: 
**************************************************************************/
void COM_Tx(void)
{
    uint8_t u8Idx = 0;
    
    if(TRUE == u8RxFlag)
    {
        u8RxFlag = FALSE;

        if(1 == stcRxFrame.Cst.Control_f.RTR)
        {
            return;
        }

        //<<主缓冲器Can Tx第一帧
        stcTxFrame.StdID         = 0;
        stcTxFrame.Control_f.DLC = 8;
        stcTxFrame.Control_f.IDE = 0;
        stcTxFrame.Control_f.RTR = 0;
        stcTxFrame.TBUF32_2[0] = 0;
        stcTxFrame.TBUF32_2[1] = 1;
        CAN_SetFrame(&stcTxFrame);
        CAN_TransmitCmd(CanPTBTxCmd);   //PTB发送命令
        
        delay1ms(1);    //若使用单一缓冲器,连续发送失败,需要中间加延时等待上一帧发送结束(实测大于500us)
        
        stcTxFrame.StdID         = 1;
        stcTxFrame.Control_f.DLC = 8;
        stcTxFrame.Control_f.IDE = 0;
        stcTxFrame.Control_f.RTR = 0;
        stcTxFrame.TBUF32_2[0] = 2;
        stcTxFrame.TBUF32_2[1] = 3;
        CAN_SetFrame(&stcTxFrame);
        CAN_TransmitCmd(CanPTBTxCmd);   //PTB发送命令

        CAN_IrqCmd(CanRxIrqEn, TRUE);
    }
}

        测试结果如下:  

【3】使用不同缓冲器发送多帧的方法:

        除了上述同一个缓冲器加延时以实现连续发送多帧数据之外,还可以依赖MCU CAN外设提供的另外几组缓冲器实现连续发送多帧数据的目的(更可靠)

        由芯片说明书可以看到,主缓冲器(PTB)只能缓冲一帧数据。而另外的副缓冲器(STB)可以缓冲4帧数据。实现代码如下(收到信号之后,一次发送5帧数据,其中第一帧挂载主缓冲器PTB上,其他四帧挂载副缓冲器STB上):

/**************************************************************************
* 函数名称: COM_Tx
* 功能描述: CAN通信发送测试
* 其他说明: 
**************************************************************************/
void COM_Tx(void)
{
    uint8_t u8Idx = 0;
    
    if(TRUE == u8RxFlag)
    {
        u8RxFlag = FALSE;

        if(1 == stcRxFrame.Cst.Control_f.RTR)
        {
            return;
        }

        //<<主缓冲器Can Tx第一帧
        stcTxFrame.StdID         = 0;
        stcTxFrame.Control_f.DLC = 8;
        stcTxFrame.Control_f.IDE = 0;
        stcTxFrame.Control_f.RTR = 0;
        stcTxFrame.TBUF32_2[0] = 0;
        stcTxFrame.TBUF32_2[1] = 1;
        CAN_SetFrame(&stcTxFrame);
        CAN_TransmitCmd(CanPTBTxCmd);   //PTB发送命令
        
//        delay1ms(1);    //若使用单一缓冲器,连续发送失败,需要中间加延时等待上一帧发送结束(实测大于500us)
        
        //<<副缓冲器Can Tx第二帧
        stcTxFrame.StdID         = 1;
        stcTxFrame.Control_f.DLC = 8;
        stcTxFrame.Control_f.IDE = 0;
        stcTxFrame.Control_f.RTR = 0;
        stcTxFrame.enBufferSel   = CanSTBSel;
        stcTxFrame.TBUF32_2[0] = 2;
        stcTxFrame.TBUF32_2[1] = 3;
        CAN_SetFrame(&stcTxFrame);
        //<<副缓冲器Can Tx第三帧
        stcTxFrame.StdID         = 2;
        stcTxFrame.Control_f.DLC = 8;
        stcTxFrame.Control_f.IDE = 0;
        stcTxFrame.Control_f.RTR = 0;
        stcTxFrame.enBufferSel   = CanSTBSel;
        stcTxFrame.TBUF32_2[0] = 4;
        stcTxFrame.TBUF32_2[1] = 5;
        CAN_SetFrame(&stcTxFrame);
        //<<副缓冲器Can Tx第四帧
        stcTxFrame.StdID         = 3;
        stcTxFrame.Control_f.DLC = 8;
        stcTxFrame.Control_f.IDE = 0;
        stcTxFrame.Control_f.RTR = 0;
        stcTxFrame.enBufferSel   = CanSTBSel;
        stcTxFrame.TBUF32_2[0] = 6;
        stcTxFrame.TBUF32_2[1] = 7;
        CAN_SetFrame(&stcTxFrame);
        //<<副缓冲器Can Tx第五帧
        stcTxFrame.StdID         = 4;
        stcTxFrame.Control_f.DLC = 8;
        stcTxFrame.Control_f.IDE = 0;
        stcTxFrame.Control_f.RTR = 0;
        stcTxFrame.enBufferSel   = CanSTBSel;
        stcTxFrame.TBUF32_2[0] = 8;
        stcTxFrame.TBUF32_2[1] = 9;
        CAN_SetFrame(&stcTxFrame);
        CAN_TransmitCmd(CanSTBTxAllCmd);    //STB所有帧发送命令

        CAN_IrqCmd(CanRxIrqEn, TRUE);
    }
}

        测试结果如下:

        综上,CAN通信收发测试完成。

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

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

相关文章

实在智能携手品牌商家,在活动会面中共谋发展

金秋十月&#xff0c;丰收的季节&#xff0c;也是商家们在双11大展拳脚的时刻。为迎战一年一度的双11大促&#xff0c;品牌商家在10月份卯足劲&#xff0c;制定一系列营销方案&#xff0c;争取为店铺带来更多流量和订单。 其中&#xff0c;舍得、同科医药、梅子熟了、宝洁、维…

海上风电应急救援vr模拟安全培训提高企业风险防范能力

相比传统的发电厂&#xff0c;海上风电作业积累的经验少&#xff0c;风险高&#xff0c;因此为了规范施工人员的行为和操作&#xff0c;保障生产安全进行&#xff0c;开展海上风电VR安全培训具有重要意义。 有助于提高员工的安全意识 通过模拟真实的海上风电作业环境&#xff0…

基于元学习神经网络的类人系统泛化

Nature 上介绍了一个关于AI在语言泛化方面的突破性研究。科学家们创建了一个具有人类般泛化能力的AI神经网络&#xff0c;它可以像人类一样将新学到的词汇融入现有词汇&#xff0c;并在新环境中使用它们。与ChatGPT 相比&#xff0c;该神经网络在系统性泛化测试中表现得更好。 …

AMD Ryzen AI 暂仅支持 Windows,Linux 系统有望后续支持

近日消息&#xff0c;最新的 AMD Ryzen 7040 系列笔记本电脑配备了基于 Xilinx IP 的专用 AI 引擎&#xff0c;名为“Ryzen AI”&#xff0c;可以加速 PyTorch 和 TensorFlow 等机器学习框架的运行。不过目前这个 Ryzen AI 只支持微软 Windows 系统。但是如果有足够的客户需求&…

NLP实践——中文指代消解方案

NLP实践——中文指代消解方案 1. 参考项目2. 数据2.1 生成conll格式2.2 生成jsonline格式 3. 训练3.1 实例化模型3.2 读取数据3.3 评估方法3.4 训练方法 4. 推理5. 总结 1. 参考项目 关于指代消解任务&#xff0c;有很多开源的项目和工具可以借鉴&#xff0c;比如spacy的基础模…

恒驰服务 | 华为云数据使能专家服务offering之数仓建设

恒驰大数据服务主要针对客户在进行智能数据迁移的过程中&#xff0c;存在业务停机、数据丢失、迁移周期紧张、运维成本高等问题&#xff0c;通过为客户提供迁移调研、方案设计、迁移实施、迁移验收等服务内容&#xff0c;支撑客户实现快速稳定上云&#xff0c;有效降低时间成本…

【Elasticsearch 未授权访问漏洞复现】

文章目录 一、漏洞描述二、漏洞复现三、修复建议 一、漏洞描述 ElasticSearch是一个基于Lucene的搜索服务器。它提供了一个分布式多用户能力的全文搜索引擎&#xff0c;基于RESTful web接口。Elasticsearch是用Java开发的&#xff0c;并作为Apache许可条款下的开放源码发布&am…

应用在阀门控制中的直流有刷驱动芯片

控制阀又称阀门&#xff0c;是流体运送系统中的控制部件&#xff0c;具有导流、截流、调节、节流、防止倒流、分流或溢流卸压等功能。阀门是一种用于控制流体&#xff08;液体、气体、粉体等&#xff09;流动的装置&#xff0c;广泛应用于工业生产、建筑、农业、能源等领域。 …

图像二值化阈值调整——cv2.threshold方法

二值化阈值调整&#xff1a;调整是指在进行图像二值化处理时&#xff0c;调整阈值的过程。阈值决定了将图像中的像素分为黑色和白色的界限&#xff0c;大于阈值的像素被设置为白色&#xff0c;小于等于阈值的像素被设置为黑色。 方法一&#xff1a; 取阈值为 127&#xff0c;…

京东科技埋点数据治理和平台建设实践 | 京东云技术团队

导读 本文核心内容聚焦为什么要埋点治理、埋点治理的方法论和实践、奇点一站式埋点管理平台的建设和创新功能。读者可以从全局角度深入了解埋点、埋点治理的整体思路和实践方法&#xff0c;落地的埋点工具和创新功能都有较高的实用参考价值。遵循埋点治理的方法论&#xff0c;…

Mac终端学习

命令1&#xff1a;ifconfig 作用&#xff1a;列出本机所有的网络设备以及其上面的配置&#xff0c;主要指的是ip地址和mac地址 其他用法&#xff1a;sudo ifconfig en4 add 10.10.10.12 netmask 255.255.255.0 作用&#xff1a;给en4加入别的网段 其他用法&#xff1a;sudo i…

延时摄影视频制作工具 LRTimelapse mac中文版特点介绍

lrTimelapse mac是一款适用于 Windows 和 macOS 系统的延时摄影视频制作软件&#xff0c;可以帮助用户创建高质量的延时摄影视频。该软件提供了直观的界面和丰富的功能&#xff0c;支持多种时间轴摄影工具和文件格式&#xff0c;并具有高度的可定制性和扩展性。 lrTimelapse ma…

ICLR 2022)ODConv:即插即用的动态卷积 (附代码)

论文地址&#xff1a;Omni-Dimensional Dynamic Convolution | OpenReview 代码地址&#xff1a;https://github.com/OSVAI/ODConv/blob/main/modules/odconv.py 1.是什么&#xff1f; ODConv是一种动态卷积算法&#xff0c;它的原理是在卷积过程中&#xff0c;根据输入数据的…

Azure机器学习 - 使用与Azure集成的Visual Studio Code实战教程

本文介绍如何启动远程连接到 Azure 机器学习计算实例的 Visual Studio Code。 借助 Azure 机器学习资源的强大功能&#xff0c;使用 VS Code 作为集成开发环境 (IDE)。 在VS Code中将计算实例设置为远程 Jupyter Notebook 服务器。 关注TechLead&#xff0c;分享AI全维度知识。…

【密评】商用密码应用安全性评估从业人员考核题库(十七)

商用密码应用安全性评估从业人员考核题库&#xff08;十七&#xff09; 国密局给的参考题库5000道只是基础题&#xff0c;后续更新完5000还会继续更其他高质量题库&#xff0c;持续学习&#xff0c;共同进步。 4001 多项选择题 网络和通信安全层面的通信主体一般包括哪些&…

Unity屏幕中涂鸦

LineRenderer LineRenderer是Unity中的一个组件&#xff0c;用于在场景中绘制简单的线段。 LineRenderer组件允许你通过设置一系列顶点来定义线段的形状和外观。它会根据这些顶点自动在场景中绘制出线段。 下面是LineRenderer的一些重要属性和方法&#xff1a; positionCou…

C++使用栈实现简易计算器(支持括号)

使用C实现&#xff0c;使用系统自带stac 支持括号处理支持小数计算支持表达式有效性检查支持多轮输入。 运行结果示例&#xff1a; 代码&#xff1a; #include <iostream> #include <stack> #include <string> using namespace std;//判断是否是数字字符 …

基于MFC的串口通信(Mscomm)

1、串口通信的概述&#xff1a; 串口是一种重要的通信资源&#xff0c;例如鼠标口、USB接口都是串口。串行端口是CPU和串行设备间的编码转换器。当数据从CPU经过端口发送出去的时候&#xff0c;字节数据会被转为串行的位&#xff0c;在接收数据时&#xff0c;串行的位被转换为…

k8s调度约束

List-Watch Kubernetes 是通过 List-Watch的机制进行每个组件的协作&#xff0c;保持数据同步的&#xff0c;每个组件之间的设计实现了解耦。 List-Watch机制 工作机制&#xff1a;用户通过 kubectl请求给 APIServer 来建立一个 Pod。APIServer会将Pod相关元信息存入 etcd 中…

【ROS系列】坐标系转换介绍和对齐

一、坐标系简介 本篇文章介绍&#xff1a;ECEF、ENU、UTM、WGS-84坐标系&#xff08;LLA) 1.1、ECEF坐标系 ECEF坐标系也叫地心地固直角坐标系。 原点&#xff1a;地球的质心&#xff0c; x轴&#xff1a;原点延伸通过本初子午线&#xff08;0度经度&#xff09;和赤道&am…