【嵌入式】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作为时钟源,因此需要先设置RCHstcPLLCfg.enInFreq    = SysctrlPllInFreq4_6MHz;     ///< RCH 4MHzstcPLLCfg.enOutFreq   = SysctrlPllOutFreq36_48MHz;  ///< PLL 输出48MHzstcPLLCfg.enPllClkSrc = SysctrlPllRch;              ///< 输入时钟源选择RCHstcPLLCfg.enPllMul    = SysctrlPllMul12;            ///< 4MHz x 12 = 48MHzSysctrl_SetPLLFreq(&stcPLLCfg);///< 选择PLL作为HCLK时钟源;stcCfg.enClkSrc  = SysctrlClkPLL;///< HCLK SYSCLK/2stcCfg.enHClkDiv = SysctrlHclkDiv1;///< PCLK 为HCLK/8stcCfg.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;    ///< 端口输入/输出值寄存器总线控制模式配置->AHBGpio_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 TxstcTxFrame.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通信收发测试完成。

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Linux中正则表达式等

grep命令&#xff1a;主要作用就是过滤查找文本内容 常用的选项有&#xff1a; -m 数字:匹配几次之后停止&#xff0c;按行匹配&#xff0c;不是按字符个数&#xff0c;例如 -v:取反 例如: -n:显示匹配的行号 例如&#xff1a; -c:仅显示匹配的行数&#xff0c;不显示匹配内…

vue-安装Vue开发者工具

极简插件&#xff1a;下载->开发者模式->拖曳安装->插件详情允许访问文件 网址&#xff1a;https://chrome.zzzmh.cn/index 搜索Vue Devtools 下载下来的安装包先解压 然后点击chrome浏览器的右上角三个点的按钮在里面找到扩展程序这个选项&#xff0c;然后点进去管理…

pycharm使用ssh连接远程jupyter

1. 安装jupyter pip install jupyter2. 生成jupyter_notebook_config.py文件 jupyter notebook --generate-config3. 设置命令参数 jupyter notebook --no-browser --allow-root --port 8900配置Jupyter服务器 将上面的代码复制到命令行实参中&#xff1a;

Qwt 使用QwtDial绘制汽车仪表盘

1.概述 QwtDial是Qwt库中的一个类&#xff0c;用于绘制一个可旋转的仪表盘。 以下是类继承关系&#xff1a; 2.常用方法 void setOrigin(double origin) 设置仪表盘的原点角度&#xff0c;以度为单位&#xff0c;默认为0度&#xff0c;正值为逆时针旋转&#xff0c;负值为顺时…

腾讯云轻量服务器“镜像类型”以及“镜像”选择方法

腾讯云轻量应用服务器镜像类型分为应用镜像、系统镜像、Docker基础镜像、自定义镜像和共享镜像&#xff0c;腾讯云百科txybk.com来详细说下不同镜像类型说明和详细介绍&#xff1a; 轻量应用服务器镜像类型说明 腾讯云轻量应用服务器 应用镜像&#xff1a;独有的应用镜像除了包…

Linux shell编程学习笔记17:for循环语句

Linux Shell 脚本编程和其他编程语言一样&#xff0c;支持算数、关系、布尔、字符串、文件测试等多种运算&#xff0c;同样也需要进行根据条件进行流程控制&#xff0c;提供了if、for、while、until等语句。 之前我们探讨了if语句&#xff0c;现在我们来探讨for循环语句。 Li…

什么是数字展览馆,数字展览馆有什么应用前景

引言&#xff1a; 数字展览馆作为一种新兴的文化艺术展示形式&#xff0c;以数字化技术和虚拟现实为基础&#xff0c;正在逐渐改变传统展览的方式。 一、什么是数字展览馆&#xff1f; 1.定义 数字展览馆是利用数字技术和虚拟现实技术打造的一种线上文化艺术展示平台。通过虚…

【网络协议】聊聊HTTPS协议

前面的文章&#xff0c;我们描述了网络是怎样进行传输数据包的&#xff0c;但是网络是不安全的&#xff0c;对于这种流量门户网站其实还好&#xff0c;对于支付类场景其实容易将数据泄漏&#xff0c;所以安全的方式是通过加密&#xff0c;加密方式主要是对称加密和非对称加密。…

git简单介绍,回车换行问题,倒计时+进度条小程序的实现+代码

目录 git--版本控制工具 介绍 使用 小程序 引入 回车换行问题 缓冲区问题 倒计时 分析 代码 进度条 分析 代码 git--版本控制工具 首先,我们需要下载git : yum install git 介绍 Git是一种分布式版本控制系统&#xff0c;用于跟踪文件和目录的变化并协调多个人之…

常用 sqlite3 命令

本次将向您讲解 SQLite 编程人员所使用的简单却有用的命令。这些命令被称为 SQLite 的点命令&#xff0c;这些命令的不同之处在于它们不以分号 ; 结束。 让我们在命令提示符下键入一个简单的 sqlite3 命令&#xff0c;在 SQLite 命令提示符下&#xff0c;您可以使 用各种 …

新晋“断货王”!西圣Olite开放式耳机掀起抢购热潮,双十一首日售罄!

在双十一首日&#xff0c;西圣olite开放式耳机凭借其超高性价比&#xff0c;一举拿下“断货王”称号&#xff0c;在双十一首日已售罄&#xff0c;现在在各大电商平台均显示无货。 这一消息引起了社交媒体的热议&#xff0c;除了其超高性价比&#xff0c;性能可以媲美千元级别耳…

操作系统 day02(特征和发展历程)

一&#xff0c;操作系统的特征 并发和并行 共享 并发和共享互为存在条件 虚拟 异步 综上所述&#xff1a; 二&#xff0c;操作系统的发展历程

使用Ansible中的playbook

目录 1.Playbook的功能 2.YAML 3.YAML列表 4.YAML的字典 5.playbook执行命令 6.playbook的核心组件 7.vim 设定技巧 示例 1.Playbook的功能 playbook 是由一个或多个play组成的列表 Playboot 文件使用YAML来写的 2.YAML #简介# 是一种表达资料序列的格式,类似XML #特…

阿里云无影升级2.0 云电脑解决方案时代到来

10月31日&#xff0c;杭州云栖大会上&#xff0c;阿里云宣布无影全新升级2.0&#xff1a;从云电脑到云上解决方案&#xff0c;帮助中小企业更便捷地构建云上办公&#xff0c;并开放无影产品及解决方案能力&#xff0c;为生态合作伙伴提供企业云平台&#xff0c;帮助其打造定制化…

Mozilla Firefox 119 现已可供下载

Mozilla Firefox 119 开源网络浏览器现在可以下载了&#xff0c;是时候先看看它的新功能和改进了。 Firefox 119 改进了 Firefox View 功能&#xff0c;现在可以提供更多内容&#xff0c;如最近关闭的标签页和浏览历史&#xff0c;你可以按日期或网站排序&#xff0c;还支持查…

【设计模式】第24节:行为型模式之“模板方法模式”

一、简介 模板方法模式在一个方法中定义一个算法骨架&#xff0c;并将某些步骤推迟到子类中实现。模板方法模式可以让子类在不改变算法整体结构的情况下&#xff0c;重新定义算法中的某些步骤。 模板模式有两大作用&#xff1a;复用和扩展。其中&#xff0c;复用指的是&#…

10.31日模拟赛总结

文章目录 考试时间及策略考试结果考试反思题解A.进步科学B.吉吉没急C.老杰克哒D.季积晓淆 考试时间及策略 没啥好说的&#xff0c;因为好像都不会。所以全场感觉都在罚坐&#xff0c;很痛苦。 考试结果 30 0 50 5 85 考试反思 T1&#xff1a;T1是个神奇状压&#xff0…

【java学习—十一】泛型(1)

文章目录 1. 为什么要有泛型Generic2. 泛型怎么用2.1. 泛型类2.2. 泛型接口2.3. 泛型方法 3. 泛型通配符3.1. 通配符3.2. 有限制的通配符 1. 为什么要有泛型Generic 泛型&#xff0c;JDK1.5新加入的&#xff0c;解决数据类型的安全性问题&#xff0c;其主要原理是在类声明时通过…

通过Google搜索广告传送的携带木马的PyCharm软件版本

导语 最近&#xff0c;一起新的恶意广告活动被发现&#xff0c;利用被入侵的网站通过Google搜索结果推广虚假版本的PyCharm软件。这个活动利用了动态搜索广告&#xff0c;将广告链接指向被黑客篡改的网页&#xff0c;用户点击链接后下载的并不是PyCharm软件&#xff0c;而是多种…

矢量图形编辑软件illustrator 2023 mac中文软件特点

illustrator 2023 mac是一款矢量图形编辑软件&#xff0c;用于创建和编辑排版、图标、标志、插图和其他类型的矢量图形。 illustrator 2023 mac软件特点 矢量图形&#xff1a;illustrator创建的图形是矢量图形&#xff0c;可以无限放大而不失真&#xff0c;这与像素图形编辑软…