UART串口通信协议

一、串行通信

串行通信分为两种方式:同步串行通信异步串行通信

同步串行通信需要通信双方在同一时钟的控制下,同步传输数据。

异步串行通信是指通信双方使用各自的时钟控制数据的发送和接收过程。

二、UART

通用异步收发传输器(Universal Asynchronous Receiver/Transmitter,UART)是一种全双工、异步串行通信方式它在发送数据时将并行数据转换成串行数据来传输,在接收数据时将接收到的串行数
据转换成并行数据,可以实现全双工传输和接收。

UART串口通信需要两根信号线来实现,一根用于串口发送,另外一根负责串口接收

UART在发送或接收过程中的一帧数据由4部分组成,起始位(低电平)、数据位、奇偶校验位和停止位(高电平)

起始位标志着一帧数据的开始,停止位标志着一帧数据的结束,数据位是一帧数据中的有效数据。

校验位分为奇校验和偶校验,用于检验数据在传输过程中是否出错。奇校验时,发送方应使数据位中1的个数与校验位中1的个数之和为奇数;接收方在接收数据时,对1的个数进行检查,若不为奇数,则说明数据在传输过程中出了差错。同样,偶校验则检查1的个数是否为偶数。

UART通信过程中的数据格式及传输速率是可设置的,数据位可选择为 5、6、7、8位(最常用);校验位可选择奇校验、偶校验或者无校验位;停止位可选择1位(默认),1.5或2位。

串口通信的速率用波特率表示,它表示每秒传输二进制数据的位数,单位是bps(位/秒),常用的波特率有9600、19200、38400、57600以及115200等。

FPGA开发串口中,为了得到串口传输一位数据所需要的系统时钟周期数,需要对系统时钟进行计数,计数方法为FPGA时钟频率/波特率。如时钟频率为50000000hz(50MHz),需要的比特率为9600bps,则需要计数50000000/9600≈5208次

三、串口实现

上位机通过串口调试助手发送数据给FPGA,FPGA 通过USB串口接收数据并将接收到的数据发送给上位机,完成串口数据环回。

首先要有一个接收模块,接收串口调试助手发送的数据

module uart_rx(

//**************输入**************
	 input			     clk,                 					   //系统50MHz时钟
    input              rst_n,               					   //系统复位,低电平有效  
    input              uart_rxd,            					   //UART接收端口
	 
//**************输出**************
    output  reg        uart_done,            				       //接收一帧数据完成标志
    output  reg        rx_flag,                                    //接收标志位
    output  reg [3:0]  rx_cnt,                                     //接收数据计数器
    output  reg [7:0]  rxdata,                                     //接收端口数据寄存
    output  reg [7:0]  uart_data                                   //接收的数据
    );
    
//************参数定义************
parameter   CLK_FREQ = 50000000;                				//系统时钟
parameter   UART_BPS = 9600;                    				//串口波特率
localparam  BPS_CNT  = CLK_FREQ/UART_BPS;      					//波特率计数,串口传输一位所需要的系统时钟周期数
																				
//************信号定义************
reg        uart_rxd_1;                                      //异步信号会带来亚稳态,常用处理方式是打拍处理,第一拍
reg        uart_rxd_2;                                      //第二拍,通常打两拍就基本上就能避免亚稳态问题
reg [15:0] clk_cnt;                              			//系统时钟计数器

wire       start_flag;                                      //起始标志位


//检测接收端口下降沿来捕获起始位,输出一个时钟周期的脉冲start fag,并进入串口接收过程
assign  start_flag = uart_rxd_2 & (~uart_rxd_1);    

//对UART接收端口的数据延迟两个时钟周期避免亚稳态
always @(posedge clk or negedge rst_n) begin 
    if (!rst_n) begin 
        uart_rxd_1 <= 1'b0;
        uart_rxd_2 <= 1'b0;          
    end
    else begin
        uart_rxd_1  <= uart_rxd;                   
        uart_rxd_2  <= uart_rxd_1;
    end   
end

//当脉冲信号start_flag有效,进入数据接收过程           
always @(posedge clk or negedge rst_n) begin         
    if (!rst_n)
        rx_flag <= 1'b0;
    else begin
	    //检测到起始位,进入数据接收过程,标志位rx_flag拉高
        if(start_flag)
            rx_flag <= 1'b1;
        //当接收数据计数器计数到9(9个波特周期)且在停止位中间(数据寄存已经完成,为检测下一帧数据起始位准备),停止接收,标志位rx_flag拉低				
        else if((rx_cnt == 4'd9) && (clk_cnt == BPS_CNT/2))     	   
            rx_flag <= 1'b0;
        else
            rx_flag <= rx_flag;
    end
end

//进入数据接收过程启动系统时钟计数器
always @(posedge clk or negedge rst_n) begin         
    if (!rst_n)                             
        clk_cnt <= 16'd0;
	//标志位rx_flag有效,处于接收过程	  
    else if (rx_flag) begin
        //计数没到一个波特率周期则一直计数,到了则清零	 
        if (clk_cnt < BPS_CNT - 1)
            clk_cnt <= clk_cnt + 1'b1;
        else
            clk_cnt <= 16'd0;
    end
	 //数据接收过程结束,计数器清零
    else                              				
        clk_cnt <= 16'd0;						
end

//进入数据接收过程启动接收数据计数器
always @(posedge clk or negedge rst_n) begin         
    if (!rst_n)                             
        rx_cnt  <= 4'd0;  
    else if (rx_flag) begin
	    //系统时钟计数到一个波特率周期,接收数据计数器加1,没到则不加
        if (clk_cnt == BPS_CNT - 1)				
            rx_cnt <= rx_cnt + 1'b1;			 //可以用来判断当前传输的是第几位
        else
            rx_cnt <= rx_cnt;       
    end
	 //接收过程结束,计数器清零
	 else
        rx_cnt  <= 4'd0;						
end

//根据接收数据计数器来寄存uart接收端口数据
always @(posedge clk or negedge rst_n) begin 
    if (!rst_n)  
        rxdata <= 8'd0;                                     
    else if(rx_flag)
	    //计数到数据中间的采样结果最稳定
        if (clk_cnt == BPS_CNT/2) begin
            case (rx_cnt)
			 //根据rx_cnt的值将uart接收端口的数据寄存到接收数据寄存器对应的位实现串并转换
             4'd1 : rxdata[0] <= uart_rxd_2;   //寄存数据位最低位
             4'd2 : rxdata[1] <= uart_rxd_2;
             4'd3 : rxdata[2] <= uart_rxd_2;
             4'd4 : rxdata[3] <= uart_rxd_2;
             4'd5 : rxdata[4] <= uart_rxd_2;
             4'd6 : rxdata[5] <= uart_rxd_2;
             4'd7 : rxdata[6] <= uart_rxd_2;
             4'd8 : rxdata[7] <= uart_rxd_2;   //寄存数据位最高位
             default:;                                    
            endcase
        end
        else 
            rxdata <= rxdata;
    else
        rxdata <= 8'd0;
end

//数据接收完毕后给出标志信号并寄存输出接收到的数据
always @(posedge clk or negedge rst_n) begin        
    if (!rst_n) begin
        uart_data <= 8'd0;                               
        uart_done <= 1'b0;
    end
	//接收数据计数器计数到停止位时,寄存输出接收到的数据并将接收完成标志位拉高
    else if(rx_cnt == 4'd9) begin                          
        uart_data <= rxdata;              
        uart_done <= 1'b1;    
    end
    else begin
        uart_data <= 8'd0;                                   
        uart_done <= 1'b0; 
    end    
end

endmodule	

 然后还需要一个发送模块,将数据发送给PC端,原理同接收模块大部分一致

module uart_tx(
    
//**************输入**************	 
	 input	           clk,                 //系统50MHz时钟
    input              rst_n,               //系统复位,低电平有效  
    input              uart_en,             //发送使能信号
	 input       [ 7:0] uart_din,            //待发送数据
	 
//**************输出**************

    output             uart_tx_busy,        //发送忙状态标志 
    output             en_flag,             //使能标志位
    output  reg        tx_flag,             //发送过程标志信号
    output  reg [ 7:0] tx_data,             //寄存发送数据
    output  reg [ 3:0] tx_cnt,              //发送数据计数器
    output  reg        uart_txd             //UART发送端口
    );
    
//************参数定义************
parameter   CLK_FREQ = 50000000;            //系统时钟频率
parameter   UART_BPS = 9600;                //串口波特率
localparam  BPS_CNT  = CLK_FREQ/UART_BPS;   //波特率计数,串口传输一位所需要的系统时钟周期数

//************信号定义************
reg        uart_en_1; 
reg        uart_en_2;  
reg [15:0] clk_cnt;                           //系统时钟计数器

//在串口发送过程中给出忙状态标志,其他模块就可以判断串口发送模块是否处于空闲状态
assign uart_tx_busy = tx_flag;

//捕获uart_en上升沿,得到一个时钟周期的脉冲信号,进入数据发送过程
assign en_flag = (~uart_en_2) & uart_en_1;

//对发送使能信号uart_en延迟两个时钟周期避免亚稳态
always @(posedge clk or negedge rst_n) begin         
    if (!rst_n) begin
        uart_en_1 <= 1'b0;                                  
        uart_en_2 <= 1'b0;
    end                                                      
    else begin                                               
        uart_en_1 <= uart_en;                               
        uart_en_2 <= uart_en_1;                            
    end
end

//当脉冲信号en_flag到达时,寄存待发送的数据,进入发送过程          
always @(posedge clk or negedge rst_n) begin         
    if (!rst_n) begin                                  
        tx_flag <= 1'b0;
        tx_data <= 8'd0;
    end 
	 //检测到发送使能上升沿,进入发送过程,标志位tx_flag拉高,寄存待发送数据
    else if (en_flag) begin                                       
            tx_flag <= 1'b1;
            tx_data <= uart_din;
        end
        //计数到停止位结束时,停止发送过程,标志位tx_flag拉低                                    
        else if ((tx_cnt == 4'd9) && (clk_cnt == BPS_CNT - (BPS_CNT/16))) begin  //提前1/16个停止位拉低,保证发送数据时间略小于接收数据时间,避免数据积累造成丢失
            tx_flag <= 1'b0;                                 
            tx_data <= 8'd0;
        end
        else begin
            tx_flag <= tx_flag;
            tx_data <= tx_data;
        end 
end

//进入发送过程后,启动系统时钟计数器
always @(posedge clk or negedge rst_n) begin         
    if (!rst_n)                             
        clk_cnt <= 16'd0;                                  
    else if (tx_flag) begin 
        if (clk_cnt < BPS_CNT - 1)
            clk_cnt <= clk_cnt + 1'b1;
        else
            clk_cnt <= 16'd0; 
    end
    else                             
        clk_cnt <= 16'd0;
end

//进入发送过程后,启动发送数据计数器
always @(posedge clk or negedge rst_n) begin         
    if (!rst_n)                             
        tx_cnt <= 4'd0;
    else if (tx_flag) begin 
        if (clk_cnt == BPS_CNT - 1)	
            tx_cnt <= tx_cnt + 1'b1;
        else
            tx_cnt <= tx_cnt;       
    end
    else                              
        tx_cnt  <= 4'd0;
end

//根据发送数据计数器来给uart发送端口赋值
always @(posedge clk or negedge rst_n) begin        
    if (!rst_n)  
        uart_txd <= 1'b1;        
    else if (tx_flag)
        case(tx_cnt)
            4'd0: uart_txd <= 1'b0;         //起始位 
            4'd1: uart_txd <= tx_data[0];   //数据位最低位
            4'd2: uart_txd <= tx_data[1];
            4'd3: uart_txd <= tx_data[2];
            4'd4: uart_txd <= tx_data[3];
            4'd5: uart_txd <= tx_data[4];
            4'd6: uart_txd <= tx_data[5];
            4'd7: uart_txd <= tx_data[6];
            4'd8: uart_txd <= tx_data[7];   //数据位最高位
            4'd9: uart_txd <= 1'b1;         //停止位
            default: ;
        endcase
    else 
        uart_txd <= 1'b1;                   //空闲时发送端口为高电平
end

endmodule	          

环回模块将串口接收模块接收到的数据发送给串口发送模块

module uart_loop(
    
//**************输入**************
    input	         clk,                       //系统时钟
    input            rst_n,                     //系统复位,低电平有效
    input            recv_done,                 //接收一帧数据完成标志
    input      [7:0] recv_data,                 //接收的数据
    input            tx_busy,                   //发送忙状态标志     
	
//**************输出**************	
    output reg       send_en,                   //发送使能信号
    output reg [7:0] send_data                  //待发送数据
    );

	 
//************信号定义************
reg recv_done_d0;
reg recv_done_d1;
reg tx_ready;

//wire define
wire recv_done_flag;


//检测到recv_done上升沿,得到一个时钟周期的脉冲信号,标志着串口接收模块接收到了一帧数据
assign recv_done_flag = (~recv_done_d1) & recv_done_d0;
                                                 
//对发送使能信号recv_done延迟两个时钟周期
always @(posedge sys_clk or negedge sys_rst_n) begin         
    if (!sys_rst_n) begin
        recv_done_d0 <= 1'b0;                                  
        recv_done_d1 <= 1'b0;
    end                                                      
    else begin                                               
        recv_done_d0 <= recv_done;                               
        recv_done_d1 <= recv_done_d0;                            
    end
end

//判断接收完成信号,并在串口发送模块空闲时给出发送使能信号
always @(posedge sys_clk or negedge sys_rst_n) begin         
    if (!sys_rst_n) begin
        tx_ready  <= 1'b0; 
        send_en   <= 1'b0;
        send_data <= 8'd0;
    end                                                      
    else begin                
        if(recv_done_flag)begin
            tx_ready  <= 1'b1                   //将tx_ready信号拉高,表示已经准备好了待发送的数据
            send_en   <= 1'b0;                  //将send_en信号拉低,为接下来产生一个上升沿作准备
            send_data <= recv_data;             //寄存接收到的数据recv_data到send_data中
        end
        else if(tx_ready && (~tx_busy)) begin   //检测串口发送模块处于空闲状态
            tx_ready <= 1'b0;                   //准备过程结束,等待下一个串口接收数据的到来
            send_en  <= 1'b1;                   //拉高发送使能信号以启动串口发送模块的发送过程将寄存到send_data中的数据发送出去
        end
    end
end

endmodule 

顶层文件

module uart(

//**************输入**************
    input           clk,                //外部50M时钟
    input           rst_n,              //外部复位信号,低有效
    input           uart_rxd,           //UART接收端口
	 
//**************输出**************
    output          uart_txd            //UART发送端口
    );

//************参数定义************
parameter  CLK_FREQ = 50000000;         //定义系统时钟频率
parameter  UART_BPS = 115200;           //定义串口波特率
    
//************信号定义************  
wire       uart_recv_done;              //UART接收完成
wire [7:0] uart_recv_data;              //UART接收数据
wire       uart_send_en;                //UART发送使能
wire [7:0] uart_send_data;              //UART发送数据
wire       uart_tx_busy;                //UART发送忙状态标志


//串口接收模块     
uart_rx #(                          
    .CLK_FREQ       (CLK_FREQ),         //设置系统时钟频率
    .UART_BPS       (UART_BPS))         //设置串口接收波特率
u_uart_recv(                 
    .clk            (clk), 
    .rst_n          (rst_n),
    
    .uart_rxd       (uart_rxd),
    .uart_done      (uart_recv_done),
    .uart_data      (uart_recv_data)
    );

//串口发送模块    
uart_tx #(                          
    .CLK_FREQ       (CLK_FREQ),         //设置系统时钟频率
    .UART_BPS       (UART_BPS))         //设置串口发送波特率
u_uart_send(                 
    .clk            (clk),
    .rst_n          (rst_n),
     
    .uart_en        (uart_send_en),
    .uart_din       (uart_send_data),
    .uart_tx_busy   (uart_tx_busy),
    .uart_txd       (uart_txd)
    );
    
//串口环回模块    
uart_loop u_uart_loop(
    .clk            (clk),             
    .rst_n          (rst_n),           
   
    .recv_done      (uart_recv_done),   //接收一帧数据完成标志信号
    .recv_data      (uart_recv_data),   //接收的数据
   
    .tx_busy        (uart_tx_busy),     //发送忙状态标志      
    .send_en        (uart_send_en),     //发送使能信号
    .send_data      (uart_send_data)    //待发送数据
    );
    
endmodule

 参考资料:

正点原子FPGA教程、小梅哥FPGA教程

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

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

相关文章

基于SpringBoot + EasyExcel + Vue + Blob实现导出Excel文件的前后端完整过程

首先前端发起HTTP请求之后&#xff0c;后端返回一个Excel输出流&#xff0c;然后前端用Blob类型接收数据&#xff0c;并且解析响应头数据以及提取源文件名&#xff0c;最后用a标签完成下载。 一、后端代码 &#xff08;1&#xff09;导入阿里巴巴的EasyExcel依赖&#xff08;…

云计算的学习(五)

五、虚拟化特性介绍 1.集群特性 1.1HA HA&#xff08;Hith Available&#xff0c;高可用特性)&#xff0c;克服单台主机的局限性&#xff0c;当一台服务器损坏&#xff0c;运行在损坏服务器上的虚拟机会自动迁移到其他运行状态正常的服务器上&#xff0c;整个迁移过程用户无感…

基于ssm的社区生活超市的设计与实现

博主介绍&#xff1a;专注于Java技术领域和毕业项目实战。专注于计算机毕设开发、定制、文档编写指导等&#xff0c;对软件开发具有浓厚的兴趣&#xff0c;工作之余喜欢钻研技术&#xff0c;关注IT技术的发展趋势&#xff0c;感谢大家的关注与支持。 技术交流和部署相关看文章…

设计模式-外观模式在Java中的使用示例

场景 外观模式 外观模式是一种使用频率非常高的结构型设计模式&#xff0c;它通过引入一个外观角色来简化客户端与子系统 之间的交互&#xff0c;为复杂的子系统调用提供一个统一的入口&#xff0c;降低子系统与客户端的耦合度&#xff0c;且客户端调用非常方便。 示例 自…

让小程序动起来-轮播图的两种方式--【浅入深出系列003】

浅入深出系列总目录在000集 如何0元学微信小程序–【浅入深出系列000】 文章目录 本系列校训学习资源的选择啥是轮播图轮播图的关键代码最常见的轮播图代码便于理解的轮播代码两种轮播代码的比较 实际操练第一步&#xff0c;就是找到文件。第二步&#xff0c;先改动一下最显眼…

HTTP1.1、HTTPS、HTTP2.0 、HTTP3.0

HTTP1.1 优点&#xff1a; 整体方面&#xff1a;简单、灵活和易于扩展、应用广泛和跨平台 性能方面&#xff1a;长连接、管道网络传输解决请求队头阻塞&#xff08;没有使用&#xff09; 缺点&#xff1a; 安全方面&#xff1a;无状态、明文窃听、伪装、篡改 性能方面&am…

基于jeecg-boot的nbcio-boot亿事达企业管理平台发布

目前这个演示系统与代码都同步&#xff0c;以后也尽量保持同步。 更多功能看演示系统 gitee源代码地址 后端代码&#xff1a; https://gitee.com/nbacheng/nbcio-boot 前端代码&#xff1a;https://gitee.com/nbacheng/nbcio-vue.git 在线演示&#xff08;包括H5&#xff…

Databend 开源周报第 102 期

Databend 是一款现代云数仓。专为弹性和高效设计&#xff0c;为您的大规模分析需求保驾护航。自由且开源。即刻体验云服务&#xff1a;https://app.databend.cn 。 Whats On In Databend 探索 Databend 本周新进展&#xff0c;遇到更贴近你心意的 Databend 。 为指定列创建 B…

MongoDB

MongoDB概述 MongoDB是一个基于分布式文件存储的数据库。由C语言编写。旨在为WEB应用提供可扩展的高性能数据存储解决方案。 MongoDB是一个介于关系数据库和非关系数据库之间的产品&#xff0c;是非关系数据库当中功能最丰富&#xff0c;最像关系数据库的。 它支持的数据结构非…

VMware安装Ubuntu(VMware版本17-Ubuntu版本16.0)

VMware安装Ubuntu&#xff08;VMware版本17-Ubuntu版本16.0&#xff09; 一&#xff0c;VMware虚拟机下载官网点击https://customerconnect.vmware.com/cn/downloads/info/slug/desktop_end_user_computing/vmware_workstation_pro/17_0 二&#xff0c;Ubuntu乌班图下载官网点…

Bootstrap编写一个兼容主流浏览器的受众巨幕式风格页面

Bootstrap编写一个兼容主流浏览器的受众巨幕式风格页面 虽然说IE6除了部分要求苛刻的需求以外已经被可以不考虑了&#xff0c;但是WIN7自带的浏览器IE8还是需要支持的。 本文这个方法主要的优点&#xff0c;个人觉得就是准备少&#xff0c;不需要上网寻找大量的图片做素材&…

Redis高级篇(一)

分布式缓存 -- 基于Redis集群解决单机Redis存在的问题 单机的Redis存在四大问题&#xff1a; 1.Redis持久化 Redis有两种持久化方案&#xff1a;RDB持久化、AOF持久化 1.1.RDB持久化 什么是RDB持久化 RDB全称Redis Database Backup file&#xff08;Redis数据备份文件&am…

Ubuntu18.04 拯救者R9-7945HX 4060 配置ZED 2i代双目相机驱动+ORBSLAM2

AMD的拯救者网卡很拉&#xff0c;研究了很久除了换网卡可以解决网络问题&#xff0c;其它没找到合适的办法&#xff0c;这里我用手机USB共享网络的方式勉强上网&#xff0c;这里不得不说华为的信号桥很好用。 之前在1050ti的电脑上布置过&#xff0c;很顺利&#xff0c;这个新…

算法竞赛字符串常用操作大全

算法竞赛字符串常用操作总结来啦~ &#x1f44a; 大家好 我是寸铁&#x1f4aa; 考前需要刷大量真题,大家一起相互监督&#xff0c;每日做N题&#xff0c;一起上岸吧✌️ ~ 冲刺蓝桥杯省一模板大全来啦 &#x1f4a5; ~ 蓝桥杯4月8号就要开始了 &#x1f64f; ~ 还没背熟模…

利用 jenkins 关联 Job 方式完善 RobotFramework 测试 Setup 以及 Teardown 后操作

目录 1.前言 2.Jekins 关联 Job 方式 1.前言 Jenkins是一个流行的持续集成和交付工具&#xff0c;它可以帮助自动化构建、测试和部署软件。与Robot Framework结合使用&#xff0c;可以实现更高效的测试工作流程。 在Robot Framework中&#xff0c;Setup和Teardown是测试用例…

Kafka 深度剖析

1、应用场景 1.1 kafka场景 Kafka最初是由LinkedIn公司采用Scala语言开发&#xff0c;基于ZooKeeper&#xff0c;现在已经捐献给了Apache基金会。目前Kafka已经定位为一个分布式流式处理平台&#xff0c;它以 高吞吐、可持久化、可水平扩展、支持流处理等多种特性而被广泛应用…

JavaSwing+MySQL的飞机订票系统(内含oracle版本)

点击以下链接获取源码&#xff1a; https://download.csdn.net/download/qq_64505944/88055544 JDK1.8 MySQL5.7 功能&#xff1a;接收客户端发来的数据、处理客户端发来的数据、发送数据包到客户端&#xff1b;客户端&#xff1a;查询所有航班的信息、查看自己所定的票、订票…

java学习路程之篇二、知识点、配置JAVA_HOME、跨平台、JVM、JRE、JDK

文章目录 1、Java背景介绍2、Java跨平台性3、JDK的下载和安装4、第一个Java程序5、HelloWorld案例详解6、JVM、JRE和JDK7、配置JAVA_HOME 1、Java背景介绍 2、Java跨平台性 3、JDK的下载和安装 4、第一个Java程序 5、HelloWorld案例详解 6、JVM、JRE和JDK 7、配置JAVA_HOME

Bash 第十行

195 第十行 给定一个文本文件 file.txt&#xff0c;请只打印这个文件中的第十行。 示例: 假设 file.txt 有如下内容&#xff1a; Line 1 Line 2 Line 3 Line 4 Line 5 Line 6 Line 7 Line 8 Line 9 Line 10 你的脚本应当显示第十行&#xff1a; Line 10 来源&#xff1a;…

详解使用JAVA将Julian date(儒略日)日期转换为年月日

一、什么是Julian date 朱莉安日历和普通日历显示是不一样的我就举例演示一下 正常的日历显示 朱莉安的日历显示 174表示的是从2016年1月1日开始到今天已有174天了 普通日历是按月计数&#xff0c;朱莉安日历是按年计数 二、用java将julian日期转换为年月日 将2023199朱莉安…