ROS中urdf文件与xacro文件的编写

本文详细介绍urdf文件的结构与编写技巧,包括link、joint标签,以及xacro的宏定义、调用和检查方法。同时涵盖xacro在代码复用和简化URDF编写中的作用,以及各自的launch文件编写范例和模型验证方式。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

urdf文件与xacro文件的编写

一、程序包及文件系统

创建工作空间 创建包等不再赘述
/工作空间/src/包 内文件结构如图
在这里插入图片描述

二、urdf的编写

1、.urdf文件的写法

1.1、机器人urdf总体框架:
<robot name="pr2">
<!-- 机器人 links  joints 等内容 -->
</robot>

机器人内容包含以下几个元素

<link>
定义杆件
<joint>
定义关节
<transmission>
定义变速模块(不常用到)
<gazebo>
用于扩展Gazebo仿真

其实我们最常用的就是link和joint两个标签,仿真会用到gazebo标签,transmission标签目前我也没接触到,以后补。

1.2、link标签

在这里插入图片描述
其实link标签只定义杆件的几何参数、惯量、坐标系零点等,不涉及父坐标系和子坐标系等问题,这些坐标关系的定义都在joint标签内进行。
link必须声明自己的名字 这个是必选属性。
link一图搞清结构
在这里插入图片描述
参考:http://wiki.ros.org/urdf/XML/link.

例子1:

 <link name="my_link">
   <inertial>
     <origin xyz="0 0 0.5" rpy="0 0 0"/>
     <mass value="1"/>
     <inertia ixx="100"  ixy="0"  ixz="0" iyy="100" iyz="0" izz="100" />
   </inertial>

   <visual>
     <origin xyz="0 0 0" rpy="0 0 0" />
     <geometry>
       <box size="1 1 1" />
     </geometry>
     <material name="Cyan">
       <color rgba="0 1.0 1.0 1.0"/>
     </material>
   </visual>

   <collision>
     <origin xyz="0 0 0" rpy="0 0 0"/>
     <geometry>
       <cylinder radius="1" length="0.5"/>
     </geometry>
   </collision>
 </link>

例子2:

  <link name="torso"><!--link名字必须声明-->
    <visual>
      <origin rpy="0 0 0" xyz="0 0 0"/>
      <geometry>
        <mesh filename="package://robot_description/meshes/base_link.DAE"/>
      </geometry>
    </visual>
    <collision>
      <origin rpy="0 0 0" xyz="-0.065 0 0.0"/>
      <geometry>
        <mesh filename="package://robot_description/meshes/base_link_simple.DAE"/>
      </geometry>
    </collision>
    <collision_checking>
      <origin rpy="0 0 0" xyz="-0.065 0 0.0"/>
      <geometry>
        <cylinder length="0.7" radius="0.27"/>
      </geometry>
    </collision_checking>
    <inertial>
      ...
    </inertial>
  </link>  
1.3、joint标签

在这里插入图片描述
joint一图搞清结构

在这里插入图片描述

type详细解释:
指定关节类型,其中type可以是下列之一:
revolute-沿轴旋转并具有由上限和下限指定的有限范围的铰链接头。
continuous-绕轴旋转的连续铰链接头,没有上限和下限。(正儿八经的旋转关节)
prismatic-沿轴滑动的滑动关节,并具有由上限和下限指定的有限范围。
fixed-这实际上不是关节,因为它无法移动。所有自由度均被锁定。这种类型的关节不需要轴,校准,动力学,极限或安全控制器。
floating-该关节允许在所有6个自由度上运动。
planar-此关节允许在垂直于轴的平面内运动。

limit标签各个参数的计算方法参考:http://wiki.ros.org/pr2_controller_manager/safety_limits.

参考: http://wiki.ros.org/urdf/XML/joint.
例子:

 <joint name="my_joint" type="floating">
    <origin xyz="0 0 1" rpy="0 0 3.1416"/>
    <parent link="link1"/>
    <child link="link2"/>

    <calibration rising="0.0"/>
    <dynamics damping="0.0" friction="0.0"/>
    <limit effort="30" velocity="1.0" lower="-2.2" upper="0.7" />
    <safety_controller k_velocity="10" k_position="15" soft_lower_limit="-2.0" soft_upper_limit="0.5" />
 </joint>
1.4、transmission标签

待补充
参考:http://wiki.ros.org/urdf/XML/Transmission.

1.5gazebo标签

待补充
参考:http://gazebosim.org/tutorials?tut=ros_urdf&cat=connect_ros.

2、个人案例:

<robot  name="robot_urdf">
  <link    name="base_link">
    <visual>
        <origin xyz = "0 0 0" rpy = "0 0 0 "/>
        <geometry>
            <box size ="0.5 0.3 0.2"/>
        </geometry>
        <material name = "yellow">
            <color rgba = "1 0.4 0 1"/>
        </material>
    </visual>
</link>

    <joint name = "left_wheel_joint" type = "continuous">
        <origin xyz = "-0.2 -0.15 -0.05" rpy = "0 0 0"/>
        <parent link = "base_link"/>
        <child link = "left_wheel_link"/>
        <axis xyz = "0 1 0"/>
    </joint>

    <link name = "left_wheel_link">
        <visual>
            <origin xyz = "0 0 0" rpy = "1.5707 0 0"/>
            <geometry>
                <cylinder radius = "0.075" length = "0.08"/>
            </geometry>
            <material name = "gray">
                <color rgba="0.75 0.75 0.75 1"/>
            </material>
        </visual>
    </link>

    <joint name = "right_wheel_joint" type = "continuous">
        <origin xyz = "-0.2 0.15 -0.05" rpy = "0 0 0"/>
        <parent link = "base_link"/>
        <child link = "right_wheel_link"/>
        <axis xyz = "0 1 0"/>
    </joint>

    <link name = "right_wheel_link">
        <visual>
            <origin xyz = "0 0 0" rpy = "1.5707 0 0"/>
            <geometry>
                <cylinder radius = "0.075" length = "0.08"/>
            </geometry>
            <material name = "gray">
                <color rgba="0.75 0.75 0.75 1"/>
            </material>
        </visual>
    </link>


    <joint name = "front_castor_joint" type = "fixed">
        <origin xyz = "0.2 0 -0.05 " rpy = "0 0 0 "/>
        <parent link = "base_link"/>
        <child link = "front_castor_link"/>
        <axis xyz = "0 1 0"/>
    </joint>

    <link name = "front_castor_link">
        <visual>
            <origin xyz = "0 0 0" rpy ="1.5707 0 0"/>
            <geometry>
                <sphere radius = "0.075"/>
            </geometry>
            <material name = "black">
                <color rgba = "0 0 0 0.95"/>
            </material>
        </visual>
    </link>

</robot>
    

上边的URDF模型定义了一个小车,和两个驱动轮一个万向轮相连

3、urdf文件检查方法

ROS为用户提供了一个检查URDF语法的工具:
未安装则使用如下命令安装:

sudo apt-get install liburdfdom-tools

安装完毕后,执行检查,二选一:

check_urdf mbot.urdf
urdf_to_graphiz mbot.urdf 

4、urdf文件的.launch文件编写

<launch>
     <!--加载机器人模型-->
    <param name = "robot_description" textfile = "$(find robot_description)/urdf/robot_urdf.urdf"/>
    <!-- 设置GUI参数,显示关节控制插件 -->
    <param name = "use_gui" value = "true"/>
	<!-- 运行joint_state_publisher_gui节点,发布机器人的关节状态  -->
    <node name ="joint_state_publisher" pkg = "joint_state_publisher" type = "joint_state_publisher"/>
    <!-- 运行robot_state_publisher节点,发布tf,将机器人的连杆关节之间的关系,通过TF的形式,整理成三维姿态发布出去  -->
    <node name = "robot_state_publisher" pkg = "robot_state_publisher" type = "state_publisher"/>
 	<!-- 运行rviz  -->
    <node name = "rviz" pkg = "rviz" type = "rviz" args = "-d $(find robot_description)/config/robot_urdf.rviz" required = "true"/>
</launch>

三、xacro的编写

1、.xacro文件写法

xacro按照我的学习理解就是解决代码可读性和代码复用问题的
urdf写多了你会感觉到好难改,因为不像c++那样有变量和循环,很多变量你需要一个个找一个个改,而xacro中不仅有变量,还可以将一个零件写的模块化。修改一个模型如果不调整整体结构,只改变大小,可以仅通过修改变量实现。
这种标签类语言我本人之前也是从未接触过的,如果有说的不对的地方,各位可以指出来。
xacro的核心在于宏定义
核心代码段就是下面的

<xacro:macro name="pr2_arm" params="suffix parent reflect">
  <pr2_upperarm suffix="${suffix}" reflect="${reflect}" parent="${parent}" />
  <pr2_forearm suffix="${suffix}" reflect="${reflect}" parent="elbow_flex_${suffix}" />
</xacro:macro>

<xacro:pr2_arm suffix="left" reflect="1" parent="torso" />
<xacro:pr2_arm suffix="right" reflect="-1" parent="torso" />

其实整个宏定义就是以

<xacro:macro name="pr2_arm" params="suffix parent reflect">

开头,

</xacro:macro>

结束
这表示整个其中的内容都是可以复用的
name参数就是起个函数名,params就是此段复用代码的输入参数
比如params =“suffix parent reflect”,就是使用了三个参数(输入的变量)
后面的<xacro:pr2_arm suffix=“left” reflect=“1” parent=“torso” />
在我理解就是调用本段宏(我个人理解的函数),输入参数(变量)为 suffix=“left” reflect=“1” parent=“torso”

<xacro:pr2_arm suffix="left" reflect="1" parent="torso" />
<xacro:pr2_arm suffix="right" reflect="-1" parent="torso" />

即为调用了两次这个宏(函数),也就是将两个机械臂(结构)实现了。
除此之外,这个宏(函数)也可以无参数,如下

<xacro:macro name="robot_base">
        <link name="base_link">
            <visual>
                <origin xyz="0 0 0" rpy="0 0 0" />
                <geometry>
                    <box size="${base_length} ${base_width} ${base_height}" />
                </geometry>
                <material name="yellow" />
            </visual>
        </link>

        <wheel prefix="left" reflect="-1"/>
        <wheel prefix="right" reflect="1"/>
    </xacro:macro>

这就表示了一个小车底盘加上了左右两个轮子。
但是这些不复用的宏都要写到一起,在这一段中将整个base_link和左右轮都放到了一起,如果将车的base_link和其他零件分割到两个宏中定义,就会查找不到base_link或者其他link,因为他们分成了两个宏,没有连接到一起。这点一定要注意!!!将表示base_link与各个零件相互连接关系的代码段放到一个宏内。(都一起放到</xacro:macro>这对标签外面,我测试了也ok,目前不知道这样做会带来什么后果,不推荐)

<xacro:macro name="robot_base">
<!--中间是对base的定义和对各个link、joint组成的宏进行调用-->
</xacro:macro>

个人案例:

<?xml version="1.0"?>
<robot name="mbot" xmlns:xacro="http://www.ros.org/wiki/xacro">

    <!-- PROPERTY LIST -->
    <xacro:property name="M_PI" value="3.1415926"/>
    <xacro:property name="base_length" value="0.5"/>
    <xacro:property name="base_width" value="0.3"/>
    <xacro:property name="base_height" value="0.2"/>

    <xacro:property name="wheel_radius" value="0.075"/>
    <xacro:property name="wheel_length" value="0.08"/>
    <xacro:property name="wheel_joint_y" value="-0.15"/>
    <xacro:property name="wheel_joint_z" value="-0.05"/>
    <xacro:property name="wheel_joint_x" value="-0.15"/>

    <xacro:property name="caster_radius" value="0.075"/> 
    <xacro:property name="caster_joint_x" value="0.18"/>
    <xacro:property name="caster_joint_z" value="-0.05"/>

    <!-- Defining the colors used in this robot -->
    <material name="yellow">
        <color rgba="1 0.4 0 1"/>
    </material>
    <material name="black">
        <color rgba="0 0 0 0.95"/>
    </material>
    <material name="gray">
        <color rgba="0.75 0.75 0.75 1"/>
    </material>
    
    <!-- Macro for robot wheel -->
    <xacro:macro name="wheel" params="prefix reflect">
        <joint name="${prefix}_wheel_joint" type="continuous">
            <origin xyz="${wheel_joint_x} ${reflect*wheel_joint_y} ${wheel_joint_z}" rpy="0 0 0"/>
            <parent link="base_link"/>
            <child link="${prefix}_wheel_link"/>
            <axis xyz="0 1 0"/>
        </joint>

        <link name="${prefix}_wheel_link">
            <visual>
                <origin xyz="0 0 0" rpy="${M_PI/2} 0 0" />
                <geometry>
                    <cylinder radius="${wheel_radius}" length = "${wheel_length}"/>
                </geometry>
                <material name="gray" />
            </visual>
        </link>
    </xacro:macro>

    <!-- Macro for robot base -->
    <xacro:macro name="mbot_base">
        <link name="base_footprint">
            <visual>
                <origin xyz="0 0 0" rpy="0 0 0" />
                <geometry>
                    <box size="0.001 0.001 0.001" />
                </geometry>
            </visual>
        </link>

        <joint name="base_footprint_joint" type="fixed">
            <origin xyz="0 0 ${base_height/2 + caster_joint_z +caster_radius}" rpy="0 0 0" />        
            <parent link="base_footprint"/>
            <child link="base_link" />
        </joint>

        <link name="base_link">
            <visual>
                <origin xyz=" 0 0 0" rpy="0 0 0" />
                <geometry>
                    <box size="${base_length} ${base_width} ${base_height}" />
                </geometry>
                <material name="yellow" />
            </visual>
        </link>

        <joint name="caster_joint" type="continuous">
            <origin xyz="${caster_joint_x} 0 ${caster_joint_z}" rpy="0 0 0"/>
            <parent link="base_link"/>
            <child link="caster_link"/>
            <axis xyz="0 1 0"/>
        </joint>

        <link name="caster_link">
            <visual>
                <origin xyz="0 0 0" rpy="0 0 0"/>
                <geometry>
                    <sphere radius="${caster_radius}" />
                </geometry>
                <material name="black" />
            </visual>
        </link>

        <wheel prefix="left" reflect="-1"/>
        <wheel prefix="right" reflect="1"/>


    </xacro:macro>
</robot>

这个文件和之前的urdf一样,也是一个base_link 左右连了两个车轮,前边一个万向轮。

2、xacro文件的调用

你的小车底盘已经都写好了,现在如果有一个雷达、一个摄像头要家装上去怎么办呢,又重写刚才的.xacro吗?不慌,他又提供了类似c的include调用。
如果小车和雷达你都有了只想装配起来使用,那就使用这样的标签进行调用,以下方法三选一:

<xacro:include filename="$(find package)/other_file.xacro" />
<xacro:include filename="other_file.xacro" />
<xacro:include filename="$(cwd)/other_file.xacro" />

文件“ other_file.xacro”将被xacro包含并扩展

个人案例:

<?xml version="1.0"?>
<robot name="robot_my" xmlns:xacro="http://www.ros.org/wiki/xacro"><!-- 声明是xacro文件 -->
    <xacro:include filename="$(find robot_description)/urdf/xacro/robot_xacro.xacro"/> <!-- 包含文件 -->
    <robot_xacro/>   <!-- 调用宏定义 -->
</robot>

注:ros noetic以上版本使用 <xacro:robot_xacro/>调用宏定义,其他版本使用 <robot_xacro/> 调用宏定义
参考:http://wiki.ros.org/xacro/.

3、xacro文件的检查方法

xacro没有一个很好的可视化检查方法,目前想检查职能转换为.urdf文件再检查,命令如下:

rosrun xacro xacro --inorder mbot.xacro> new_mbot_from_xacro.urdf

即为:

rosrun xacro xacro --inorder 原有xacro文件.xacro> 转格式后的urdf文件.urdf

之后执行urdf检查,和上面的urdf的检查方法一致,二选一:

check_urdf new_mbot_from_xacro.urdf
urdf_to_graphiz new_mbot_from_xacro.urdf 

注:在做检查步骤之前一定要搞好环境变量
source devel/setup.bash 之后cd到你的xacro文件所在的目录下

4、xacro文件的.launch文件写法

xacro模型的launch文件的写法其实和urdf的launch文件差不多。只有加载机器人代码段不同,如下:

<launch>
	<!-- 加载机器人(小车)xacro文件 -->
	<arg name="model" default="$(find xacro)/xacro --inorder '$(find mbot_description)/urdf/xacro/mbot.xacro'" />
	<arg name="gui" default="true" />

	<!-- 设置GUI参数,显示关节控制插件 -->
	<param name="robot_description" command="$(arg model)" />
	<param name="use_gui" value="$(arg gui)"/>

    <!-- 运行joint_state_publisher节点,发布机器人的关节状态  -->
	<node name="joint_state_publisher" pkg="joint_state_publisher" type="joint_state_publisher" />

	<!-- 运行robot_state_publisher节点,发布tf  -->
	<node name="robot_state_publisher" pkg="robot_state_publisher" type="robot_state_publisher" />

    <!-- 运行rviz可视化界面 -->
	<node name="rviz" pkg="rviz" type="rviz" args="-d $(find mbot_description)/config/mbot.rviz" required="true" />

</launch>

5、执行.launch文件后查看效果

roslaunch mbot_description display_mbot_base_xacro.launch

urdf的显示方法同理 roslaunch 你的urdf模型对应的.launch文件
选择正确的坐标系,我这里选择base_footprint
添加机器人模型显示
点击add 选择robotmodel
在这里插入图片描述

在这里插入图片描述
如果想查看tf坐标关系就在add打开的窗口中选择TF添加,此处不再赘述。

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值