【FPGA入门】第六篇、异步串口通信

news2025/1/17 1:38:47

目录

第一部分、相关知识     

1、UART和RS232的区别

2、UART与USART的区别

3、全双工?

4、RS232通信协议

5、波特率

6、如何将外部异步信号变为内部同步信号?

7、什么时间点让FPGA去采集rx线上的数据?

第二部分、串口通信时序图

1、发送线RX流程

1.1、模块图

1.2、端口介绍

1.3、时序波形图

1.4、RX接收代码

2、接收线TX流程

2.1、模块图

2.2、时序波形图

2.3、 TX发送代码

3、Top层代码

第三部分、仿真代码的编写

1、testbench代码

 2、仿真结果

3、上板测试

 第四部分、总结


第一部分、相关知识     

1、UART和RS232的区别

  • UART就是一堆电路,是异步串行通信的一种电路实现;
  • RS232属于异步串行通信方式,跟UART相比,区别仅在于UART使用的是TTL电平,而RS232使用的是另一种电平标准。

2、UART与USART的区别

UART:UART 的全称叫做通用异步收发传输器。

  • 将数据在串行通信和并行通信间的传输转换。通俗的讲就是把多比特的数据转化为单比特的数据,或者把单比特的数据转化为多比特的数据。工作原理是将数据的每个 bit 一位接一位地传输。
  • UART 是一种通用串行数据总线,用于异步通信。该总线双向通信,可以实现全双工传输和接收

USART:通用同步和异步收发器

  • 当进行异步通信时,这两者是没有区别的。
  • 区别在于USART比UART多了同步通信功能,因此USART可以提供一个主动的时钟线。使用外部时钟使 USART 的数据速率远高于标准 UART 的数据速率。

3、全双工?

        全双工:表示发送和接收不会相互影响。

4、RS232通信协议

        rs232 是 uart 的一种,有两根线,分别是 rx 和 tx,这两根线都是 1 比特位宽的。其中 rx 是接收线, tx 是发送线。

  • rx,位宽为 1 比特, pc 机通过串口往 FPGA 发 8 比特数据时, FPGA 通过串口线 rx 一位一位地接收,从最低位到最高位依次接收,最后在 FPGA 里面位拼接成8 比特数据。
  • tx,位宽为 1 比特, FPGA 通过串口往 pc 机发 8 比特数据时, FPGA 把 8 比特数据通过 tx 线一位一位的传给 pc 机,从最低位到最高位依次发送,最后上位机通过串口助手把这一位一位的数据位拼接成 8 比特数据。

        在不发送或者不接收数据的情况下, rx 和 tx 处于空闲状态,此时 rx 和 tx 线都保持高电平

        如果有数据传递,首先会有一个起始位(0),然后是 8 比特的数据位,接着有 1 比特的停止位

        如果停止位以后不再发数据,将进入空闲状态,否则又将数据线拉低(进入起始位状态)。

        注意:起始位和停止位各算一位,因此一共是10位数据,而停止位不需要去做判断。

5、波特率

        波特率在串口通信时的速率,单位时间内载波变化的次数,这里选用的是9600Bd,即发送一比特数据需要的时间为 1/9600 秒。

        用串口发送或者接收数据(起始位、数据位、停止位)时,每发送或者接收一位数据的时间都需要 1 个波特,即 1/9600 秒。

        串口发送或者接收一比特数据的时间为一个波特(1/9600),因此如果用 50M的系统时钟来计数,就需要记数 cnt=(1/9600s)/20ns≈5208 个系统时钟,才再次发送或者接收下一个数据。

        注意:cnt=(1/9600s)/20ns≈5208.33 ,累计误差0.33 * 10 = 3ns,这个误差太小了,是不会影响到传输的。而且计数器每次都是从0开始计数,因此,该误差也不会累加。

        上位机通过串口发 8 比特数据时,会自动在发 8 位有效数据前发一个波特时间的起始位,也会自动在发完 8 位有效数据后发一个停止位。同理,串口助手接收 fpga 发送的数据前,必须检测到一波特的起始位才会接收数据,接收完数据后,再接收一个停止位,所以 FPGA 通过串口除了发数据以外,还要发起始位和停止位。

6、如何将外部异步信号变为内部同步信号?

        通过前面的介绍可知,RS232为通用异步收发传输器,那么rx传到FPGA内部的数据与系统的时钟是不匹配的,没有任何逻辑关系,因此要借助寄存器将rx进行打两拍操作后,才能用来进行逻辑运算。

        这么做的目的是尽量降低亚稳态,如果不做这样 的处理,就可能导致采集的数据不稳定,只有做了亚稳态处理后的数据才在后面使用。

7、什么时间点让FPGA去采集rx线上的数据?

        外部的数据通过 rx 线传到 FPGA, FPGA 采数据时,在中间时刻采数可以保证采的数据最 稳定。

第二部分、串口通信时序图

1、发送线RX流程

        1.1、模块图

         1.2、端口介绍

         1.3、时序波形图

        1.4、RX接收代码

// -----------------------------------------------------------------------------
// Copyright (c) 2014-2023 All rights reserved
// -----------------------------------------------------------------------------
// Author : BigFartPeach
// CSDN   : 大屁桃
// E-mail : 2624507313@qq.com
// File   : uart_rx.v
// Create : 2023-06-20 12:45:49
// -----------------------------------------------------------------------------
module uart_rx(
	input wire sclk,
	input wire s_rst_n,
	input wire rx,
	output reg [7:0]po_data,
	output reg po_flag
	);
// //9600波特率参量
// parameter BAUD_END = 5207;//波特率9600,接收间隔最大计数MAXEND = 50M/9600 = 5208.33 - 1 = 5207
// parameter BAUD_HALF = 2602;//5208/2 - 1 = 2603,要在2603这里点看到波形
// parameter BAUD_BIT_WIDTH = 12;//5207需要13位来保存,[12:0]

//115200波特率参量
parameter BAUD_END = 433;//波特率115200,接收间隔最大计数MAXEND = 50M/115200 = 434.02 - 1 = 433
parameter BAUD_HALF = 216;//434/2 - 1 = 216
parameter BAUD_BIT_WIDTH = 8;//5207需要9位来保存,[8:0]

//变量
reg rx1;//用来降低亚稳态
reg rx2;//用来降低亚稳态
reg rx2_reg;//用来打拍rx2,判断下降沿

reg rx_flag;//rx信号有效标志位

reg [12:0]baud_cnt;//9600波特率的1个波特时间1/9600s的计数器

reg capture_flag;//中间采集数据的时间点

reg [3:0]capture_cnt;//采集数据的计数器

//利用移位寄存器打拍,打拍信号可以不加复位
always @(posedge sclk) begin
	{rx2_reg,rx2,rx1} <= {rx2,rx1,rx};
end

//rx有效标志位,rx_flag
always @(posedge sclk or negedge s_rst_n) begin
	if (!s_rst_n) begin
		rx_flag <= 1'b0;
	end
	//rx_flag受baud_cnt影响,究竟是什么样的影响?
	else if(capture_flag == 1'b1 && capture_cnt == 'd8)begin
		rx_flag <= 1'b0;
	end
	else if (rx2 == 1'b0 && rx2_reg == 1'b1) begin
		rx_flag <= 1'b1;
	end
end

//一波特时间对应的计数器
always @(posedge sclk or negedge s_rst_n) begin
	if (!s_rst_n) begin
		baud_cnt <= 'd0;
	end
	else if(baud_cnt == BAUD_END)begin
		baud_cnt <= 'd0;
	end
	//baud_cnt会计数到2603,若判断条件换成rx_flag == 1'b0,baud_cnt会计数到2604
	else if(capture_flag == 1'b1 && capture_cnt == 'd8)begin
		baud_cnt <= 'd0;
	end
	else if (rx_flag == 1'b1) begin
		baud_cnt <= baud_cnt + 1'b1;
	end
end

//中间采集数据的时间点标志
always @(posedge sclk or negedge s_rst_n) begin
	if (!s_rst_n) begin
		capture_flag <= 1'b0;
	end
	else if (baud_cnt == BAUD_HALF) begin
		capture_flag <= 1'b1;
	end
	else begin
		capture_flag <= 1'b0;
	end
end

//采集数据的计数
always @(posedge sclk or negedge s_rst_n) begin
	if (!s_rst_n) begin
		capture_cnt <= 'd0;
	end
	else if (capture_cnt == 'd8 && capture_flag == 1'b1) begin
		capture_cnt <= 'd0;
	end
	else if(capture_flag == 1'b1)begin
		capture_cnt <= capture_cnt + 1'b1;
	end
end

//输出po_data
always @(posedge sclk or negedge s_rst_n) begin
	if (!s_rst_n) begin
		po_data <= 'd0;
	end
	else if (capture_cnt >= 1'b1 && capture_flag == 1'b1) begin
		po_data <= {rx2_reg,po_data[7:1]};
	end
end

//输出结束标志
always @(posedge sclk or negedge s_rst_n) begin
	if (!s_rst_n) begin
		po_flag <= 1'b0;
	end
	else if (capture_cnt == 'd8 && capture_flag == 1'b1) begin
		po_flag <= 1'b1;
	end
	else begin
		po_flag <= 1'b0;
	end
end


endmodule

2、接收线TX流程

        2.1、模块图

         2.2、时序波形图

        2.3、 TX发送代码

// -----------------------------------------------------------------------------
// Copyright (c) 2014-2023 All rights reserved
// -----------------------------------------------------------------------------
// Author : BigFartPeach
// CSDN   : 大屁桃
// E-mail : 2624507313@qq.com
// File   : uart_tx.v
// Create : 2023-06-20 14:14:25
// -----------------------------------------------------------------------------
module uart_tx(
	input wire sclk,
	input wire s_rst_n,
	input wire [7:0]pi_data,
	input wire pi_flag,
	output reg tx
	);
// //9600波特率参量
// parameter BAUD_END = 5207;//波特率9600,接收间隔最大计数MAXEND = 50M/9600 = 5208.33 - 1 = 5207


//115200波特率参量
parameter BAUD_END = 433;//波特率115200,接收间隔最大计数MAXEND = 50M/115200 = 434.02 - 1 = 433

//变量
reg [7:0]pi_data_reg;//保存输入的数据
reg tx_flag;		 //发送数据的有效位,服务于计数器 
reg [12:0]baud_cnt;  //9600波特率的1个波特时间1/9600s的计数器
reg bit_flag;		 //计满flag
reg [3:0]bit_cnt;	 //发送数据计数。


//保存输入的数据,因为po_data在接收数据的时候一直在变
always @(posedge sclk or negedge s_rst_n) begin
	if (!s_rst_n) begin
		pi_data_reg <= 'd0;
	end
	else if(pi_flag == 1'b1)begin//这里是寄存的作用,不是直接传过来
		pi_data_reg <= pi_data;
	end
end

//发送数据有效时间段
always @(posedge sclk or negedge s_rst_n) begin
	if (!s_rst_n) begin
		tx_flag <= 1'b0;
	end
	else if(bit_flag == 1'b1 && bit_cnt == 'd8)begin
		tx_flag <= 1'b0;
	end
	else if (pi_flag == 1'b1) begin
		tx_flag <= 1'b1;
	end
end

//一波特时间对应的计数器
always @(posedge sclk or negedge s_rst_n) begin
	if (!s_rst_n) begin
		baud_cnt <= 'd0;
	end
	else if (baud_cnt == BAUD_END) begin
		baud_cnt <= 'd0;
	end
	else if(tx_flag == 1'b1)begin
		baud_cnt <= baud_cnt + 1'b1; 
	end
end

//bit_flag
always @(posedge sclk or negedge s_rst_n) begin
	if (!s_rst_n) begin
		bit_flag <= 1'b0;
	end
	else if (baud_cnt == BAUD_END - 1'b1) begin
		bit_flag <= 1'b1;
	end
	else begin
		bit_flag <= 1'b0;
	end
end

//发送数据计数
always @(posedge sclk or negedge s_rst_n) begin
	if (!s_rst_n) begin
		bit_cnt <= 'd0;
	end
	else if(bit_cnt == 'd8 && bit_flag == 1'b1)begin
		bit_cnt <= 'd0;
	end
	else if (bit_flag == 1'b1) begin
		bit_cnt <= bit_cnt + 1'b1;
	end
end

//发送数据
always @(posedge sclk or negedge s_rst_n) begin
	if (!s_rst_n) begin
		tx <= 1'b1;
	end
	else if(pi_flag == 1'b1)begin
		tx <= 1'b0;//起始位,判断到有输入,立马拉低
	end
	else if (bit_flag == 1'b1 && bit_cnt <= 'd7) begin
		tx <= pi_data_reg[bit_cnt];
	end
	else if(bit_flag == 1'b1 && bit_cnt == 'd8)begin
		tx <= 1'b1;//停止位
	end
end

endmodule

3、Top层代码

// -----------------------------------------------------------------------------
// Copyright (c) 2014-2023 All rights reserved
// -----------------------------------------------------------------------------
// Author : BigFartPeach
// CSDN   : 大屁桃
// E-mail : 2624507313@qq.com
// File   : top.v
// Create : 2023-06-20 14:31:06
// -----------------------------------------------------------------------------
module top(
	input wire clk,
	input wire rst_n,
	input wire rx,
	output wire tx
	);

wire [7:0]data;
wire flag;


uart_rx inst_uart_rx (
		.sclk    (clk),
		.s_rst_n (rst_n),
		.rx      (rx),
		.po_data (data),
		.po_flag (flag)
	);

uart_tx  inst_uart_tx (
		.sclk    (clk),
		.s_rst_n (rst_n),
		.pi_data (data),
		.pi_flag (flag),
		.tx      (tx)
	);


endmodule

第三部分、仿真代码的编写

        注意:在testbench同文件位置建立如下文本

00000001
00000010
00000011
00000100
00000101
00000110
00000111
00001000
00001001
00001010
00001011
00001100
00001101
00001110
00001111
11111111

1、testbench代码

          关于top层的testbench代码如下: 

// -----------------------------------------------------------------------------
// Copyright (c) 2014-2023 All rights reserved
// -----------------------------------------------------------------------------
// Author : BigFartPeach
// CSDN   : 大屁桃
// E-mail : 2624507313@qq.com
// File   : top_sim.v
// Create : 2023-04-25 11:24:00
// -----------------------------------------------------------------------------
module top_sim();

reg clk;
reg rst_n;
reg rx;
wire tx;

reg [7:0]mem[15:0];

initial begin
	clk = 0;
	rst_n = 0;
	rx = 1;
	#100;
	rst_n = 1;
end

always #10 clk = ~clk;

initial begin
	$readmemb("./data.txt",mem);
end

initial begin
	#200;
	read_text();
end

//多次读文本数据
task read_text();
	integer i;
	begin
		for(i = 0; i < 16; i = i + 1)begin
			send_bit_data(mem[i]);
		end
	end
endtask

task send_bit_data(input [7:0]data_txt);
	integer i;
	begin
		for(i = 0; i < 10;i = i + 1)begin
			case (i)
				0: rx = 0;//起始位
				1: rx = data_txt[0];//rx = mem[i - 1];	
				2: rx = data_txt[1];//rx = mem[i - 1];	
				3: rx = data_txt[2];//rx = mem[i - 1];	
				4: rx = data_txt[3];//rx = mem[i - 1];	
				5: rx = data_txt[4];//rx = mem[i - 1];	
				6: rx = data_txt[5];//rx = mem[i - 1];	
				7: rx = data_txt[6];//rx = mem[i - 1];	
				8: rx = data_txt[7];//rx = mem[i - 1];	
				9: rx = 1;//停止位
			endcase
			#104166;//1/9600波特时间
		end
	end
endtask

top inst_top (
	.clk(clk), 
	.rst_n(rst_n),
	.rx(rx),
	.tx(tx)
	);

endmodule

 2、仿真结果

        可以看到rx与tx的波形一致。

3、上板测试

        波特率为9600发送后返回的数据

             波特率为115200发送后返回的数据

 第四部分、总结

        本篇博客介绍了串口的基本知识以及驱动代码的编写,博客中若有错误欢迎大家及时私信我。

        最后,希望我的博客对你有帮助😎😎😎😎,有需要的小伙伴可以查看本专栏更多的往期文章专栏链接如下:FPGA的学习之旅_大屁桃的博客-CSDN博客

        博客中涉及到的工程链接如下,工程基于ISE软件,没有积分的小伙伴评论留下邮箱即可

        FPGA入门第六篇、异步串口通信工程资源-CSDN文库

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

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

相关文章

OWASP之SSRF服务器伪造请求

文章目录 一、SSRF定义二、形成原因1.提供请求功能2.地址没做限制 三、漏洞危害1.可以对服务器所在内网、本地进行端口扫描&#xff0c;获取一些服务的信息等2.目标网站本地敏感数据的读取3.内外网主机应用程序漏洞的利用4.内外网Web站点漏洞的利用 四、ssrf挖掘1.从WEB功能上寻…

SpringBoot相关知识

SpringBoot知识 1 SpringBoot 介绍及其使用原因 Spring Boot是一个用于创建独立的、基于Java的生产级别的应用程序的框架。它旨在简化Spring应用程序的开发过程&#xff0c;减少开发人员的配置工作&#xff0c;从而提高开发效率。 原因: (1) 简化开发&#xff1a;Spring Boot…

Hadoop(HA)

文章目录 1、HA 概述2、HDFS-HA 集群搭建3、HDFS-HA 核心问题4、HDFS-HA 手动模式4.1 环境准备4.2 规划集群4.3 配置 HDFS-HA 集群4.4 启动 HDFS-HA 集群 5、HDFS-HA 自动模式5.1 HDFS-HA 自动故障转移工作机制5.2 HDFS-HA 自动故障转移的集群规划5.3 配置 HDFS-HA 自动故障转移…

java企业工程项目管理系统平台源码

工程项目管理软件&#xff08;工程项目管理系统&#xff09;对建设工程项目管理组织建设、项目策划决策、规划设计、施工建设到竣工交付、总结评估、运维运营&#xff0c;全过程、全方位的对项目进行综合管理 工程项目各模块及其功能点清单 一、系统管理 1、数据字典&#…

基于pyqt5、mysql、yolov7、chatgpt的小麦病害检测系统的设计与实现(基础版)

基于pyqt5、mysql、yolov7、chatgpt的小麦病害检测系统设计与实现 一、界面设计1.1安装pyqt51.2创建用户子窗体1.3创建管理员主窗体1.4创建管理员子窗体1.5创建系统登陆界面 二、环境搭建2.1pyqt5工具配置2.2mysql5.7安装 三、编程实现3.1初始化数据库3.2创建用户数据库sdk文件…

从零构建后端项目-配置Shiro+JWT

目录 Shiro和JWT技术 一、Shiro简介 什么是认证&#xff1f; 什么是授权&#xff1f; Shiro靠什么做认证与授权的&#xff1f; 二、JWT简介 JWT可以用在单点登录的系统中 JWT兼容更多的客户端 创建JwtUtil工具类 一、导入依赖库 二、定义密钥和过期时间 三、创…

强大的Kotlin也能搞定,测试利器MockK你不能不知道

目录 前言&#xff1a; 为什么需要MockK 关键字 Mock Kotlin的类时报错 静态方法如何Mock Jmockit MockK使用示例 普通使用 mockkObject mockkStatic mock private method Context Mock 遇到的一些小坑 最后 前言&#xff1a; MockK是一个强大且易于使用的Kotli…

爆肝整理,手机App接口测试大全指南,看这篇就够了...

目录&#xff1a;导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结&#xff08;尾部小惊喜&#xff09; 前言 App和Web请求服务…

用prim和kruskal算法求最小生成树问题

最短网络 题目http://ybt.ssoier.cn:8088/problem_show.php?pid1350 #include<bits/stdc.h> using namespace std; const int N110; int w[N][N]; bool st[N]; int dist[N]; int n,res0; void prim() {memset(dist,0x3f,sizeof dist);dist[1]0;//初始化第一个点到自己…

优化回声消除过程:舒适噪声生成算法的应用与原理

在回声消除中&#xff0c;舒适噪声生成(Comfort Noise Generator&#xff0c;CNG)算法是一种常用的技术&#xff0c;它被用来减轻回声消除过程中产生的不适感和声音失真。舒适噪声生成算法通过添加特定的噪声信号来模拟人耳的听觉特性&#xff0c;以改善声音的自然度和舒适度。…

Cpp面试题:main函数执行以前,还会执行什么代码?

Cpp面试题:main函数执行以前&#xff0c;还会执行什么代码&#xff1f; 在 main() 函数执行之前和之后&#xff0c;C 程序可能会执行以下代码&#xff1a; 静态对象的构造函数&#xff1a;如果程序中有静态对象&#xff08;全局变量或静态成员变量&#xff09;&#xff0c;它们…

深入了解 vcruntime140 文件,从多方面解析vcruntime140

vcruntime140 是一个重要的文件&#xff0c;它在 Windows 系统中扮演着重要的角色。如果不小心缺失了&#xff0c;那么你的电脑就会出现问题&#xff0c;今天我们就来探讨一下vcruntime140这个文件&#xff0c;分别从背景和预防丢失&#xff0c;到如何修复丢失vcruntime140来给…

Elasticsearch(十)搜索---搜索匹配功能①--查询所有文档和term级别查询

一、前言 之前的学习我们已经了解了搜索的辅助功能&#xff0c;从这一章开始就是ES真正核心的功能&#xff0c;搜索。针对不同的数据类型&#xff0c;ES提供了很多搜索匹配功能&#xff1a;既有进行完全匹配的term搜索&#xff0c;也有按照范围匹配的range搜索&#xff1b;既有…

一文让你学会接口自动化测试框架!

目录 前言&#xff1a; 自动化测试 接口自动化测试的价值 接口自动化测试如何开展 接口自动化测试框架 前言&#xff1a; 接口自动化测试是指利用程序自动化地执行API接口测试&#xff0c;可以提高测试效率和准确性。 自动化测试 自动化测试&#xff0c;这几年行业内的…

一个悄然崛起的AI开源项目!

众所周知&#xff0c;最近这半年AI相关的话题实在是火到出圈。尤其是生成式AI的流行&#xff0c;让我们普通人也可以近距离地接触和应用AI。这其中最典型的就是ChatGPT。 那除了ChatGPT&#xff0c;还有一个非常实用的领域&#xff0c;也是我们今天要讨论的话题&#xff0c;那…

​低代码让传统软件开发土掉渣了

正所谓“让机器去做无聊的事情&#xff0c;让人类去创造美好的事物”。 在当今数码化时代&#xff0c;企业如何更快捷、高效的开发应用是众所周知的难题。传统开发方式需要多名开发人员耗费大量时间精力开发&#xff0c;期间还需要经历漫长的测试和上线过程。 要在这个竞争激烈…

AI绘图软件分享:Midjourney 基础教程(二)

大家好&#xff0c;我是权知星球&#xff0c;今天继续给大家介绍AI绘图软件分享&#xff1a;Midjourney 基础教程&#xff08;二&#xff09; ⼀、Midjourney 服务器介绍 1.Discord 软件介绍 Midjourney AI 绘画服务基于 Discord 软件的&#xff0c;它的绘画功能&#xff0c;…

【AUTOSAR】UDS协议的代码分析与解读(十一)----UDS例程控制31h请求下载 34h

8.15 例程控制 RoutineControl (31h) 此服务用于启动程序 、停止程序和请求程 序执行结果。例程由 两字节的例程标识符 (RoutineIdentifier)来确定。 8.15.1 报文格式 表 71 例程控制服务的请求报文 Byte Name Cvt Value (Hex) #1 RequestServiceIdentifier M 31 #2 …

部署运行jar包方法全解docker镜像打包部署等

基本方法 java -jar 对应的jar包名字 永久后台方法 有一种叫做“nohup”的命令&#xff0c;该命令可以让您的应用程序在后台运行&#xff0c;即使您已经断开了与终端的连接也能保持运行状态。 nohup 命令的语法为&#xff1a; nohup command arg1 arg2 ... argN &其中…

金三银四互联网大厂秋招精选 1160 道 Java 面试题答案整理(2023 最新版)

今年的大环境而言&#xff0c;面试成功的难度比往年高了很多&#xff0c;很明显的感受就是&#xff1a;对于今年的 java 开发朋友面试&#xff0c;无论一面还是二面&#xff0c;都开始考验一个 Java 程序员的技术功底和基础。Java 基础掌握不牢&#xff0c;对于一个开发人员来说…