正点原子[第二期]Linux之ARM(MX6U)裸机篇学习笔记-19讲 串口实验UART

前言:

本文是根据哔哩哔哩网站上“正点原子[第二期]Linux之ARM(MX6U)裸机篇”视频的学习笔记,在这里会记录下正点原子 I.MX6ULL 开发板的配套视频教程所作的实验和学习笔记内容。本文大量引用了正点原子教学视频和链接中的内容。

引用:

正点原子IMX6U仓库 (GuangzhouXingyi) - Gitee.com

《【正点原子】I.MX6U嵌入式Linux驱动开发指南V1.5.2.pdf》

正点原子资料下载中心 — 正点原子资料下载中心 1.0.0 文档

正文:

本文是 “正点原子[第二期]Linux之ARM(MX6U)裸机篇--第19 讲” 的读书笔记。第19讲主要是介绍I.MX6U处理器的UART串口并实现串口UART的字符格式化打印实验。本节将参考正点原子的视频教程第18讲和配套的正点原子开发指南文档进行学习。

0. 概述

不管是单片起开发还是嵌入式Linux开发,串口都是最常用到的外设。可以通过串口将开发板与电脑相连,然后电脑上通过串口调试助手来调试程序。还有很多的模块,比如蓝牙,GPS,GPRS 等都使用的串口来与主控进行通信的,在嵌入式LInux中一般使用串口作为控制台,所以掌握串口是必备的技能。本章我们就来学习如何驱动 I.MX6U 上的串口,并使用串口和电脑进行通信。

1. I.MX6U 串口简介

1.1 UART简介

1. UAR通信格式

串口的全称叫做串行接口,通常也叫做 COM 接口,串行接口指的是数据一个一个顺序传输,通信线路简单。使用两条线即可实现双向通信,一条用于发送,一条用于接收。串口通信距离远,但是速度相对会低,串口是一种很常用的工业接口。I.MX6u 自带的UART外设就是串口的一种,UART的全称是 Universal Asynchronous Receiver/Transimitter ,也就是异步串行收发器。既然有异步串行收发器,那肯定也有同步串行收发器,学过STM32的同学应该知道,STM32除了UART(Univeral Asynchronous Receiver/Transmitter)外,还另外有一个叫做USART的东西。USART的全称是 Universal Synchronous/Asynchronous Receiver/Transmitter,也就是异步/同步串行收发器。想不UART多了一个同步功能,在硬件上体现出来就是多了一条时钟线。一般USART是可以作为 UART使用的,也就是不使用其同步功能。

UART作为串口的一种,器工作原理也是将数据一位一位的的进行传输,发送和接受各使用一条线,因此通过UART接口与外界通信最少需要三条线:TXD(发送),RXD(接收)和GND(地线)。图21.1.1.1 就是UART通信的格式:

图 21.1.1.1 中各位的含义如下:
 

描述
空闲位数据线在空闲状态的时候逻辑为“1”状态,也就是高电平表示,表示没有数据,数据线空闲没有数据传输。
起始位当要传输数据的时候线传输一个逻辑“0”,也就是将数据线拉低,表示开始数据传输。
数据位数据位就是实际要传输的数据,数据位数可选择5~8位,我们一般都是按照字节传输数据的,一个自己8位,因此数据位通常是8位的。低位在前,先传输,高位最后传输。
奇偶校验位这是对数据中“1”的位数进行奇偶校验用的,可以不使用奇偶校验功能。
停止位数据传输完成的标志位,停止位的位数可以选择1位,1.5位,或2位高电平,一般都选择1位停止位。
波特率波特率就是UART数据传输的速率,也就是每秒传输的数据位数,一般选择9600, 19200, 115200 等。

2. UART的电平标准

UART一般的接口电平有TTL和RS-232,一般开发板上都有TXD和RXD这样的引脚,这些引脚低电平表示逻辑0,高电平表示逻辑1,这个就是TTL电平。RS-232采用差分线,-3~-15V 表示逻辑1,+3~+15V 表示逻辑0。一般21.1.1.2 中的接口就是TTL电平。

图 21.1.1.2 中的模块就是 USB 转 TTL 模块,TTL接口的部分有VCC,GND,RXD,TXD,RTS和CTS。RTS和CTS基本用不到,使用的时候通过杜邦线和其他模块的TTL接口相连即可。

RS-232电平需要使用DB-9接口,I.MX6U-ALPHA 开发板上的COM3(UART3)接口就是RS-232接口的,如图 21.1.1.3 所示:

由于现在的电脑都没有DB9接口了,取而代之的是USB接口,所以就催生了很多USB转串口TTL芯片,比如CH340,PL2302等。通过这些芯片就可以实现串口TTL转USB。I.MX6U-APLHA开发板就是用了CH340芯片来来完成UART1和电脑之间的连接,只需要一条USB线即可,如图 21.1.1.4 所示。

2. I.MX6U UART 简介

上一小节介绍了UART接口,本小节具体看一下I.MX6U的UART接口,I.MX6U 一共有8个UART,其主要特性如下:

  1. 兼容 TIA/EIA-232F 标准,速度最高可到 5Mbit/S。
  2. 支持串行 IR 接口,兼容 IrDA,最高可到 115.2Kbit/s。
  3. 支持 9 位或者多节点模式(RS-485)。
  4.  1 或 2 位停止位。
  5. 可编程的奇偶校验(奇校验和偶校验)。
  6. 自动波特率检测(最高支持 115.2Kbit/S)

I.MX6U 的UART功能很多,但是我们本章就只是用到最基本的串口功能,关于 UART 其它功能的介绍请参考《I.MX6ULL 参考手册》第 3561 页的“Chapter 55 Universal Asynchronous Receiver/Transmitter(UART)”章节。
 

UART的时钟源石油寄存器CCM_CSCDR1的 UART_CLK_DEL(bit) 位来选择的,当为0的时候UART的时钟源为 pll3_80m(80MHz),如果为 1 的时候 UART 的时钟源为 osc_clk(24M),一般选择 pll3_80m 作为 UART 的时钟源。

CCM_CSCDR1的UART_CLK_PODF(bit5:0)为是UART的时钟分频值,可设置0~63,分别对应1~64分频,因此最终进入UART的时钟为 80MHz = PLL3 480MHz/6。

80MHz = PLL3 480MHz/6

2.1 UARTx_UCRx 寄存器

接下来看一下UART的几个重要的寄存器,第一个就是UART的控制寄存器去,寄UART1_UCR1(x=1~8),此寄存器的结构如下图所示:

描述
ADBR(bit14)
 
自动波特率检测使能位,为 0 的时候关闭自动波特率检测,为1 的时候使能自动波特率检测。
UARTEN(bit0)UART 使能位,为 0 的时候关闭 UART,为 1 的时候使能 UART
 

接下俩看一下UART的控制寄存器2,即 UARTx_UCR2,此寄存器的结构如图 21.1.2.2 所
示:

描述
IRTS bit[14]为0的时候使能RTS引脚,为1的时候忽略RTS引脚
PREN bit[8]Parity Check 奇偶校验使能位,为0的时候关闭奇偶校验,为1的时候使能机构校验。Odd Number 奇数,Even Numver 偶数。
PROE bit[7]奇偶校验模式选择位,开启奇偶校验以后此位如果为 0 的话就使用偶校验,此位为 1 的话就使能奇校验
STPB bit[6]Stop Bit 停止位数量,为 0 的话 1 位停止位,为 1 的话 2 位停止位
WS bit[5]数据位长度,为 0 的时候选择 7 位数据位,为 1 的时候选择 8 位数据位
 
TXEN bit[2]发送使能位,为0的时候关闭UART的发送功能,为1的时候打开UART的发送功能。
RXEN bit[1]接收使能位,为0的时候关闭UART的接收功能,为1的时候打开UART的接收功能。
SRST bit[0]软件复位,为0的时候软件复位UART,为1的时候表示软件复位完成。复位万策划给你以后此位会自动置1,表示复位万策划给你。此位只能写0,写1会被忽略掉。

接下来看一下 UARTx_UCR3 寄存器,此寄存器结构如图 21.1.2.3 所示


本章实验就用到了寄存器 UARTx_UCR3 中的位 RXDMUXSEL(bit2),这个位应该始终为 1,这个在《I.MX6ULL 参考手册》第 3624 页有说明。

这个正点原子的视频教程里有说,当时正点原子的左盟主在准备I.MX6U UART 视频教程备课的时候写的源码这里没有置1,找了很久才找到是这里UARTx_UCR3 这个寄存器这里RXDMUXSEL 位没有置1的原因。这里是经验教训。

接下来看一下寄存器 UARTx_USR2,这个是 UART 的状态寄存器 2,此寄存器结构如图21.1.2.4 所示

寄存器 UARTx_USR2 用到的重要位如下:

  • TXDC(bit3):发送完成标志位,为 1 的时候表明发送缓冲(TxFIFO)和移位寄存器为空,也就是发送完成,向 TxFIFO 写入数据此位就会自动清零。
  • RDR(bit0):数据接收标志位,为 1 的时候表明至少接收到一个数据,从寄存器UARTx_URXD 读取数据接收到的数据以后此位会自动清零。
     

2.2 UART 分频值寄存器

接 下 来 看 一 下 寄 存 器 UARTx_UFCR 、 UARTx_UBIR 和 UARTx_UBMR , 寄 存 器
UARTx_UFCR 中我们要用到的是位 RFDIV(bit9:7),用来设置参考时钟分频,设置如表 21.1.2.1
所示:

  • Ref Freq:经过分频以后进入 UART 的最终时钟频率。
  • UBMR:寄存器 UARTx_UBMR 中的值。
  • UBIR:寄存器 UARTx_UBIR 中的值。

通过 UARTx_UFCR 的 RFDIV 位、 UARTx_UBMR 和 UARTx_UBIR 这三者的配合即可得到我们想要的波特率。比如现在要设置 UART 波特率为 115200,那么可以设置 RFDIV 为
5(0b101),也就是 1 分频,因此 Ref Freq=80MHz。设置 UBIR=71, UBMR=3124,根据上面的公式可以得到:

2.3 UART的 RXD 和TXD寄存器去

最后来看一下寄存器 UARTx_URXD 和 UARTx_UTXD,这两个寄存器分别为 UART 的接收和发送数据寄存器,这两个寄存器的低八位为接收到的和要发送的数据。读取寄存器UARTx_URXD 即可获取到接收到的数据,如果要通过UART发送数据,直接将数据写入到寄存器UARTx_UTXD 即可。

注意:

从I.MX6U UART 寄存器的的UARTx_URXD和UARTx_UTXD寄存器里读取和发送数据的时候,需要检查 UARTx_USR2 寄存器里的 TXDC (Transmit Data Complete)和 RDR (Receiver Data Ready) 标志位,检查到标志位为1说明发送完成或者接收数据准备好,才可以读取/写入。


关于 UART 的寄存器就介绍到这里,关于这些寄存器详细的描述,请参考《I.MX6ULL 参考手册》第 3608 页的 55.15 小节。

3. 编写UART驱动程序

本章我们使用 I.MX6U 的 UART1 来完成开发板与电脑串口调试助手之间串口通信, UART1 的配置步骤如下:

  1. 设置UART1的时钟源
    设置UART1的时钟源为 pll3_80m,设置寄存器 CCM_CSCDR1 的 UART_CLK_SEL 位为0即可。
  2. 初始化UART1
    初始化UART1所使用的IO,设置UART1的寄存器 UARTx_UCR1~UARTx_UCR3,设置内容包括波特率,奇偶校验位,停止位,数据位等等。
  3. 使能UART1
    UART1初始化完成以后就可以使能UART1了,设置寄存器UARTx_UCR1的为 UARTEN 为1。
  4. 编写UART1的数据收发函数
    编写两个用于UART1的数据收发操作。

3.1 硬件原理分析

本实验用到的资源如下:

  1. 一个LED灯:LED0
  2. 串口1

I.MX6U-ALPHA 开发板串口 1 硬件原理图如图 21.2.1 所示:

将串口 1 的 RXD、 TXD 两个引脚分别与 P116、 P117 连接一起,如图 21.2.2 所示

硬件连接设置好以后就可以开始软件编写了,本章实验我们初始化好 UART1,然后等待 SecureCRT  给开发板发送一个字节的数据,开发板接收到 SecureCRT 发送过来的数据以后在通过串口 1 发送给 SecureCRT。

3.2 UART IO引脚复用

查阅《I.MX6ULL 参考手册》3563页的“55.2 External Signals” 小节,可以看到I.MX6U芯片的UART1 引脚IO复用的方法。

I.MX6U 芯片的封装引脚复用 IOMUXC,UART1_TX 可以来自于两种的芯片引脚复用:

  • 一种是复用UART1_RX_DATA 的复用 ALT0
  • 另一种是复用 GPIO1_IO02 的复用 ALT

因为在I.MX6U 芯片的 GPIO1_IO03 在正点原子的 I.MX6ULL ALPHA/Mini 开发板上已经使用为LED0 的GPIO输出引脚,所以我们应该使用 UART1 的 UART1_RX_DATA 和 UART1_TX_DATA 引脚复用。 

对 IOMUXC_UART1_TX_DATA IO接口复用

4. 源码编写

bsp/bsp_uart.c 源码如下,参考了正点原子提供的示例源码:

#include "bsp_uart.h"
#include "bsp_beep.h"void uart_init(void)
{/* UART 时钟源选择 */CCM->CSCDR1 &= ~(1 << 6);		/* 选择pll3_80m作为时钟源 */CCM->CSCDR1 &= ~(0x3f << 0);	/* 选择1分频 */uart_disable(UART1);uart_io_int();					/* UART IO复用 */uart_reset();					/* UART 软件复位 *//* UARTx_UCR1 */UART1->UCR1 = 0;			/* 关闭UART1, 关闭UART1自动波特率检测 *//* UARTx_UCR2 */UART1->UCR2 &= ~(1 << 8);	/* Parity 奇偶校验不启用 */UART1->UCR2 &= ~(1 << 6);	/* StopBit 停止位,1个停止位 */UART1->UCR2 |= (1 << 14);UART1->UCR2 |= (1 << 5);	/* WordSize 字宽,8个数据位 */UART1->UCR2 |= (1 << 2);	/* TXEN 使能Tx */UART1->UCR2 |= (1 << 1);	/* RXEN 使能Rx *//* UARTx_UCR3 */UART1->UCR3 |= (1 << 2);	/* I.MX6ULL手册指出此位必须被设置为1 *//* UART波特率设置 */UART1->UFCR &= ~(7 << 7);UART1->UFCR |= (5 << 7);	/* 1分频 */UART1->UBIR |= (71 & 0xFFFF);UART1->UBMR |= (3124 & 0xFFFF);uart_enable(UART1);
}void uart_io_int(void)
{/* 0x10B0 = 0001 0000 1011 0000* bit[0]		SRE			0 低压摆率* bit[2:1]		Reserved	00 保留未使用* bit[5:3]		DSE			110 驱动能力DSE_6_R0_6 — R0/6* bit[7:6]		SPEED		10  medium(100MHz)* bit[10:8]	Reserved	000 保留未使用* bit[11]		ODE			0 Open-Drain 开漏输出(推挽输出)* bit[12]		PKE			1 Pull/Keeper使能* bit[13]		PUE			0 Pull/Keeper Select 选择,0选择Keeper* bit[15:14]	PUS			00 Pull UP/Down Config, 00选择100K Ohm Pull Down*//* UART1_TX_DATA IO复用 */IOMUXC_SetPinMux(IOMUXC_UART1_TX_DATA_UART1_TX, 0);IOMUXC_SetPinConfig(IOMUXC_UART1_TX_DATA_UART1_TX, 0x10B0);/* UART1_RX_DATA IO复用 */IOMUXC_SetPinMux(IOMUXC_UART1_RX_DATA_UART1_RX, 0);IOMUXC_SetPinConfig(IOMUXC_UART1_RX_DATA_UART1_RX, 0x10B0);
}void uart_enable(UART_Type *base)
{base->UCR1 |= (1 << 0);
}void uart_disable(UART_Type *base)
{base->UCR1 &= ~(1 << 0);
}unsigned char getc(void)
{unsigned char c;while((UART1->USR2 & (1 << 0)) == 0);	/* 检查UART RDR标志位,确认Rx是否完成 */c = (UART1->URXD & 0xFF);return c;
}void putc(unsigned char c)
{/* 检查UART TXDC标志位,确认Tx是否完成 */while((UART1->USR2 & (1 << 3)) == 0);UART1->UTXD = (c & 0xFF);
}void puts(char *str)
{char *p = str;while(*p){putc(*p++);}
}void uart_reset(void)
{UART1->UCR2 &= ~(1 << 0);				/* UART 软件复位 */while((UART1->UCR2 & (1<< 0)) == 0);	/* 等待UART软件复位完成 */
}

4.1 编译并修改Makefile

执行‘make’命令编译时,编译器提示如下错误:

编译器提示错误的是因为在GCC编译器里有内置的(build-in)的 'putc()' 和 'puts()' 函数和我们自己的uart 驱动程序源码里定义的 'putc', 'puts' 函数冲突。所以修改Makefile编译命令,对 “*.c” 文件的编译增加 "-no-buildin" 编译选项。

5. 编译烧写SD卡验证实验结果


译修改主频后源码烧录SD卡验证本节的 I.MX6U UART串口实验。预期烧录SD卡后正点原子I.MX6ULL ALPHA/Mini 开发板后,UART串口可以在串口工具,如SecureCRT或者Xshell上打印字符串输出。

我本地验证的结果是基于GPT定时器的高精度延时实验结果正常,UART串口可以在串口工具XShell上打印字符串。

6. 总结和实验遇到的问题记录

6.1 问题1:烧录UART.bin镜像上电启动之后,执行到某个函数卡死

错误原因是寄存器写错了,软件复位应该是UCR1,错误的写成了USR1。修复之后就可以了。

6.2 问题2:烧录UART.bin镜像上电启动之后,串口没有输出。

问题2错误原因,是这里错写成和0x0F进行与运算,这样打印出来的字符就只有低4位有效。正确的应该是和0xFF进行与运算,因为是最后的低8位有效。修复之后串口就可以正常打印字符串了。

6.3 问题3:需要在UARTx->UCR2 中配置bit 14 忽略 "RTS pin" 引脚

需要在UARTx->UCR2 中配置bit 14 忽略 "RTS pin" 引脚。

如果UART不忽略 RTS (Reqeust To Send pin 引脚信号),那么UART的 tx 方向在收到 RTS pin (电平有效)信号之前不会发送数据。对于我们的这个UART实验也就是卡在 uart tx 发送数据的地方。 

7. 附加信息:UART RTS和CTS

参考如下链接介绍了UART 的 RTS 引脚和 CTS引脚的作用:

RTSRequest To Send
CTSClear To Send

串口流控 UART 中 CTS RTS RX TX (串口模组和MCU直接的通信)_cts low to txd valid-CSDN博客文章浏览阅读1.6w次,点赞11次,收藏66次。硬件介绍:_cts low to txd validhttps://blog.csdn.net/EAyayaya/article/details/112801769

RTS:(Request To Send 请求发送)

模组的RTS是给mcu说准备好了,低电平,如果模组没有准备好,MCU给模组发数据,可能会丢包 

CTS:(Clear To Send 清除发送,允许发送)

模组的cts必须要外部的mcu给低电平,模组才能发送数据, 

 RXD: (Receive Data 接收数据)

接收数据

TXD:( Transmit Data 发送数据)

发送数据 

8. 结束

本文至此结束。
 

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

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

相关文章

水平垂直居中的六种方法

1. 使用 Flexbox Flexbox 是一个现代的布局模型&#xff0c;可以轻松实现元素的水平和垂直居中。 .container {display: flex;justify-content: center; /* 水平居中 */align-items: center; /* 垂直居中 */height: 100vh; /* 视窗高度 */ }2. 使用 Grid CSS Grid 是另一种强…

【Golang】 Go语言中如何将参数添加到URL中

文章目录 前言一、参数解释二、代码实现三、总结 前言 在开发Web应用程序时&#xff0c;我们经常需要将参数添加到URL中&#xff0c;以便将数据发送到服务器。这些参数通常用于GET请求&#xff0c;以向服务器传递查询条件或其他信息。在Go语言中&#xff0c;我们可以使用net/u…

Vivado IP核的快速入门 官方手册和例程

在IP Catalog中选择要使用的IP核&#xff0c;可以查看支持的器件与资料。 在设计源sources页面中选中配置完成的IP核点击右键选择 Open IP Example Design&#xff0c;等待工程加载完成即可&#xff0c;可以点击Run Simulation进行功能仿真进行IP核的学习。 参考&#xff1…

Mac Pro中的开源虚拟机UTM安装ubuntu(Applce M1,M2芯片)(1)

MacPro安装UTM 1 UTM 下载UTM虚拟机链接: https://mac.getutm.app/ 建议官网下载&#xff1a; 下载 Ubuntu Arm 64版 下载 Ubuntu Arm 64版链接: https://cn.ubuntu.com/download/server/arm 2 安装UTM 2.1 安装在mac上 2.2 点Open 2.3 建虚拟机### 2.4 点出虚拟机 2.5 O…

系统分析师-案例分析-数据库

系统分析师-案例分析-数据库 更多软考资料 https://ruankao.blog.csdn.net/ 文章目录 系统分析师-案例分析-数据库数据库考察知识点规范化函数依赖范式1NF2NF3NF 规范化问题不规范化反规范化设计反规范化设计同步问题 并发控制性能优化完整性约束视图安全分布式数据库特点优点…

前端性能优化:从系统分析讲到实践策略

前言 在过去几年&#xff0c;我曾经写过几篇和性能优化相关的文章&#xff0c;例如有性能优化方法相关的&#xff0c;有性能监控相关的。但是都只关注于局部&#xff0c;没有从整体上去看待、分析性能优化。所以本文打算尝试从整体上去分析前端性能优化&#xff0c;从性能指标…

使用 RisingWave 和 Redash 处理和可视化实时数据

在创建流处理管道时&#xff0c;需要两个关键组件&#xff1a;一个用于处理和转换数据&#xff0c;一个用于数据可视化。RisingWave 和 Redash 就提供了一个优秀的解决方案。 RisingWave 是一个支持实时数据处理的分布式 SQL 流数据库。它提供增量更新的物化视图&#xff0c;使…

TRICONEX 3720 技术特点

TRICONEX 3720是一款安全管理系统通讯模块&#xff0c;通常用于工业自动化和安全系统中。它的主要作用是在控制系统中处理和管理数据通讯&#xff0c;确保信息在各个组件之间的有效传递。以下是关于TRICONEX 3720的详细介绍&#xff1a; 功能定位&#xff1a;TRICONEX 3720作为…

计算机操作系统总结(1)

1操作系统的概念&#xff08;定义&#xff09;功能和目标 (1)什么是操作系统&#xff1f; &#xff08;2&#xff09;操作系统的功能和目标—作为系统资源的管理者 &#xff08;3&#xff09;操作系统的功能和目标—向上层提供方便易用的服务 &#xff08;4&#xff09;操作系…

IP学习——ospf1

OSPF:开放式最短路径优先协议 无类别IGP协议&#xff1a;链路状态型。基于 LSA收敛&#xff0c;故更新量较大&#xff0c;为在中大型网络正常工作&#xff0c;需要进行结构化的部署---区域划分、ip地址规划 支持等开销负载均衡 组播更新 ---224.0.0.5 224.0.0.6 …

CCF-GESP 等级考试 2023年12月认证C++一级真题解析

2023年12月真题 单选题&#xff08;每题 2 分&#xff0c;共 30 分&#xff09; 第 1 题 以下C不可以作为变量的名称的是( )。 A. CCF GESPB. ccfGESPC. CCFgespD. CCF_GESP 正确答案&#xff1a;A 解析&#xff1a; A: 变量名不能包含空格&#xff0c;因此 “CCF GESP” 是不…

区块链开发:区块链软件开发包装相关解析

区块链开发是指设计、构建和维护基于区块链技术的应用程序或系统的过程。区块链是一种分布式账本技术&#xff0c;它通过去中心化的方式记录和验证数据&#xff0c;确保数据的透明性、不可篡改性和安全性。区块链开发者使用各种编程语言和框架来创建这些应用程序。 在加密货币领…

【Linux】-Linux文件的上传和下载、压缩和解压[9]

目录 前言 一、上传和下载 1、使用finalshell对Linux系统进行上传下载 2、rz、sz命令 二、解压和压缩 1、压缩格式 2、tar命令压缩 3、tar命令压缩 4、zip命令压缩文件 5、unzip命令解压文件 前言 在Linux系统中&#xff0c;文件的上传和下载、压缩和解压是非常重要…

redis 使用场景、比较快的原因

https://www.jianshu.com/p/40dbc78711c8 https://www.cnblogs.com/cye9971-/p/11267431.html redis 读取速度比较快的原因 1、数据直接从缓存读取 也支持持久化&#xff0c;需要子进程去处理 2、单线程模式&#xff0c;省去上下文切换的开销 3、合理的数据结构 4、I/O 多路复…

暴风雨的短视频:成都鼎茂宏升文化传媒公司

暴风雨的短视频&#xff1a;大自然的力与美 ​随着科技的进步和网络的普及&#xff0c;短视频已经成为我们生活中不可或缺的一部分。在这些短暂而精彩的瞬间里&#xff0c;我们得以窥见世界的每一个角落&#xff0c;感受生活的多样性和复杂性。成都鼎茂宏升文化传媒公司而当我…

kotlin基础之lambda编程

在Kotlin中&#xff0c;Lambda表达式是一种非常强大的工具&#xff0c;允许我们以更简洁、更声明式的方式编写代码。Kotlin对集合的操作提供了丰富的函数式API&#xff0c;这使得处理集合数据变得更加直观和高效。以下是如何在Kotlin中使用Lambda表达式、集合的创建与遍历以及集…

ubuntu nginx 配置php 网站

1.安装nginx sudo apt-get install nginx 2.安装php 和php-fpm sudo apt-get install phpsudo apt-get install php-fpm 3.配置ngixn 进入/etc/nginxg/sites-enabled vim default //notice 这里只能有一个default的文件 放两个nginx会把报错 完整配置如下 ## # You sho…

go语言之基本数据类型

文章目录 基础数据类型分类整数类型有符号整数无符号整数默认整数类型 浮点类型复数类型布尔类型字符类型字符串replace字符串获取字符串长度字符串的拼接字符串获取指定位置字符更多string操作 数据类型之间的转换其它基本类型转字符串类型fmt包中的Sprintfstrconv包中函数 字…

开源与闭源:AI模型发展的双重路径之争

前言 随着人工智能&#xff08;AI&#xff09;技术的飞速发展&#xff0c;AI模型的应用已经渗透到各行各业&#xff0c;从医疗、金融到制造、教育&#xff0c;无不受到AI技术的深刻影响。在讨论一个AI模型“好不好”“有没有发展”时&#xff0c;绕不过“开源”和“闭源”两条…

java 解决异常 Class path contains multiple SLF4J bindings

前面 为了使用 j2cache pom中导入了 <dependency><groupId>net.oschina.j2cache</groupId><artifactId>j2cache-core</artifactId><version>2.8.4-release</version> </dependency><dependency><groupId>net.osc…