与关节交互#
本教程展示了如何与模拟中的关节机器人进行交互。这是 与刚性物体交互 教程的延续,其中我们学习了如何与刚体对象交互。除了设置根状态外,我们还将看到如何设置关节状态并向关节机器人应用命令。
代码#
该教程对应于 source/standalone/tutorials/01_assets
目录中的 run_articulation.py
脚本。
代码 run_articulation.py 的内容
1# Copyright (c) 2022-2025, The Isaac Lab Project Developers.
2# All rights reserved.
3#
4# SPDX-License-Identifier: BSD-3-Clause
5
6"""This script demonstrates how to spawn a cart-pole and interact with it.
7
8.. code-block:: bash
9
10 # Usage
11 ./isaaclab.sh -p source/standalone/tutorials/01_assets/run_articulation.py
12
13"""
14
15"""Launch Isaac Sim Simulator first."""
16
17
18import argparse
19
20from omni.isaac.lab.app import AppLauncher
21
22# add argparse arguments
23parser = argparse.ArgumentParser(description="Tutorial on spawning and interacting with an articulation.")
24# append AppLauncher cli args
25AppLauncher.add_app_launcher_args(parser)
26# parse the arguments
27args_cli = parser.parse_args()
28
29# launch omniverse app
30app_launcher = AppLauncher(args_cli)
31simulation_app = app_launcher.app
32
33"""Rest everything follows."""
34
35import torch
36
37import omni.isaac.core.utils.prims as prim_utils
38
39import omni.isaac.lab.sim as sim_utils
40from omni.isaac.lab.assets import Articulation
41from omni.isaac.lab.sim import SimulationContext
42
43##
44# Pre-defined configs
45##
46from omni.isaac.lab_assets import CARTPOLE_CFG # isort:skip
47
48
49def design_scene() -> tuple[dict, list[list[float]]]:
50 """Designs the scene."""
51 # Ground-plane
52 cfg = sim_utils.GroundPlaneCfg()
53 cfg.func("/World/defaultGroundPlane", cfg)
54 # Lights
55 cfg = sim_utils.DomeLightCfg(intensity=3000.0, color=(0.75, 0.75, 0.75))
56 cfg.func("/World/Light", cfg)
57
58 # Create separate groups called "Origin1", "Origin2"
59 # Each group will have a robot in it
60 origins = [[0.0, 0.0, 0.0], [-1.0, 0.0, 0.0]]
61 # Origin 1
62 prim_utils.create_prim("/World/Origin1", "Xform", translation=origins[0])
63 # Origin 2
64 prim_utils.create_prim("/World/Origin2", "Xform", translation=origins[1])
65
66 # Articulation
67 cartpole_cfg = CARTPOLE_CFG.copy()
68 cartpole_cfg.prim_path = "/World/Origin.*/Robot"
69 cartpole = Articulation(cfg=cartpole_cfg)
70
71 # return the scene information
72 scene_entities = {"cartpole": cartpole}
73 return scene_entities, origins
74
75
76def run_simulator(sim: sim_utils.SimulationContext, entities: dict[str, Articulation], origins: torch.Tensor):
77 """Runs the simulation loop."""
78 # Extract scene entities
79 # note: we only do this here for readability. In general, it is better to access the entities directly from
80 # the dictionary. This dictionary is replaced by the InteractiveScene class in the next tutorial.
81 robot = entities["cartpole"]
82 # Define simulation stepping
83 sim_dt = sim.get_physics_dt()
84 count = 0
85 # Simulation loop
86 while simulation_app.is_running():
87 # Reset
88 if count % 500 == 0:
89 # reset counter
90 count = 0
91 # reset the scene entities
92 # root state
93 # we offset the root state by the origin since the states are written in simulation world frame
94 # if this is not done, then the robots will be spawned at the (0, 0, 0) of the simulation world
95 root_state = robot.data.default_root_state.clone()
96 root_state[:, :3] += origins
97 robot.write_root_link_pose_to_sim(root_state[:, :7])
98 robot.write_root_com_velocity_to_sim(root_state[:, 7:])
99 # set joint positions with some noise
100 joint_pos, joint_vel = robot.data.default_joint_pos.clone(), robot.data.default_joint_vel.clone()
101 joint_pos += torch.rand_like(joint_pos) * 0.1
102 robot.write_joint_state_to_sim(joint_pos, joint_vel)
103 # clear internal buffers
104 robot.reset()
105 print("[INFO]: Resetting robot state...")
106 # Apply random action
107 # -- generate random joint efforts
108 efforts = torch.randn_like(robot.data.joint_pos) * 5.0
109 # -- apply action to the robot
110 robot.set_joint_effort_target(efforts)
111 # -- write data to sim
112 robot.write_data_to_sim()
113 # Perform step
114 sim.step()
115 # Increment counter
116 count += 1
117 # Update buffers
118 robot.update(sim_dt)
119
120
121def main():
122 """Main function."""
123 # Load kit helper
124 sim_cfg = sim_utils.SimulationCfg(device=args_cli.device)
125 sim = SimulationContext(sim_cfg)
126 # Set main camera
127 sim.set_camera_view([2.5, 0.0, 4.0], [0.0, 0.0, 2.0])
128 # Design scene
129 scene_entities, scene_origins = design_scene()
130 scene_origins = torch.tensor(scene_origins, device=sim.device)
131 # Play the simulator
132 sim.reset()
133 # Now we are ready!
134 print("[INFO]: Setup complete...")
135 # Run the simulator
136 run_simulator(sim, scene_entities, scene_origins)
137
138
139if __name__ == "__main__":
140 # run the main function
141 main()
142 # close sim app
143 simulation_app.close()
代码解析#
场景设计#
与上一个教程相似,我们在场景中添加了地面平面和远程光线。现在,我们不再产生刚性物体,而是从其 USD 文件中生成一个车杆关节。车杆是一个简单的机器人,由一个小车和一个连接到它的杆组成。小车可以沿 x 轴自由移动,而杆可以围绕小车自由旋转。车杆的 USD 文件包含了机器人的几何形状、关节和其他物理性质。
对于车杆,我们使用其预定义的配置对象,这是 assets.ArticulationCfg
类的一个实例。这个类包含关于关节产生策略、默认初始状态、不同关节的执行器模型和其他元信息的信息。如何创建这个配置对象的更深入介绍在 编写资产配置 教程中提供。
正如在上一个教程中看到的,我们可以通过创建 assets.Articulation
类的实例并将配置对象传递给它的构造函数的方式类似地将关节放入场景中。
# Create separate groups called "Origin1", "Origin2"
# Each group will have a robot in it
origins = [[0.0, 0.0, 0.0], [-1.0, 0.0, 0.0]]
# Origin 1
prim_utils.create_prim("/World/Origin1", "Xform", translation=origins[0])
# Origin 2
prim_utils.create_prim("/World/Origin2", "Xform", translation=origins[1])
# Articulation
cartpole_cfg = CARTPOLE_CFG.copy()
cartpole_cfg.prim_path = "/World/Origin.*/Robot"
cartpole = Articulation(cfg=cartpole_cfg)
运行模拟循环#
从上一个教程中继续,我们定期重置模拟,对关节机器人设置命令,推进模拟,并更新关节机器人的内部缓冲区。
重置模拟#
与刚体物体类似,关节机器人也有一个根状态。这个状态对应于关节机器人树中的根体。在根状态之上,关节机器人还有关节状态。这些状态对应于关节的位置和速度。
为了重置关节机器人,我们首先通过调用 Articulation.write_root_link_pose_to_sim()
和 Articulation.write_root_com_velocity_to_sim()
方法设置根状态。类似地,我们通过调用 Articulation.write_joint_state_to_sim()
方法设置关节状态。最后,我们调用 Articulation.reset()
方法重置任何内部缓冲区和缓存。
# reset the scene entities
# root state
# we offset the root state by the origin since the states are written in simulation world frame
# if this is not done, then the robots will be spawned at the (0, 0, 0) of the simulation world
root_state = robot.data.default_root_state.clone()
root_state[:, :3] += origins
robot.write_root_link_pose_to_sim(root_state[:, :7])
robot.write_root_com_velocity_to_sim(root_state[:, 7:])
# set joint positions with some noise
joint_pos, joint_vel = robot.data.default_joint_pos.clone(), robot.data.default_joint_vel.clone()
joint_pos += torch.rand_like(joint_pos) * 0.1
robot.write_joint_state_to_sim(joint_pos, joint_vel)
# clear internal buffers
robot.reset()
推进模拟#
向关节机器人应用命令涉及两个步骤:
设置关节目标: 这为关节机器人设置期望的关节位置、速度或努力目标。
将数据写入模拟: 根据关节机器人的配置,这一步处理任何 动作转换 并将转换后的值写入 PhysX 缓冲区。
在本教程中,我们使用关节努力命令来控制关节机器人。为此工作,我们需要将关节机器人的刚度和阻尼参数设置为零。这是在车杆的预定义配置对象内部预先完成的。
在每一步中,我们随机采样关节努力并通过调用 Articulation.set_joint_effort_target()
方法将它们设置给关节机器人。在设置好目标后,我们通过调用 Articulation.write_data_to_sim()
方法将数据写入 PhysX 缓冲区。最后,我们推进模拟。
# Apply random action
# -- generate random joint efforts
efforts = torch.randn_like(robot.data.joint_pos) * 5.0
# -- apply action to the robot
robot.set_joint_effort_target(efforts)
# -- write data to sim
robot.write_data_to_sim()
更新状态#
每个关节机器人类包含一个 assets.ArticulationData
对象。这个对象保存了关节机器人的状态。为了更新缓冲区内的状态,我们调用 assets.Articulation.update()
方法。
# Update buffers
robot.update(sim_dt)
代码执行#
要运行代码并查看结果,让我们从终端运行脚本:
./isaaclab.sh -p source/standalone/tutorials/01_assets/run_articulation.py
这个命令应该会打开一个包含地面平面、灯光和两个随机移动的cart-poles的场景。要停止模拟,您可以关闭窗口,或在终端中按 Ctrl+C
。
在本教程中,我们学会了如何创建和交互一个简单的关节机器人。我们看到了如何设置关节机器人的状态(其根和关节状态),以及如何对其应用命令。我们还看到了如何更新其缓冲区,以从模拟中读取最新状态。
除了这个教程之外,我们还提供了一些生成不同机器人的其他脚本。这些包括在 source/standalone/demos
目录中。您可以运行这些脚本,比如:
# Spawn many different single-arm manipulators
./isaaclab.sh -p source/standalone/demos/arms.py
# Spawn many different quadrupeds
./isaaclab.sh -p source/standalone/demos/quadrupeds.py