【7. ROS 中的 IMU 惯性测量单元消息包】

news2024/10/6 5:58:53

欢迎大家阅读2345VOR的博客【6. 激光雷达接入ROS】🥳🥳🥳
2345VOR鹏鹏主页: 已获得CSDN《嵌入式领域优质创作者》称号👻👻👻,座右铭:脚踏实地,仰望星空🛹🛹🛹
🎏🎏主要开发专栏🎏🎏
《arduino学习》:学习最简单开源便利的单片机Arduino,与时俱进😆😆😆
《Arduino编程参考》:本专栏围绕Arduino语法和Arduino库使用开发;🌻🌻🌻
《 Arduino小项目开发》:本专栏围绕Arduino生态结合实际需求设计综合的小项目开发。🌼🌼🌼
《HomeAssistant》:介绍homeassistant中基本开发, 重点设计esphome和nodered开发,包含小爱同学打印机等诸多设备添加。🎉🎉🎉
本文章属于《Ubuntu学习》和《ROS机器人学习》
:围绕Ubuntu系统基本配置及相关命令行学习记录!机器人操作系统 (ROS) 是一组软件库和工具,可帮助您构建机器人应用程序。👍👍👍

1. 前言

Ubuntu环境搭建
【经典Ubuntu20.04版本U盘安装双系统教程】
【Windows10安装或重装ubuntu18.04双系统教程】
【Ubuntu同步系统时间】
【Ubuntu中截图工具】
【Ubuntu安装QQ】
【Ubuntu安装后基本配置】
【Ubuntu启动菜单的默认项】
【ubuntu系统中修改hosts配置】
【18.04Ubuntu中解决无法识别显示屏】
【ROS 开发神器 Visual Studio Code 的安装和设置】
ROS学习笔记
【1. Ubuntu18.04安装ROS Melodic】
【2. 在Github上寻找安装ROS软件包】
【3. 初学ROS,年轻人的第一个Node节点】
【4. ROS的主要通讯方式:Topic话题与Message消息】
【5. ROS机器人的运动控制】
【6. 激光雷达接入ROS】
【7. ROS 中的 IMU 惯性测量单元消息包】
接下来学习ROS 中的 IMU 惯性测量单元消息包和导航,IMU 惯性测量单元是用来测量机器人的空间姿态!
在这里插入图片描述

2. IMU 惯性测量单元

2.1 sensor_msgs

进入ROS Index官网搜索sensor_msgs
在这里插入图片描述

在这里插入图片描述
进入website
在这里插入图片描述

2.2 IMU 惯性测量单元的格式定义

在消息中找到Imu
在这里插入图片描述

这就打开了IMU 惯性测量单元的格式定义

在这里插入图片描述

在这里插入图片描述
在这里插入图片描述

3. 使用C++实现IMU获取数据节点

3.1 通用IMU姿态数据格式

在这里插入图片描述
直接获取IMU融合好的机器人姿态四元数

3.2 构思功能的思路和步骤

构思
在这里插入图片描述

实现步骤

  1. 构建一个新的软件包,包名叫做imu_pkg。
  2. 在软件包中新建一个节点,节点名叫做imu_node。
  3. 在节点中,向ROS大管家NodeHandle申请订阅话题/imu/data,并设置回调函数为IMUCallback()。
  4. 构建回调函数IMUCallback(),用来接收和处理IMU数据。
  5. 使用TF工具将四元数转换成欧拉角。
  6. 调用ROS_INFO()显示转换后的欧拉角数值。

3.3 创建imu_pkg包

在工作空间src文件创建基于sensor_msgs模板的imu_pkg

cd ~/catkin_ws/src/
catkin_create_pkg imu_pkg roscpp rospy sensor_msgs

在这里插入图片描述

在imu_pkg文件夹下src中创建imu_node.cpp
在这里插入图片描述

3.4 编写imu_node订阅者节点

imu_node源码

#include <ros/ ros.h>
#include <sensor msgs/Imu.h>
#include "tf/tf.h"

void IMUCallback(sensor msgs::Imu msg)
{
	if(msg.orientation_covariance[0] <0)
		return;
	tf::Quaternion quaternion(
		msg.orientation.x,
		msg.orientation.y,
		msg.orientation.z,
		msg.orientation.w
	);
	double roll, pitch, yaw;
	tf::Matrix3x3(quaternion).getRPY( roll, pitch, yaw);
	roll = roll*180/M_PI;
	pitch = pitch*180/M_PI;
	yaw = yaw*180/M_PI;
	ROS_INFO(“滚转= %.0f 俯仰=%.0f 朝向= %.0f" , roll, pitch, yaw);
}
int main(int argc,char *argv[])
{
	setlocale(LC_ALL,"");
	ros::init(argc, argv,"imu_node" );
	ros::NodeHandle n;
	ros::Subscriber imu_sub = n.subscribe( "/imu/data" ,10,IMUCallback);
	ros::spin( );
	return 0;
}

ctrl+s快捷保存

3.5 设置C++编译规则

打开CMake文件


add_executable(imu_node src/imu_node.cpp)
add dependencies(imu_node ${${PROJECT_NAVE}_EXPORTED_TARGETS} ${catkin_EXPORTED_TARGETS})
target_link_libraries(imu_node
	${catkin_LIBRARIES}
)


ctrl+s快捷保存
在这里插入图片描述

ctrl+shift+b快捷编译

3.6 编译运行imu_node节点

编译,打开终端

cd ~/catkin_ws/
catkin_make

采用wpr_simulation开源工程,打开三个终端分别运行三条指令

roscore
roslaunch wpr_simulation wpb_simple.launch
rosrun imu_pkg imu_node


在这里插入图片描述
拖动机器人绕z轴正方向旋转90度
在这里插入图片描述
在这里插入图片描述

可参照可以打开wpr_simulation里src文件夹下的demo_imu_data.cpp文件,对照一下代码,排查错误

在这里插入图片描述

4. 用python获取IMU 数据节点

4.1 通用IMU姿态数据格式

在这里插入图片描述
直接获取IMU融合好的机器人姿态四元数

4.2 构思功能的思路和步骤

构思
在这里插入图片描述
实现步骤

  1. 构建一个新的软件包,包名叫做imu_pkg。
  2. 在软件包中新建一个节点,节点名叫做imu_node.py。
  3. 在节点中,向ROS大管家rospy申请订阅话题/imu/data,并设置回调函数为imu_callback()。
  4. 构建回调函数imu_callback(),用来接收和处理IMU数据。
  5. 使用TF工具将四元数转换成欧拉角。
  6. 调用loginfo()显示转换后的欧拉角数值。

4.3 创建imu_pkg包

在工作空间src文件创建基于sensor_msgs模板的imu_pkg,编译

cd ~/catkin_ws/src/
catkin_create_pkg imu_pkg roscpp rospy sensor_msgs
cd ..
catkin_make

在这里插入图片描述

在imu_pkg文件夹下新建script文件夹中创建imu_node.py
在这里插入图片描述

4.4 编写imu_node订阅者节点

先引入python包,设置中文utf-8显示

  • ros>=20.04,采用python3
  • ros<20.04,采用python

lidar_node.py源码

#!/usr/bin/env python3
#coding=utf-8
import rospy
from sensor msgs.msg import Imu
from tf.transformations import euler_from_quaternion
import math

def imu callback(msg):
	if msg.orientation covariance[0]< 0:
		return
	quaternion =[
		msg .orientation.x,
		msg.orientation.y,
		msg.orientation.z,
		msg.orientation.w
	]
	(roll,pitch , yaw) = euler_from_quaternion(quaternion)
	roll = roll*180/math.pi
	pitch = pitch*180/math.pi
	yaw = yaw*180/math.pi
	rospy.loginfo(滚转=%.0f俯仰= %.0f朝向= %.of" ,roll,pitch,yaw)

if _name ="_main_":
	rospy.init_node( "imu _node")
	imu_sub = rospy.subscriber( "/imu/data",Imu,imu_callback,queue_size=10)
	rospy.spin()

ctrl+s快捷保存

4.5 添加可执行的权限

在所在文件夹打开终端

cd catkin_ws/src/imu_pkg/scripts/
ls
chmod +x imu_node.py
ls

文件名变成绿色表示权限添加成功

在这里插入图片描述

4.6 运行imu_node节点

采用wpr_simulation开源工程,打开三个终端分别运行三条指令

roscore
roslaunch wpr_simulation wpb_simple.launch
rosrun vel_pkg vel_node.py


在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

可参照可以打开wpr_simulation里的script文件夹中创建imu_node.py

在这里插入图片描述

5. 用C++编写 IMU 航向锁定节点

在这里插入图片描述

基于前面学习的机器人运动控制和 IMU 惯性测量单元数据,下面将联系这两点编写 IMU 航向锁定节点,我们可以直接在前面实验的程序上做修改

5.1 构思功能的思路和步骤

  1. 让大管家NodeHandle 发布速度控制话题/cmd_vel。
  2. 设定一个目标朝向角,当姿态信息中的朝向角和目标朝向角不一致时,控制机器人转向目标朝向角。

5.2 修改imu_node.cpp

在这里插入图片描述

见3.4源码
修改成如下lidar_node源码

#include <ros/ ros.h>
#include <sensor msgs/Imu.h>
#include "tf/tf.h"
#include "geometry msgs/Twist.h"

ros::Publisher vel_pub;


void IMUCallback(sensor msgs::Imu msg)
{
	if(msg.orientation_covariance[0] <0)
		return;
	tf::Quaternion quaternion(
		msg.orientation.x,
		msg.orientation.y,
		msg.orientation.z,
		msg.orientation.w
	);
	double roll, pitch, yaw;
	tf::Matrix3x3(quaternion).getRPY( roll, pitch, yaw);
	roll = roll*180/M_PI;
	pitch = pitch*180/M_PI;
	yaw = yaw*180/M_PI;
	ROS_INFO(“滚转= %.0f 俯仰=%.0f 朝向= %.0f" , roll, pitch, yaw);
	
	double target_ yaw = 90;
	double diff_angle = target_yaw - yaw;
	geometry_msgS::Twist vel_cmd ;
	vel_cmd.angular.z = diff_angle * 0.0l;
	vel_pub.publish(vel_cmd):

}
int main(int argc,char *argv[])
{
	setlocale(LC_ALL,"");
	ros::init(argc, argv,"imu_node" );
	ros::NodeHandle n;
	ros::Subscriber imu_sub = n.subscribe( "/imu/data" ,10,IMUCallback);
	vel_pub = n.advertise<geometry_msgs::Twist>("/cmd_vel",10);
	ros::spin( );
	return 0;
}

ctrl+s快捷保存

ctrl+shift+b快捷编译
在这里插入图片描述

可参照开源项目wpr_simulation下的src文件夹的demo_imu_behavior.cpp

在这里插入图片描述

5.4 运行imu_node节点

采用wpr_simulation开源工程,打开三个终端分别运行三条指令

roscore
roslaunch wpr_simulation wpb_simple.launch
rosrun imu_pkg imu_node


在这里插入图片描述
拖动机器人绕Z轴旋转,也会自动转回去

在这里插入图片描述

5.4 优化航向策略

当机器人检测前方障碍物时,最简单把转弯角度调大一点,原地转弯
imu_node源码

#include <ros/ ros.h>
#include <sensor msgs/Imu.h>
#include "tf/tf.h"
#include "geometry msgs/Twist.h"

ros::Publisher vel_pub;


void IMUCallback(sensor msgs::Imu msg)
{
	if(msg.orientation_covariance[0] <0)
		return;
	tf::Quaternion quaternion(
		msg.orientation.x,
		msg.orientation.y,
		msg.orientation.z,
		msg.orientation.w
	);
	double roll, pitch, yaw;
	tf::Matrix3x3(quaternion).getRPY( roll, pitch, yaw);
	roll = roll*180/M_PI;
	pitch = pitch*180/M_PI;
	yaw = yaw*180/M_PI;
	ROS_INFO(“滚转= %.0f 俯仰=%.0f 朝向= %.0f" , roll, pitch, yaw);
	
	double target_ yaw = 90;
	double diff_angle = target_yaw - yaw;
	geometr_msgS::Twist vel_cmd ;
	vel_cmd.angular.z = diff_angle * 0.0l;
	vel_cmd.linear.x=0.1;
	vel_pub.publish(vel_cmd):

}
int main(int argc,char *argv[])
{
	setlocale(LC_ALL,"");
	ros::init(argc, argv,"imu_node" );
	ros::NodeHandle n;
	ros::Subscriber imu_sub = n.subscribe( "/imu/data" ,10,IMUCallback);
	vel_pub = n.advertise<geometry_msgs::Twist>("/cmd_vel",10);
	ros::spin( );
	return 0;
}

ctrl+s快捷保存

ctrl+shift+b快捷编译
采用wpr_simulation开源工程,打开三个终端分别运行三条指令

roscore
roslaunch wpr_simulation wpb_simple.launch
rosrun imu_pkg imu_node

在这里插入图片描述

6. 用python编写 IMU 航向锁定节点

在这里插入图片描述

基于前面学习的机器人运动控制和 IMU 惯性测量单元数据,下面将联系这两点编写 IMU 航向锁定节点,我们可以直接在前面实验的程序上做修改

6.1 构思功能的思路和步骤

构思
在这里插入图片描述
实现步骤

  1. 让大管家NodeHandle 发布速度控制话题/cmd_vel。
  2. 设定一个目标朝向角,当姿态信息中的朝向角和目标朝向角不一致时,控制机器人转向目标朝向角。

6.2 修改imu_node.py

打开4.4编写imu_node.py

在这里插入图片描述

修改imu_node.py源码

#!/usr/bin/env python3
#coding=utf-8
import rospy
from sensor msgs.msg import Imu
from tf.transformations import euler_from_quaternion
import math
from geometry msgs.msg import Twist

def imu callback(msg):
	if msg.orientation covariance[0]< 0:
		return
	quaternion =[
		msg .orientation.x,
		msg.orientation.y,
		msg.orientation.z,
		msg.orientation.w
	]
	(roll,pitch , yaw) = euler_from_quaternion(quaternion)
	roll = roll*180/math.pi
	pitch = pitch*180/math.pi
	yaw = yaw*180/math.pi
	rospy.loginfo(滚转=%.0f俯仰= %.0f朝向= %.of" ,roll,pitch,yaw)
	
	target_yaw = 90
	diff_angle = target yaw - yaw
	vel_cmd = Twist()
	vel_cmd.angular.z = diff_angle * 0.01
	global vel_pub
	vel_pub.publish(vel_cmd)


if _name ="_main_":
	rospy.init_node( "imu _node")
	imu_sub = rospy.subscriber( "/imu/data",Imu,imu_callback,queue_size=10)
	vel_pub = rospy.Publisher( "/cmd_vel",Twist,queue_size=10)

	rospy.spin()

ctrl+s快捷保存

6.3 运行imu_node节点

采用wpr_simulation开源工程,打开三个终端分别运行三条指令

roscore
roslaunch wpr_simulation wpb_simple.launch
rosrun imu_pkg imu_node.py


在这里插入图片描述
在这里插入图片描述

6.4 优化航向策略

修改imu_node.py源码

#!/usr/bin/env python3
#coding=utf-8
import rospy
from sensor msgs.msg import Imu
from tf.transformations import euler_from_quaternion
import math
from geometry msgs.msg import Twist

def imu callback(msg):
	if msg.orientation covariance[0]< 0:
		return
	quaternion =[
		msg .orientation.x,
		msg.orientation.y,
		msg.orientation.z,
		msg.orientation.w
	]
	(roll,pitch , yaw) = euler_from_quaternion(quaternion)
	roll = roll*180/math.pi
	pitch = pitch*180/math.pi
	yaw = yaw*180/math.pi
	rospy.loginfo(滚转=%.0f俯仰= %.0f朝向= %.of" ,roll,pitch,yaw)
	
	target_yaw = 90
	diff_angle = target yaw - yaw
	vel_cmd = Twist()
	vel_cmd.angular.z = diff_angle * 0.01
	vel_cmd.linear.x = 0.1
	global vel_pub
	vel_pub.publish(vel_cmd)


if _name ="_main_":
	rospy.init_node( "imu _node")
	imu_sub = rospy.subscriber( "/imu/data",Imu,imu_callback,queue_size=10)
	vel_pub = rospy.Publisher( "/cmd_vel",Twist,queue_size=10)

	rospy.spin()

ctrl+s快捷保存
采用wpr_simulation开源工程,打开三个终端分别运行三条指令

roscore
roslaunch wpr_simulation wpb_simple.launch
rosrun imu_pkg imu_node.py


在这里插入图片描述

可参照可以打开wpr_simulation里的script文件夹中demo_imu_behavior.py
在这里插入图片描述

7. 总结

本节学习了ROS机器人的IMU 惯性测量单元,尝试C++和python两种语言编写获取机器人姿态信息,并且结合前面的机器人运动编写了航向锁定节点,接下来会介绍机器人的更精彩的操作,期待你的关注。🎉🎉🎉

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

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

相关文章

vue3回到上一个路由页面

学习链接 Vue Router获取当前页面由哪个路由跳转 在Vue3的setup中如何使用this beforeRouteEnter 在这个路由方法中不能访问到组件实例this&#xff0c;但是可以使用next里面的vm访问到组件实例&#xff0c;并通过vm.$data获取组件实例上的data数据getCurrentInstance 是vue3提…

Java --- springboot2请求参数处理

目录 一、请求参数处理 1.1、请求映射 1.2、自定义请求规则 1.3、请求处理 1.4、普通参数与基本注解 1.4.1、注解 1.5、参数处理原则 1.6、复杂参数 1.7、自定义参数对象 1.8、自定义Converter 一、请求参数处理 1.1、请求映射 // RequestMapping(value "…

c#笔记-下载编辑器

IDE IDE是指集成开发环境&#xff08;Integrated Development Environment&#xff09;&#xff0c;是一种将软件开发所需的软件组合在一起&#xff0c;可以从同一操作界面以统一的操作方式使用的软件包。通常包括代码编辑器、编译器、链接器、调试器、测试工具、版本管理软件等…

自动化运维工具一Ansible Playbook语法实战

目录 一、Ansible Playbook剧本初识 1.1 Ansible Playbook 基本概述 1.1.1 什么是playbook 1.1.2 Ansible playbook 与AD-Hoc的关系 1.2 Ansible Playbook 书写格式 1.2.1安装NFS 服务 1.3 Playbook变量详解 1.3.1 使用 vars定义变量 1.3.2 使用 vars_flies定义变量 …

Java每日一练(20230501)

目录 1. 路径交叉 &#x1f31f;&#x1f31f; 2. 环形链表 &#x1f31f;&#x1f31f; 3. 被围绕的区域 &#x1f31f;&#x1f31f; &#x1f31f; 每日一练刷题专栏 &#x1f31f; Golang每日一练 专栏 Python每日一练 专栏 C/C每日一练 专栏 Java每日一练 专栏…

17自由度人形机器人实现行走功能

1. 功能说明 本文示例将实现R307样机17自由度人形机器人行走的功能。该项目利用探索者平台制作&#xff0c;其驱动系统采用伺服电机。 2. 仿人形机器人结构设计 人型机器人是一种旨在模仿人类外观和行为的机器人&#xff08;robot&#xff09;&#xff0c;尤其特指具有和人类相…

VS快捷键大全 | 掌握这些快捷键,助你调试快人一步

欢迎关注博主 Mindtechnist 或加入【Linux C/C/Python社区】一起学习和分享Linux、C、C、Python、Matlab&#xff0c;机器人运动控制、多机器人协作&#xff0c;智能优化算法&#xff0c;滤波估计、多传感器信息融合&#xff0c;机器学习&#xff0c;人工智能等相关领域的知识和…

Linux常见指令-1

本期我们开始学习Linux&#xff0c;首先我们来学习Linux的常见指令 目录 操作系统是什么 Linux下基本指令 1.ls指令 2.pwd指令 3.cd指令 4.touch指令 5.mkdir指令 6.rmdir指令 && rm 指令 7.man指令 8.cp指令 9.mv指令 10.cat指令 11.more指令 12.less指…

UE5中实现沿样条线创建网格体

本文是对UE官方教程&#xff1a;https://www.bilibili.com/video/BV1eU4y1c7XL的重现&#xff0c;原教程中通过构造函数实现非运行时执行脚本&#xff0c;并通过UE的样条线组件辅助创建路径网格体。该功能最终实现的效果如下&#xff1a; 1.创建基础蓝图 首先创建一个Actor蓝…

手把手教你 ,带你彻底掌握八大排序算法【数据结构】

文章目录 插入排序直接插入排序希尔排序 选择排序选择排序堆排序升序 交换排序冒泡排序快速排序递归hoare版本挖坑法前后指针版本 三数取中法选key递归到小的子区间时&#xff0c;可以考虑使用插入排序 归并排序递归实现非递归实现 排序算法复杂度以及稳定性 插入排序 直接插入…

计算机操作系统学习-引论

本专栏是对计算机操作系统学习的记录&#xff1a;《现代操作系统 第四版》&#xff0c;电子版的可以在评论区自取。 1 计算机硬件简介 操作系统与运行该操作系统的计算机硬件密切相关。如图1所示&#xff0c;我们可以将自己的计算机抽象为&#xff0c;CUP&#xff0c;内存和I/…

【数学建模】Day01——层次分析法

文章目录 1. 引出层次分析法1.1 思考问题1.2 平台借力1.3 分而治之的思想1.4 一致矩阵1.5 一致性检验1.6 一致矩阵计算权重1.7 判断矩阵求权重 2. 层次分析法2.1 定义2.2 具体步骤2.3 局限性 1. 引出层次分析法 1.1 思考问题 我们评价的目标是什么&#xff1f;我们为了达到这…

C语言:指针详解【进阶】后篇

目录 函数指针函数指针数组指向函数指针数组的指针回调函数 前言&#xff1a; 在C语言&#xff1a;指针详解【进阶】前篇中我们深入学习了字符指针&#xff0c;数组指针&#xff0c;指针数组以及数组传参和指针传参。我们对指针的应用有了较为深刻的认识&#xff0c;今天这里我…

BusterNet网络Python模型实现学习笔记之二

文章目录 一、squeeze函数的用法二、nn.CrossEntropyLoss函数三、isinstance函数四、定义冻结层 freeze_layers五、SummaryWriter 基础用法六、Python 基础语法1.变量嵌入到字符串2. enumerate() 函数3. 进度条库tqdm4. 字典&#xff08;dict&#xff09;展开为关键字参数&…

TAPFixer总结

相关工作 Menshen 检测属性用户写 et al检测属性就简单三个 未来工作&#xff1a; liveness; implicit; 数据集&#xff1b; 抽象方式合并&#xff1b;抽象规则配置&#xff1b;缓解谓词爆炸&#xff1b;concurrency的说明; 代码简化工作&#xff1b;给出能修复的漏洞种类 …

《基于光电容积法和机器学习的冠状动脉疾病患者出血风险预测》阅读笔记

目录 一、论文摘要 二、论文十问 三、论文亮点与不足之处 四、与其他研究的比较 五、实际应用与影响 六、个人思考与启示 参考文献 一、论文摘要 在冠状动脉疾病&#xff08;CAD&#xff09;患者的抗血栓治疗过程中&#xff0c;出血事件是关注的主要焦点。本研究旨在探讨…

浅谈一下布隆过滤器的设计之美

1 缓存穿透 2 原理解析 3 Guava实现 4 Redisson实现 5 实战要点 6 总结 布隆过滤器是一个非常有用的数据结构。它可以在大规模数据中高效地判断某个元素是否存在。布隆过滤器的应用非常广泛&#xff0c;不仅在搜索引擎、防垃圾邮件等领域中经常用到&#xff0c;而且在许多…

R语言单因素方差分析

R中的方差分析 介绍用于比较独立组的不同类型的方差分析&#xff0c;包括&#xff1a; 单因素方差分析&#xff1a;独立样本 t 检验的扩展&#xff0c;用于在存在两个以上组的情况下比较均值。这是方差分析检验的最简单情况&#xff0c;其中数据仅根据一个分组变量&#xff0…

【数据结构】七大排序总结

目录 &#x1f33e;前言 &#x1f33e; 内部排序 &#x1f308;1. 直接插入排序 &#x1f308;2. 希尔排序 &#x1f308;3. 直接选择排序 &#x1f308;4. 堆排序 &#x1f308;5. 归并排序 &#x1f308;6. 冒泡排序 &#x1f308;7. 快速排序 &#x1f33e;外部排序 &…

4 月份 火火火火 的开源项目

盘点 4 月份 GitHub 上 Star 攀升最多的开源项目&#xff0c;整个 4 月份最火项目 90% 都是 AI 项目&#xff08;准确的说&#xff0c;最近半年的热榜都是 AI 项目&#xff09; 本期推荐开源项目目录&#xff1a; 1. AI 生成逼真语音 2. 复旦大模型 MOSS&#xff01; 3. 让画中…