双足机器人简单步态生成

   让机器人行走最简单的方法是先得到一组步态曲线,即腿部每个关节随时间运动的角度值。可以在ADAMS或3D Max、Blender等软件中建立好机构/骨骼模型,设计出脚踝和髋关节的运动曲线,然后进行逆运动学解算,测量每个关节在运动过程中的转角,最后将得到的曲线导出。拿到曲线数据后我们就可以用单片机读取,然后发送给机器人的舵机去执行运行。这种方法的缺点是机器人只能按照固定的步态行走,不够灵活,比如抬脚高度、步长等参数都是定死的,如果需要修改还得再使用别的软件导出新的步态数据。

  最简单的腿部结构如下图所示,在髋关节、膝关节和踝关节各有一个转动自由度,可以通过三角形余弦定理求得机构的运动学逆解。这种机器人只能在矢状面上直线前进,不能转弯。对行走过程进行一定的简化和假设:

1. 髋关节始终保持恒定的高度(实际上会有微小的波动)

2. 机器人脚面始终平行于地面

  为了确定每个关节的角度,需要设计行走过程中踝关节点的运动轨迹。这里采用简单的正弦曲线作为其轨迹(也可以采用样条曲线、Bézier 曲线等),正弦曲线的幅值对应抬脚最大高度。

 

  在Python中实现导入双腿模型文件,生成指定的步态数据后让其循环运动,就可以模拟机器人行走。代码如下(很糙...只是实现了基本功能,细节还有待完善)

#!/usr/bin/env python
import vtk
import math
from vtk.util.colors import *
import numpy as np
import timeactor  = list()   # the list of links
filenames = ["link-1.stl","link-2.stl","link-3.stl","link-4.stl","link-5.stl","link-6.stl"]
renWin = vtk.vtkRenderWindow()joint1 = vtk.vtkAssembly()
joint2 = vtk.vtkAssembly()
joint3 = vtk.vtkAssembly()
joint4 = vtk.vtkAssembly()
joint5 = vtk.vtkAssembly()
joint6 = vtk.vtkAssembly()ThighLength = 100.0
ShankLength = 100.0
HipHeight = 180.0
FootLift = 10
StrideLength = 60
Subdivision = 20leg_joint = np.zeros(3, dtype=np.float)
patterns = np.zeros((2*Subdivision ,6), dtype=np.float)
_p = 0txt = vtk.vtkTextActor()
distance = 0.0def Rad2Deg(rad):return  rad * 180.0 / math.pidef FootHeight(x): return (HipHeight - FootLift * math.cos(abs(x * math.pi / StrideLength)))def LegIK(x, y):global leg_jointdist = math.sqrt(x**2 + y**2)leg_joint[0] = math.acos(dist / (2 * ShankLength)) + math.atan2(x, y)leg_joint[1] = math.pi - math.acos((ThighLength**2 + ShankLength**2 - dist**2) / (2 * ThighLength* ShankLength))leg_joint[2] = leg_joint[1] - leg_joint[0]def GenerateGait():global leg_jointglobal patterns# Move left leg forward.  for i in range(Subdivision):x = (i - Subdivision/2) * (StrideLength / Subdivision)y = FootHeight(x)        LegIK(x, y)            patterns[i, :3] = Rad2Deg(leg_joint)# Move left leg backward.  for i in range(Subdivision):x = (Subdivision/2 - i) * (StrideLength / Subdivision)y = HipHeightLegIK(x, y)                patterns[i+Subdivision, :3] = Rad2Deg(leg_joint)  # Build right leg from phase shift clone of left. for i in range(2*Subdivision):patterns[i, 3:] = -patterns[(i + Subdivision) % (2*Subdivision), :3]# Customize vtkInteractorStyleTrackballCamera 
class MyInteractor(vtk.vtkInteractorStyleTrackballCamera):def __init__(self,parent=None):self.AddObserver("CharEvent",self.OnCharEvent)self.AddObserver("KeyPressEvent",self.OnKeyPressEvent)def OnCharEvent(self,obj,event):passdef OnKeyPressEvent(self,obj,event):global _pglobal distance# Get the compound key strokes for the eventkey = self.GetInteractor().GetKeySym()GenerateGait()if(key == "Return"):# start animationjoint1.SetPosition(0, 0, HipHeight-ThighLength-ShankLength)joint4.SetPosition(0, 0, HipHeight-ThighLength-ShankLength)if (_p == 2*Subdivision):_p = 0joint1.SetOrientation(0, -patterns[_p][0], 0)joint2.SetOrientation(0, patterns[_p][1], 0)joint3.SetOrientation(0, -patterns[_p][2], 0)joint4.SetOrientation(0, patterns[_p][3], 0)joint5.SetOrientation(0, -patterns[_p][4], 0)joint6.SetOrientation(0, patterns[_p][5], 0)_p = _p + 1distance = distance + StrideLength/(2 * Subdivision * 1000.0)txt.SetInput("Distance: " + str(distance) + "m")renWin.Render()returndef CreateCoordinates():# create coordinate axes in the render windowaxes = vtk.vtkAxesActor() axes.SetTotalLength(40, 40, 40)  # Set the total length of the axes in 3 dimensions # Set the type of the shaft to a cylinder:0, line:1, or user defined geometry. 
    axes.SetShaftType(0) transform = vtk.vtkTransform() transform.Translate(0.0, 0.0, 200.0)axes.SetUserTransform(transform)axes.SetCylinderRadius(0.02) axes.GetXAxisCaptionActor2D().SetWidth(0.03) axes.GetYAxisCaptionActor2D().SetWidth(0.03) axes.GetZAxisCaptionActor2D().SetWidth(0.03) return axesdef CreateGround():# create plane sourceplane = vtk.vtkPlaneSource()plane.SetXResolution(20)plane.SetYResolution(20)plane.SetCenter(0,0,0)plane.SetNormal(0,0,1)# mappermapper = vtk.vtkPolyDataMapper()mapper.SetInputConnection(plane.GetOutputPort())# actoractor = vtk.vtkActor()actor.SetMapper(mapper)actor.GetProperty().SetRepresentationToWireframe()actor.GetProperty().SetColor(light_grey)transform = vtk.vtkTransform()transform.Scale(400,400,1)actor.SetUserTransform(transform)return actordef LoadSTL(filename):reader = vtk.vtkSTLReader()reader.SetFileName(filename)mapper = vtk.vtkPolyDataMapper() # maps polygonal data to graphics primitives
    mapper.SetInputConnection(reader.GetOutputPort())actor = vtk.vtkLODActor() actor.SetMapper(mapper)return actor   # represents an entity in a rendered scenedef CreateScene():# Create a rendering window and rendererren = vtk.vtkRenderer()renWin.AddRenderer(ren)# Create a renderwindowinteractoriren = vtk.vtkRenderWindowInteractor()iren.SetRenderWindow(renWin)style = MyInteractor()style.SetDefaultRenderer(ren)iren.SetInteractorStyle(style)for id, file in enumerate(filenames):actor.append(LoadSTL(file))r = vtk.vtkMath.Random(.4, 1.0)g = vtk.vtkMath.Random(.4, 1.0)b = vtk.vtkMath.Random(.4, 1.0)actor[id].GetProperty().SetDiffuseColor(r, g, b)actor[id].GetProperty().SetDiffuse(.8)actor[id].GetProperty().SetSpecular(.5)actor[id].GetProperty().SetSpecularColor(1.0,1.0,1.0)actor[id].GetProperty().SetSpecularPower(30.0)joint1.AddPart(actor[0])joint1.AddPart(joint2)joint2.AddPart(actor[1])joint2.AddPart(joint3)joint3.AddPart(actor[2])joint4.AddPart(actor[3])joint4.AddPart(joint5)joint5.AddPart(actor[4])joint5.AddPart(joint6)joint6.AddPart(actor[5])joint1.SetOrigin(0, 0, 200)joint4.SetOrigin(0, 0, 200)joint2.SetOrigin(0, 0, 100)joint5.SetOrigin(0, 0, 100)ren.AddActor(joint1)ren.AddActor(joint4)# Add coordinatesaxes = CreateCoordinates()ren.AddActor(axes)# Add groundground = CreateGround()ren.AddActor(ground)# create a text actortxt.SetInput("Distance: 0m")txtprop=txt.GetTextProperty()txtprop.SetFontFamilyToArial()txtprop.SetFontSize(18)txtprop.SetColor(1,1,1)txt.SetDisplayPosition(450,550)# assign actor to the renderer
    ren.AddActor(txt)# Set background colorren.SetBackground(.1, .1, .1)# Set window sizerenWin.SetSize(600, 600)# Enable user interface interactor
    iren.Initialize()iren.Start()if __name__ == "__main__":CreateScene()
View Code

  按住回车键,一帧一帧播放动画。最后的效果是这样的:

 

 

参考:

https://github.com/Rhoban/IKWalk

Using Inverse Kinematics to Develop a Biped Robot Walking Gait C#

8 DOF Biped Robot using Dynamixel AX-12A Servos and Arduino

转载于:https://www.cnblogs.com/21207-iHome/p/6052159.html

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

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

相关文章

重新访问了访客模式

访客模式是面向对象设计中最被高估但又被低估的模式之一。 高估了它,因为它常常被选择得太快( 可能是由建筑宇航员选择的 ),然后以错误的方式添加时会膨胀本来非常简单的设计。 如果您不遵循教科书示例,那么它可能会非…

ASP.NET—013:实现带控件的弹出层(弹出框)

http://blog.csdn.net/yysyangyangyangshan/article/details/38458169 在页面中用到弹出新页面的情况比较多的,一般来说都是使用JS方法showModalDialog("新页面相对路径?参数1&参数2",window,"新页面样式");然后会新弹出一个模态的page页。…

java 二进制 归属权限_【Java EE 学习 75 上】【数据采集系统第七天】【二进制运算实现权限管理】【权限分析和设计】...

一、权限计算相关分析1.如何存储权限首先说一下权限保存的问题,一个系统中最多有多少权限呢?一个大的系统中可能有成百上千个权限需要管理。怎么保存这么多的权限?首先,我们使用一个数字中的一位保存一种权限,那么如果…

GWT和HTML5 Canvas演示

这是我对GWT和HTML5 Canvas的第一个实验。 我的第一个尝试是创建矩形,仅用几行代码就得出了这样的内容: 码: public class GwtHtml5 implements EntryPoint {static final String canvasHolderId "canvasholder";static final St…

Solr管理界面详解

转载于:https://www.cnblogs.com/gslblog/p/6553813.html

一个实用的却被忽略的命名空间:Microsoft.VisualBasic:

当你看到这个命名空间的时候,别因为是vb的东西就匆忙关掉网页,那将会是您的损失,此命名空间中的资源最初目的是为了简化vb.net开发而创建的,所以microsoft.visualbasic并不属于system命名空间,而是独立存在的。虽然是为…

Linux基础之命令练习Day2-useradd(mod,del),groupadd(mod,del),chmod,chown,

作业一: 1) 新建用户natasha,uid为1000,gid为555,备注信息为“master” 2) 修改natasha用户的家目录为/Natasha 3) 查看用户信息配置文件的最后一行 4) 为natasha用户设置密码“123” 5) 查看用户密码配置文件的最后一行 6) 将nat…

动态表单,JSF世界早已等待

新的PrimeFaces扩展版本0.5.0带来了新的DynaForm组件。 通常,如果知道行/列的数量,元素的位置等,则可以通过h:panelGrid或p:panelGrid来构建非常简单的表单。 对于静态表单,这是正确的。 但是,如…

C# 定时器事件(设置时间间隔,间歇性执行某一函数,控制台程序)

定时器事件代码 static void Main(string[] args) {Method();#region 定时器事件 Timer aTimer new Timer();aTimer.Elapsed new ElapsedEventHandler(TimedEvent);aTimer.Interval seconds * 1000; //配置文件中配置的秒数aTimer.Enabled true;#endregionstring strLi…

Vmware安装Centos NAT方式设置静态IP

【Vmware中在搭建集群环境等,DHCP自动获取IP方式不方便,为了固定IP减少频繁更改配置信息,建议使用静态IP来配置,网络连接主要有三种方式 1.nat 2.桥接,3主机模式 ,在这里主要介NAT方式, 为什么使…

1 TB /节点时快速,可预测且高度可用

世界正每秒从移动设备,Web和各种小工具向应用程序推送大量数据。 如今,更多的应用程序必须处理此数据。 为了保持性能,这些应用程序需要快速访问数据层。 在过去的几年中,RAM价格下降了,我们现在可以便宜得多地获得具有…

java jni 内存_Android开发之JNI内存模型

Java 与JNI 内存管理是怎样的想要弄清楚Java与JNI的内存管理的关系,首先要弄清楚JVM的内存模型JVM内存模型.png其中本地方法栈就是运行时调用native 方法的数据保存区。本地方法栈的大小可以设置成固定的或者是动态扩展。Java中的内存泄露JAVA 编程中的内存泄漏&…

04 linux用户群组和权限

作业一: 1)新建用户natasha,uid为1000,gid为555,备注信息为“master” 2)修改natasha用户的家目录为/Natasha 3)查看用户信息配置文件的最后一行 4)为natasha用户设置密码“123” 5)查看用户密码配置文件的最后一行 6)将natasha用…

基于 CoreText 实现的高性能 UITableView

引起UITableView卡顿比较常见的原因有cell的层级过多、cell中有触发离屏渲染的代码(譬如:cornerRadius、maskToBounds 同时使用)、像素是否对齐、是否使用UITableView自动计算cell高度的方法等。本文将从cell层级出发,以一个仿朋友…

Web Magic 总体架构

1.2 总体架构 WebMagic的结构分为Downloader、PageProcessor、Scheduler、Pipeline四大组件,并由Spider将它们彼此组织起来。这四大组件对应爬虫生命周期中的下载、处理、管理和持久化等功能。WebMagic的设计参考了Scapy,但是实现方式更Java化一些。 而S…

SpringMVC搭建+实例

想做一点自己喜欢的东西,研究了一下springMVC,所以就自己搭建一个小demo,可供大家吐槽。 先建一个WEB工程,这个相信大家都会,这里不在多说。去网上下载spring jar包,然后在WEB-INF下新建一个lib文件,将下载的jar包放进…

php8更新,PHP 8 中新特性以及重大调整

PHP 8,PHP 的一个新的大版本,预计将于2020年12月3日发布,这意味着将不会有 PHP 7.5 版本。PHP8目前正处于非常活跃的开发阶段,所以在接下来的几个月里,情况可能会发生很大的变化。在这篇文章中,我会维持一个…

Javascript学习之函数(function)

http://www.cnblogs.com/royalroads/p/4418587.html 在JS中,Function(函数)类型实际上是对象;每个函数都是Function类型的实例,而且都与其他引用类型一样具有属性和方法。由于函数是对象,因此函数名实际上也是一个指向函数对象的指针。 一 函数的声明方式 //1.函数声…

用于Spring JPA2后端的REST CXF

在本演示中,我们将使用spring / jpa2后端生成一个REST / CXF应用程序。 该演示演示了分钟项目的轨迹REST-CXF 。 演示2中的模型保持不变。 浓缩保持不变。 但是轨道改变了 添加的是2层: 在JPA2之上具有弹簧集成的DAO层 具有JAX-RS批注的REST-CXF层…

完整的WebApplication JSF EJB JPA JAAS –第1部分

这篇文章将是迄今为止我博客中最大的一篇文章! 我们将看到完整的Web应用程序。 最新的技术将完成此工作(直到今天),但是我将给出一些提示以显示如何使本文适用于较旧的技术。 在本文的结尾,您将找到要下载的源代码。 您…