V-rep机器人仿真软件学习笔记

常用的机器人仿真软件有哪些?为什么选择V-rep?
目前常用的机器人物理仿真软件有Gazebo、V-rep、Webots等,这三款都是开源软件,自己使用过前两种,Gazebo配合ROS使用功能十分强大,但是要在Linux系统下使用,会有些不方便;V-rep是一个跨平台的仿真软件,在windows和Linux下都可以很稳定的运行,而且软件体积较小。考虑到要在windows下使用仿真软件,以及V-rep方便的Python API接口和易用性,最终选择使用V-rep。
V-rep在自己项目的应用情况?
自己使用V-rep主要用来进行机械臂规划和控制方面算法的验证和仿真实验,主要用于以下两个方面:
利用V-rep的物理仿真环境,建立类似OpenAI Gym的交互环境,进行基于强化学习的机械臂规划和控制方法方面的仿真实验。
利用V-rep进行双臂机器人协作规划和控制算法研究
V-rep中内置了很多商用机械臂模型。如果使用Gazebo,可以搭配ROS Moveit!进行机械臂相关的仿真和实物规划与控制。

一、V-rep 基本知识
1.Tutorials: Building a clean model tutorial
1.1 Building the visible shapes
建议使用的CAD模型包含的triangles少于20000个,这样有利于快速计算
导入CAD模型后使用V-rep进一步简化
Automatic mesh division: [Menu bar --> Edit --> Grouping/Merging --> Divide selected shapes] 和[Menu bar --> Edit -> Grouping/Merging --> Merge selected shapes]
Extract the convex hull: [Menu bar --> Edit --> Morph selection into convex shapes]
Decimate the mesh: [Menu bar --> Edit --> Decimate selected shape...]
Remove the inside of the mesh: [Menu bar --> Edit --> Extract inside of selected shape]
可以再shape属性中查看模型的triangle数
将一个整体模型按照连接关系(关节)分割:
Automatic mesh division: [Menu bar --> Edit --> Grouping/merging --> Divide selected shapes].
Manual mesh division: 通过 triangle edit mode分割
1.2 Building the joints
使用Models/tools/Denavit-Hartenberg joint creator.ttm通过D-H参数创建关节
1.3 Building the dynamic shapes
1.4 Model definition
2.关于V-rep内置Lua脚本的学习

Lua中的脚本包括:

main script
child script
Tips:v-rep是默认打开19997端口

2.1 main script
main script主脚本相当于主函数,有且只有一个。主脚本由以下几个函数组成:

初始化函数sysCall_init:在仿真开始阶段执行一次
执行函数sysCall_actuation:在每个仿真回合都会执行,用于控制所有仿真对象的执行函数。有三个函数比较重要:
sim.launchThreadedChildScripts
sim.resumeThreads
sim.handleChildScripts
传感器函数sysCall_sensing:在每个仿真回合都会执行,用于控制所有传感器函数。有三个函数比较重要:
sim.resumeThreads
sim.handleChildScripts
恢复函数sysCall_cleanup:仅在仿真结束时执行一次,用于将仿真环境恢复为仿真开始前的状态。
2.2 child scripts
child script数量不限,与仿真场景中的物体相关联,被main scripts调用。child script与场景中物体的关联有以下优势:很好的可移植性和继承性、不同版本模型间没有冲突、和仿真循环很好的进行同步。

child script有两种类型 non-threaded child scripts(非线程子脚本) 和threaded child scripts(线程子脚本):


2.2.1 non-threaded child scripts
non-threaded child scripts包含一组阻塞函数,即每次被调用时将执行一些任务,然后返回控制权,如果没有返回则整个仿真一直停滞。每个仿真间隔中non-threaded child scripts将被main script调用两次:main script’s actuation and sensing 。
Tip:优先选择non-threaded child scripts来写子脚本。

non-threaded child scripts应当被分为4个函数来写:

初始化函数sysCall_init:仅在child script被调用时执行一次
执行函数sysCall_actuation:每个仿真循环都被调用
传感器函数sysCall_sensing:每个仿真循环都被调用
恢复函数sysCall_cleanup:仅在仿真结束被调用一次
2.2.2 threaded child scripts
threaded child scripts将会启用一个线程,线程的启用和恢复由主脚本的sim.launchThreadedChildScripts和sim.resumeThreads函数控制。当线程子脚本正在执行时,不会二次启动它。
Tips:threaded child scripts存在耗时的问题

threaded child scripts包括两个函数部分:

线程主部分函数sysCall_threadmain
恢复函数sysCall_cleanup
可以通过以下三个函数控制threaded child scripts的执行: sim.setThreadSwitchTiming、sim.setThreadAutomaticSwitch、sim.switchThread

2.3 Script simulation parameters
脚本仿真参数,在脚本中可以通过以下函数读取和设置仿真参数:

sim.getScriptSimulationParameter
sim.setScriptSimulationParameter
2.4 Customization scripts
自定义脚本主要用于对仿真场景的拓展,比如控制仿真地面的大小:


2.5 Callback functions
回调函数包括以下三种;

Dynamics callback functions:动态回调函数,通常被non-threaded child script或customization script使用。
Joint callback functions:关节回调函数,通常用于编写低级的控制程序(PID控制),每个仿真间隔将被调用10次以上。
The contact callback function:用于物理引擎检测障碍物,满足某个条件时触发contact callback function。
2.5 Script dialog
可以通过脚本对话框Script dialog:[Menu bar --> Tools --> Scripts]自行添加脚本文件。

2.6 Script execution order
脚本的执行顺序如下:

threaded child scripts are launched/resumed (order can be adjusted via sim.setThreadResumeLocation)
non-threaded child scripts are called
customization scripts are called
add-on scripts are called

3.遇到的Vrep中Python Api函数–>属于client端的远程API函数
simxCallScriptFunction: 调用Vrep中Lua脚本中的函数
simxStart:其中timeOutInMs参数为正数时,表示初次连接等待时间,而函数执行的阻塞时间为默认5000ms;当该参数为负数时,正数部分为函数执行的阻塞时间,而初次连接的等待时间为5000ms。
simxStartSimulation:启动仿真。vrep端可以不用提前启动,而通过该函数来启动仿真。
simxGetObjectHandle:获取物体句柄函数,物体名字后需要添加“#”,因为vrep默认的取名风格objectName#x。
4.遇到的Vrep中Lua Api函数–>属于server端的常规API函数,在Regular API目录下查找API文档
单位使用说明:所有API的单位都是米、千克、秒和弧度,或者它们的组合(除非另有明确说明)。用户界面的单位是米、千克、秒和度。

一般函数:

sim.getObjectName:获取物体名称,得到的名称带有下标(e.g. “myRobot#42”)
sim.getNameSuffix:从带有下标的物体名称中取出下标和名称前部分
sim.getObjectHandle:获取物体句柄
sim.getJointPosition:获取关节位置
sim.displayDialog:显示消息对话框
sim.getObjectMatrix:获取物体的位姿矩阵,以table形式返回,表中元素数为12(类似作用的函数有:sim.getObjectOrientation,sim.getObjectPosition)
sim.setObjectMatrix: 设置物体的变换矩阵
sim.getCollectionHandle:根据集合句柄的名称获取集合句柄
sim.getObjectSpecialProperty:获取场景对象的特殊属性,:collidable、measurable.etc
sim.boolOr32:在两个数字之间执行32位布尔或运算
sim.addDrawingObject:添加将在场景中显示的绘图对象
sim.setJointPosition:设置关节的固有位置
sim.addDrawingObjectItem:向之前的绘图对象插入新的属性,绘图主函数
getObjectAssociatedWithScript:获取脚本所对应的对象的句柄
sim.getSimulationState:获取当前的仿真状态
sim.setThreadAutomaticSwitch:是否允许控制线程进行自动切换
sim.switchThread:允许指定当前线程切换到另一个线程的确切时刻
sim.clearStringSignal:清除字符串信号
路径规划相关函数:

sim.getIkGroupHandle:获取逆运动学解算IK组的句柄
sim.getConfigForTipPose:随机搜索与给定末端执行器位置/方向匹配的机械手关节配置,若存在则返回一组关节角。通过返回值判断目标位置对应的末端执行器是否存在且是否产生碰撞。
sim.generateIkPath:生成一条直线(即笛卡尔空间中的最短路径)驱动机器人从当前配置到目标配置的路径。
sim.setStringSignal: 设置字符串信号的值
sim.getStringSignal:获取字符串信号的值
sim.packFloatTable:将浮点数表打包成字符串
常见的常量:

sim.objectspecialproperty_collidable:物体可碰撞属性
OMPL API:
目录:V-REP API framework–>Other Interfaces–>OMPL Plugin API reference或者More details here.

使用OMPL进行运动规划的一般步骤
创建任务–>选定规划算法–>设置状态空间–>设置碰撞对–>设置开始和目标状态–>计算出一条或多条路径–>销毁创建的任务

常见函数

simOMPL.createTask:创建任务
simOMPL.setAlgorithm:选定规划算法
simOMPL.createStateSpace:为每个关节创建状态空间
simOMPL.setStateSpace:设置整个任务的状态空间
simOMPL.setCollisionPairs:设置碰撞对
simOMPL.setStartState/simOMPL.setGoalState:设置起始和目标状态
simOMPL.addGoalState:添加目标状态,但不清除以前设置的目标状态(如果有)。
simOMPL.setStateValidityCheckingResolution:设置状态有效性检验的分辨率,表示为状态空间范围的分数
simOMPL.compute:使用OMPL找到此运动规划任务的解决方案。
OMPL运动规划库中常见的规划器介绍
规划器非常多,根据不同的用途和要求进行选择。中文介绍参考网站:https://zhenshenglee.github.io/2016/09/03/160903-ompl1/

5.Vrep中子脚本线程执行
Vrep中thread script的执行是以协程模拟多线程的方式进行的,即执行多线程过程分为很多个时间片,自动在不同的线程中切换,实现多线程的效果。所以在实际使用thread script时,要合理控制线程切换,使程序执行时间合理。
在目录:Writing code in and around V-REP–>Embedded scripts–>Simulation scripts–>The main and child scripts–>Child scripts
参考博客,线程切换控制的常用函数:

sim.setThreadAutomaticSwitch()
sim.switchthread()
sim.setThreadSwitchTiming
示例代码,路径规划中机械臂响应位置控制指令的thread script:

-- This threaded script is only used for executing some precalculated motion on this robot
function sysCall_threadmain()
    -- Initialization:
    jh={-1,-1,-1,-1,-1,-1}
    for i=1,6,1 do
        jh[i]=sim.getObjectHandle('IRB4600_joint'..i)
    end
    model=sim.getObjectAssociatedWithScript(sim.handle_self)
    modelName=sim.getObjectName(model)

    -- Main function:
    while sim.getSimulationState()~=sim.simulation_advancing_abouttostop do
        local path=sim.getStringSignal(modelName..'runPath')
        if path and #path>0 then
            path=sim.unpackFloatTable(path)
            sim.setThreadAutomaticSwitch(false)  --the thread switching is temporarily disabled
            for i=1,#path/6,1 do
                for j=1,6,1 do
                    sim.setJointPosition(jh[j],path[(i-1)*6+j])
                end
                sim.switchThread()
            end
            sim.setThreadAutomaticSwitch(false)
            sim.clearStringSignal(modelName..'runPath')
        end
        sim.switchThread()
    end
end

6.Vrep中路径规划的两种情况
具体程序参考Vrep的Python程序例程:pathPlanningTest.py

目标点由笛卡尔空间的位姿点给出,此时的思路是:
位姿点–>逆解算得到配置空间对应的多个关节坐标(满足一定的精度即可,space metric作为判定条件。然后根据起始点和目标点坐标的配置空间距离进行排序),并验证关节坐标的无碰撞和可行性–>在配置空间进行路径规划
目标点由配置空间的关节坐标直接给出,此时的思路是:
在配置空间直接进行路径规划
路径规划要考虑的四要素:

机械臂运动学
关节限位
机械臂防自碰撞
机械臂与障碍物间的防碰撞
Tips:

早期Vrep内建路径规划功能(不建议使用)
7.关于Vrep自带模型的D-H参数获取的办法
使用vrep在tool目录下提供的D-H参数提取工具DH_extractor进行获取
根据机械臂的名称去官网下载其CAD模型,利用CAD软件进行获取
8.远程API工作方式分析(官方文档)
API函数操作模式:

simx_opmode_oneshot
simx_opmode_oneshot_wait
simx_opmode_streaming
simx_opmode_oneshot_split
simx_opmode_discontinue
simx_opmode_buffer
simx_opmode_remove
阻塞式和非阻塞式
一个重要例子:通过远程API控制多个关节同时转动,可以通过先暂停通信线程,发送完指令后再恢复通信线程实现。暂停通信线程时,远程客户端不会与远程服务器之间进行通信。示例代码:

# 控制命令需要同时方式,故暂停通信,用于存储所有控制命令一起发送
vrep.simxPauseCommunication(clientID, True)
for i in range(jointNum):
   vrep.simxSetJointTargetPosition(clientID, jointHandle[i], 120/RAD2DEG, vrep.simx_opmode_oneshot)
vrep.simxPauseCommunication(clientID, False)

数据流的形式
实现的效果是定时从某个句柄中获取需要的数据,通过simx_opmode_streaming和simx_opmode_buffer实现,示例代码:

// Streaming operation request (subscription) (function returns immediately (non-blocking)):
simxGetJointPosition(clientID,jointHandle,&jointPosition,simx_opmode_streaming);
// The control loop:
 while (simxGetConnectionId(clientID)!=-1) // while we are connected to the server..
{ 
    // Fetch the newest joint value from the inbox (func. returns immediately (non-blocking):
    if (simxGetJointPosition(clientID,jointHandle,&jointPosition,simx_opmode_buffer==simx_return_ok) 
    { 
        // here we have the newest joint position in variable jointPosition!    
    }
    else
    {
        // once you have enabled data streaming, it will take a few ms until the first value has arrived. So if
        // we landed in this code section, this does not always mean we have an error!!!
    }
}

// Streaming operation is enabled/disabled individually for each command and
// object(s) the command applies to. In above case, only the joint position of
// the joint with handle jointHandle will be streamed.

同步操作
。。。

9.Vrep中关节不同的属性
参考文档:Entities–>Scene objects–>Joint–>Joint types and operation
不同的关节属性对应不同要的仿真要求,比如位置控制、动力学仿真、关节力矩控制等。

根据关节的属性使用不同函数设置关节的位置、速度:

simSetJointPosition() (when your joint is not in force/torque mode 比如说inverse kinema模式)
simSetJointTargetPosition() (when your joint is in force/torque mode, its motor enabled and its control loop also enabled)
simSetJointTargetVelocity (when your joint is in force/torque mode, its motor enabled, and its control loop NOT enabled)
10.Vrep中仿真过程不同状态的介绍
仿真状态分为:停止、开始、暂停三个状态。在仿真内部还有一些内部状态用来表示下一时刻的状态,使程序更好地控制仿真过程,状态切换图如下:

获取状态的函数为:sim.getSimulationState,返回的状态即为上图中的状态。

11.参考网站
https://blog.csdn.net/u013528298/article/category/7519401
二、V-rep进阶知识点
1. vrep使用b0RemoteApi工作模式
与之前版本的Remote API相比,新版本的Vrep推荐b0RemoteApi,官方说这种方式更灵活好用。

运行C:\Program Files\V-REP3\V-REP_PRO_EDU\programming\b0RemoteApiBindings\python目录下的simpleTest.py程序,遇到如下错误:

OSError: [WinError 193] %1 不是有效的 Win32 应用程序。

解决过程:

开始参考如下网站使用anaconda重建一个32位的python环境:
https://blog.csdn.net/vample/article/details/88877745 ,最终发现并不是这个原因导致的,因为提供的DLL文件都是64位的,所以不是因为python版本不匹配的原因报错;
最终google之后发现解决办法:https://stackoverflow.com/questions/19849077/error-loading-dll-in-python-not-a-valid-win32-application
是因为之前添加的DLL不够全,将C:\Program Files\V-REP3\V-REP_PRO_EDU目录下所有DLL文件添加至\programming\b0RemoteApiBindings\python目录下,然后即可正常simpleTest.py程序。
常用conda命令:

# 设置32位set CONDA_FORCE_32BIT=1
# 切回系统默认set CONDA_FORCE_32BIT=
# 查看当前环境详细信息 conda info
# 查看当前环境信息 conda info --envs
# 切换环境 conda activate env_name,conda deactivate env_name
# 移除环境 conda remove -n env_name --all
# 安装包:activate env_name,conda install pandas

1.0 Vrep b0RemoteApi工作模式解析
类似ROS中的松耦合通讯机制。
与vrep regular API相比,B0-based remote API函数需要一个额外的参数:用于执行函数调用的话题(Topic)或者通讯频道(communication channel)参数。Topic可以从以下5个函数的返回值获取:

simxServiceCall: 用于阻塞模式,即client发送命令到server端执行后,等待server返回一个响应给client。通常用于只需要server端执行一次的命令,比如simxGetObjectHandle
simxDefaultPublisher: 用于非阻塞模式,client端向server端发送命令后不等待server端的回应。比如simxSetJointPosition
simxDefaultSubscriber: 该方式让server端持续执行某函数,并且将函数的响应结果持续返回至client端,client端在回调函数中收到返回的数据流。通常用于需要在server端持续执行的命令,比如simxGetJointForce。此外,传入的回调函数将通过执行simxSpinOnce函数进行调用。
simxCreatePublisher: 与simxDefaultPublisher类似,不同的是创建一个专用的publisher Topic,即创建一个专用的发布数据的频道。主要用于特定的命令或者函数,比如数据量大的函数simxSetVisionSensorImage。
simxCreateSubscriber: 与simxDefaultSubscriber类似,不同的是创建一个专用的subscriber Topic,即创建一个专用的订阅数据的频道。主要用于特定的命令或者函数,比如数据量大的函数simxGetVisionSensorImage。
原文如下:


默认情况下,B0-based remote API的Client和Server之间是异步工作的。可以通过设置使Client单独触发每个仿真时间片,达到同步工作模式。以下是Python形式Client端同步工作模式的示例代码:

import b0RemoteApi
import time

with b0RemoteApi.RemoteApiClient('b0RemoteApi_pythonClient','b0RemoteApi') as client:    
    doNextStep=True

    def simulationStepStarted(msg):
        simTime=msg[1][b'simulationTime'];
        print('Simulation step started. Simulation time: ',simTime)
        
    def simulationStepDone(msg):
        simTime=msg[1][b'simulationTime'];
        print('Simulation step done. Simulation time: ',simTime);
        global doNextStep
        doNextStep=True
        
    client.simxSynchronous(True)
    client.simxGetSimulationStepStarted(client.simxDefaultSubscriber(simulationStepStarted));
    client.simxGetSimulationStepDone(client.simxDefaultSubscriber(simulationStepDone));
    client.simxStartSimulation(client.simxDefaultPublisher())
    
    startTime=time.time()
    while time.time()<startTime+5: 
        if doNextStep:
            doNextStep=False
            client.simxSynchronousTrigger()
        client.simxSpinOnce()
    client.simxStopSimulation(client.simxDefaultPublisher())

在Server端添加b0RemoteApiServer,参考官方文档:Writing code in and around V-REP–>V-REP API framework–>Remote API–>BØ-based remote API–>Enabling the B0-based remote API - server side
直接通过File–>Load Model将Models/tools/B0 remote Api server.ttm添加至当前场景即可。

 

1.1 vrep b0RemoteApi工作模式下API之间的关系:
Client端:使用B0-based remote API:b0RemoteApi.py,实现方式是直接将server端b0 API中的函数名进行再次封装,然后发送至server端。
Server端:使用server端b0 API,API文件为C:\Program Files\V-REP3\V-REP_PRO_EDU\lua\b0RemoteApiServer.lua,server端b0 API文件b0RemoteApiServer.lua内部实现主要是将regular API进行封装。
2. vrep中吸盘工具suction pad的使用
参考以下资料:

场景C:\Program Files\V-REP3\V-REP_PRO_EDU\scenes\reflexxesMotionLibraryType2Demo.ttt中有吸盘工具的使用
模型C:\Program Files\V-REP3\V-REP_PRO_EDU\models\robots\non-mobile\MTB robot.ttm中有吸盘工具的使用
http://www.forum.coppeliarobotics.com/viewtopic.php?f=9&t=5317&p=21341&hilit=vrep+suction+pad#p21341
添加吸盘工具至机械臂末端,根据吸盘的子程序:

function sysCall_init() 
    s=sim.getObjectHandle('suctionPadSensor')
    l=sim.getObjectHandle('suctionPadLoopClosureDummy1')
    l2=sim.getObjectHandle('suctionPadLoopClosureDummy2')
    b=sim.getObjectHandle('suctionPad')
    suctionPadLink=sim.getObjectHandle('suctionPadLink')

    infiniteStrength=sim.getScriptSimulationParameter(sim.handle_self,'infiniteStrength')
    maxPullForce=sim.getScriptSimulationParameter(sim.handle_self,'maxPullForce')
    maxShearForce=sim.getScriptSimulationParameter(sim.handle_self,'maxShearForce')
    maxPeelTorque=sim.getScriptSimulationParameter(sim.handle_self,'maxPeelTorque')

    sim.setLinkDummy(l,-1)
    sim.setObjectParent(l,b,true)
    m=sim.getObjectMatrix(l2,-1)
    sim.setObjectMatrix(l,-1,m)
end

function sysCall_cleanup() 
    sim.setLinkDummy(l,-1)
    sim.setObjectParent(l,b,true)
    m=sim.getObjectMatrix(l2,-1)
    sim.setObjectMatrix(l,-1,m)
end 

function sysCall_sensing() 
    parent=sim.getObjectParent(l)
    if (sim.getScriptSimulationParameter(sim.handle_self,'active')==false) then
        if (parent~=b) then
            sim.setLinkDummy(l,-1)
            sim.setObjectParent(l,b,true)
            m=sim.getObjectMatrix(l2,-1)
            sim.setObjectMatrix(l,-1,m)
        end
    else
        if (parent==b) then
            -- Here we want to detect a respondable shape, and then connect to it with a force sensor (via a loop closure dummy dummy link)
            -- However most respondable shapes are set to "non-detectable", so "sim.readProximitySensor" or similar will not work.
            -- But "sim.checkProximitySensor" or similar will work (they don't check the "detectable" flags), but we have to go through all shape objects!
            index=0
            while true do
                shape=sim.getObjects(index,sim.object_shape_type)
                if (shape==-1) then
                    break
                end
                if (shape~=b) and (sim.getObjectInt32Parameter(shape,sim.shapeintparam_respondable)~=0) and (sim.checkProximitySensor(s,shape)==1) then
                    -- Ok, we found a respondable shape that was detected
                    -- We connect to that shape:
                    -- Make sure the two dummies are initially coincident:
                    sim.setObjectParent(l,b,true)
                    m=sim.getObjectMatrix(l2,-1)
                    sim.setObjectMatrix(l,-1,m)
                    -- Do the connection:
                    sim.setObjectParent(l,shape,true)
                    sim.setLinkDummy(l,l2)
                    break
                end
                index=index+1
            end
        else
            -- Here we have an object attached
            if (infiniteStrength==false) then
                -- We might have to conditionally beak it apart!
                result,force,torque=sim.readForceSensor(suctionPadLink) -- Here we read the median value out of 5 values (check the force sensor prop. dialog)
                if (result>0) then
                    breakIt=false
                    if (force[3]>maxPullForce) then breakIt=true end
                    sf=math.sqrt(force[1]*force[1]+force[2]*force[2])
                    if (sf>maxShearForce) then breakIt=true end
                    if (torque[1]>maxPeelTorque) then breakIt=true end
                    if (torque[2]>maxPeelTorque) then breakIt=true end
                    if (breakIt) then
                        -- We break the link:
                        sim.setLinkDummy(l,-1)
                        sim.setObjectParent(l,b,true)
                        m=sim.getObjectMatrix(l2,-1)
                        sim.setObjectMatrix(l,-1,m)
                    end
                end
            end
        end
    end
end 

可以发现吸盘子程序已经很完善,会根据吸盘末端和物体之间的距离来判断是否吸取物体,同时吸盘子程序设置了以下参数:active,infiniteStrength,maxPullForce,maxShearForce,maxPeelTorque,前两个分别表示吸盘是否激活、吸盘吸力是否无穷大,默认吸盘激活且吸力无穷大,参数可以通过吸盘子程序的参数服务器修改:

我们需要控制的就是移动机械臂,使末端接近被吸物体,然后吸盘就会自动吸取物体。需要将被吸物体释放时,需要修改吸盘子程序并添加自定义程序,一种思路是根据某个条件设置active参数为false,然后释放被吸物体即可,参考模型C:\Program Files\V-REP3\V-REP_PRO_EDU\models\robots\non-mobile\MTB robot.ttm中吸盘工具的使用。

3. vrep中力/力矩传感器(force sensor)的使用
添加force sensor:add->force sensor,然后使用力传感器刚性地连接两个物体后就可以测量力和力矩。
具体过程:以C:\Program Files\V-REP3\V-REP_PRO_EDU\models\robots\non-mobile\MTB robot.ttm为例。通过File->Load Model载入MTB robot.ttm,可以看到机械臂末端连接了吸盘:

接下来的目标是:在机械臂末端和吸盘之间添加力传感器,测量被吸物体对机械臂末端施加的重力,步骤如下:

通过add->force sensor添加force sensor
使用Assemble/Disassemble工具断开MTB_link3Respondable和suctionPad的连接
调整Force_sensor的位置到MTB_link3Respondable和suctionPad之间
使用Assemble/Disassemble工具连接MTB_link3Respondable和Force_sensor
使用Assemble/Disassemble工具连接Force_sensor和suctionPadBodyRespondable。因为Force_sensor连接两个刚性物体,所以要将Force_sensor和suctionPadBodyRespondable相连,而不是和suctionPad相连。
完成以上步骤后的模型关系图为:

之后再参考博客添加一个Graph,并将Force_sensor的Z轴方向的力作为显示数据添加至Graph,运行MTB robot.ttm就可以看到物体的重力。
Tips: 吸盘工具自带了力传感器,suctionPad就是一个力传感器。

参考资料:

官方文档:Entities->Scene objects->Force sensors
测量值不稳定的解决:http://www.forum.coppeliarobotics.com/viewtopic.php?f=9&t=6614
相关博客:https://blog.csdn.net/weixin_34375054/article/details/85882691
4. Vrep中不同物理仿真引擎的使用
Bullet和ODE主要用于游戏中,而Vortex专注于科学物理仿真,计算准确度以及稳定性肯定会比游戏物理引擎要好。所以VREP提供了4个物理引擎还是有道理的,要根据它们的长处来选择合适的引擎进行仿真,不能总是用默认的Bullet。
经验分享: 关于物理引擎 “Bullet, ODE, 和 Newton” (Vortex 自动忽略)使用经验,对于复杂系统(32个自由度),如果涉及到 V-rep 和 MATLAB, Python, 或 ROS 某一个或者某几个之间的通讯问题,ODE 最佳,Bullet 可用,Newton 不推荐。
不涉及通讯,直接 LUA,ODE 依然最佳,Newton 相差无几,甚至会比 ODE 更稳定,Bullet 也可用。

V-REP与Vortex Studio的安装,需要申请License,暂且没有操作做成功。

5. 使用Vrep控制实物机器人的思路
http://www.forum.coppeliarobotics.com/viewtopic.php?t=5780

6. 使用Reflexxes Motion Library(在线运动轨迹生成库)进行轨迹规划
Reflexxes Motion Library是在线运动轨迹生成库,在其官网有关于功能方面详细的介绍。在完成实时轨迹规划时,算法的输入输出如下图:

基于位置的在线轨迹规划详细输入输出如下图:

输入:当前的运动状态(位置、姿态、速度、加速度)、目标运动状态、运动约束(最大速度、最大加速度、最大加加速度)
输出:新的运动状态和轨迹(位姿、速度、加速度)
更多的Reflexxes Motion Library轨迹规划知识参考官方文档,官方文档详细介绍了Reflexxes Motion Library各轴同步行为(不同步、时间同步、相位同步)、API结构介绍(应用接口层、算法层、数学计算层)等方面的实现。
Vrep集成了Reflexxes Motion Library,具体API在Regular API function-->Reflexxes Motion Library type II or IV:

 在空白场景中拖入UR5型机械臂,打开机械臂对应的子脚本程序,可以看到Reflexxes Motion Library API的使用:

-- This is a threaded script, and is just an example!

function sysCall_threadmain()
    jointHandles={-1,-1,-1,-1,-1,-1}
    for i=1,6,1 do
        jointHandles[i]=sim.getObjectHandle('UR5_joint'..i)
    end

    -- Set-up some of the RML vectors:
    vel=180
    accel=40
    jerk=80
    currentVel={0,0,0,0,0,0,0}
    currentAccel={0,0,0,0,0,0,0}
    maxVel={vel*math.pi/180,vel*math.pi/180,vel*math.pi/180,vel*math.pi/180,vel*math.pi/180,vel*math.pi/180}
    maxAccel={accel*math.pi/180,accel*math.pi/180,accel*math.pi/180,accel*math.pi/180,accel*math.pi/180,accel*math.pi/180}
    maxJerk={jerk*math.pi/180,jerk*math.pi/180,jerk*math.pi/180,jerk*math.pi/180,jerk*math.pi/180,jerk*math.pi/180}
    targetVel={0,0,0,0,0,0}

    targetPos1={90*math.pi/180,90*math.pi/180,-90*math.pi/180,90*math.pi/180,90*math.pi/180,90*math.pi/180}
    sim.rmlMoveToJointPositions(jointHandles,-1,currentVel,currentAccel,maxVel,maxAccel,maxJerk,targetPos1,targetVel)
    sim.wait(2)
    targetPos2={-90*math.pi/180,45*math.pi/180,90*math.pi/180,135*math.pi/180,90*math.pi/180,90*math.pi/180}
    sim.rmlMoveToJointPositions(jointHandles,-1,currentVel,currentAccel,maxVel,maxAccel,maxJerk,targetPos2,targetVel)

    targetPos3={0,0,0,0,0,0}
    sim.rmlMoveToJointPositions(jointHandles,-1,currentVel,currentAccel,maxVel,maxAccel,maxJerk,targetPos3,targetVel)
end

sim.rmlMoveToJointPositions()和sim.rmlMoveToPosition()分别在关节空间和迪卡文空间进行轨迹规划,这两个函数均为阻塞型函数。**使用sim.rmlMoveToPosition()并配合逆运动学可以实现机器人的路径控制,**具体操作可以参考该博客内容。

参考资料:

Reflexxes Motion Library的使用:https://www.cnblogs.com/21207-iHome/p/6344467.html
Reflexxes Motion Library API结构介绍:https://www.cnblogs.com/21207-iHome/p/7724694.html
Reflexxes Motion Library官方API结构介绍:http://www.reflexxes.ws/software/typeiirml/v1.2.6/docs/page__source_code.html
7.vrep不同类型的刚体表示,作用不同


Simple random shape和Compound random shape主要用来显示
Simple convex shape和Compound convex shape主要用来仿真计算

三、V-rep在机械臂仿真中遇到的一些问题
1.vrep机械臂散架问题
添加kuka LBR4p机械臂,所有关节原始工作模式是Torque or force mode,将所有关节改为Inverse kinematics mode后运行仿真会出现机械臂散架的问题,这个时候问题解决。

在动力学仿真设计章节有讲,上述问题的主要原因是机械臂各个关节设置为动态模式后,关节必须处于Torque or force mode或者hybrid fashion,否则就会由于重力直接跌落,导致机械臂散架。

 

所以此时机械臂处于Inverse kinematics mode+Torque or force mode的模式,即使配置好ik target-tip,因为关节之间存在力矩,所以在移动target时机械臂无法完成逆解运动。

2.Vrep逆运动学解算,使用sim.getConfigForTipPose函数输出结果为nil的问题
问题描述:
sim.getConfigForTipPose函数的功能是:给定逆运动学解算器、关节句柄、碰撞对等参数,可以根据ik_target在笛卡尔空间的位姿逆解算出关节空间的坐标,运行一次得到一个关节坐标或nil(如果逆解不存在)。使用sim.getConfigForTipPose得到目标点多个对应的关节坐标,为下一步使用OMPL进行路径规划奠定基础。
当前遇到的问题是: sim.getConfigForTipPose函数一直输出nil值,没有得到有效的关节坐标。
场景如下:

 将kuka LBR4p机械臂安装在Onmirob移动底盘上。对应的逆解对LBR4p_ik_targetLBR4p_ik_tip,以及逆解算组已经设置好,重写LBR4p原始脚本,进行sim.getConfigForTipPose的测试:

-- This is a threaded script, and is just an example!

ikTest=function()
-- Initialization phase:
    jh={-1,-1,-1,-1,-1,-1,-1}
    for i=1,7,1 do
        jh[i]=sim.getObjectHandle('LBR4p_joint'..i)
    end
    ikGroup=sim.getIkGroupHandle('LBR4p')
    ikTarget=sim.getObjectHandle('LBR4p_ik_target')
    collisionPairs={sim.getCollectionHandle('LBR4PP'),sim.getCollectionHandle('not_LBR4PP')}
    target1=sim.getObjectHandle('testTarget3')
    --approachDirectionObstacle=sim.getObjectHandle('approachDirectionObstacle')
    metric={0.5,1,1,0.5,0.1,0.2,0.1}
    forbidLevel=0
    ikShift=0.1
    ikSteps=50

--------------------------
------for ik test---------
    local m=sim.getObjectMatrix(target1,-1)
    sim.setObjectMatrix(ikTarget,-1,m)
    local c=sim.getConfigForTipPose(ikGroup,jh,0.65,10,nil,collisionPairs)
    print(c)
    while true do
        local m=sim.getObjectMatrix(target1,-1)
        sim.setObjectMatrix(ikTarget,-1,m)
        local c=sim.getConfigForTipPose(ikGroup,jh,0.65,10,nil,collisionPairs)
        print(c)
        if c then
            local gc = c
            sim.rmlMoveToJointPositions(jointHandles,-1,currentVel,currentAccel,maxVel,maxAccel,maxJerk,gc,targetVel)
        end
    end
------for ik test---------
--------------------------
end


function sysCall_threadmain()
    jointHandles={-1,-1,-1,-1,-1,-1,-1}
    for i=1,7,1 do
        jointHandles[i]=sim.getObjectHandle('LBR4p_joint'..i)
    end

    -- Set-up some of the RML vectors:
    vel=110
    accel=40
    jerk=80
    currentVel={0,0,0,0,0,0,0}
    currentAccel={0,0,0,0,0,0,0}
    maxVel={vel*math.pi/180,vel*math.pi/180,vel*math.pi/180,vel*math.pi/180,vel*math.pi/180,vel*math.pi/180,vel*math.pi/180}
    maxAccel={accel*math.pi/180,accel*math.pi/180,accel*math.pi/180,accel*math.pi/180,accel*math.pi/180,accel*math.pi/180,accel*math.pi/180}
    maxJerk={jerk*math.pi/180,jerk*math.pi/180,jerk*math.pi/180,jerk*math.pi/180,jerk*math.pi/180,jerk*math.pi/180,jerk*math.pi/180}
    targetVel={0,0,0,0,0,0,0}

    targetPos1={90*math.pi/180,90*math.pi/180,170*math.pi/180,90*math.pi/180,90*math.pi/180,90*math.pi/180,0}
    --sim.rmlMoveToJointPositions(jointHandles,-1,currentVel,currentAccel,maxVel,maxAccel,maxJerk,targetPos1,targetVel)

    targetPos2={-90*math.pi/180,90*math.pi/180,180*math.pi/180,90*math.pi/180,90*math.pi/180,90*math.pi/180,0}
    --sim.rmlMoveToJointPositions(jointHandles,-1,currentVel,currentAccel,maxVel,maxAccel,maxJerk,targetPos2,targetVel)

    targetPos3={0,0,0,0,0,0,0}
    --sim.rmlMoveToJointPositions(jointHandles,-1,currentVel,currentAccel,maxVel,maxAccel,maxJerk,targetPos3,targetVel)
    
    ikTest()

end

问题原因
sim.getConfigForTipPose无法正常输出有效值的主要是因为该函数的collisionPairs参数没有正确设置的问题,开始时设置的碰撞对为kuka LBR4p机械臂本体和除开kuka LBR4p机械臂本体以外的物体:

 

对应图中Collections的LBR4p和not_LBR4p。这样设置的问题是因为LBR4p机械臂安装在移动底盘上,移动底盘和机械臂之间也会进行碰撞检测,因此导致sim.getConfigForTipPose无法正常输出有效值。

解决办法
将机械臂和移动底盘整体设为一个collection,除外的部分设为另一个collection,对应图中Collections的LBR4PP和not_LBR4PP。此时sim.getConfigForTipPose可以正常输出对应的关节坐标。

其他问题:
开启碰撞检测功能后,即上图Calculation Modules中的collision功能,机械臂显示为碰撞状态,这是因为将机械臂和移动底盘检测为发生碰撞,此时可以考虑将基座连杆部分的碰撞属性取消,然后碰撞状态消失:

 

3.Vrep b0通讯模式报错

  • 问题描述:
  • Traceback (most recent call last):
    File “b0_drl.py”, line 111, in
    client.simxDefaultPublisher())
    File “C:\Sam\Robotics\V-rep\Vrep_Python\python_b0\b0RemoteApi.py”, line 44, in exit
    self._handleFunction(‘Ping’,[0],self.simxDefaultSubscriber(self._pingCallback))
    File “C:\Sam\Robotics\V-rep\Vrep_Python\python_b0\b0RemoteApi.py”, line 118, in simxDefaultSubscriber
    self._handleFunction(‘setDefaultPublisherPubInterval’,[topic,publishInterval],channel)
    File “C:\Sam\Robotics\V-rep\Vrep_Python\python_b0\b0RemoteApi.py”, line 70, in _handleFunction
    rep = msgpack.unpackb(self._serviceClient.call(packedData))
    File “C:\Sam\Robotics\V-rep\Vrep_Python\python_b0\b0.py”, line 234, in call
    rep_bytes = bytearray(outarr.contents)
    ValueError: NULL pointer access
    2019-04-25 19:38:26 error: HB: Context was terminated

    在V-rep社区提问后得到问题原因和解决办法:

    问题原因是: client端simxCallScriptFunction调用server端的函数时,server端被调用函数是阻塞型的,在b0RemoteApi.py的第25行self._serviceClient.set_option(3,1000)参数设置server在1000ms不返回数据就会报上述错误
    所以解决办法是:将这一行代码里的参数进行修改,改为self._serviceClient.set_option(3,25000) #read timeout of 1000ms,然后就解决了上述问题。

4.Vrep动力学仿真的注意事项
官方文档关于动力学仿真实验设计:如果要使关节进行动力学仿真或者力传感器可以进行动力学仿真,必须要遵循一定的设计规则:

 

5.关于sim.setJointPosition和sim.setJointTargetPosition的使用
sim.setJointPosition函数不能用于关节处于torque/force mode的模式,如果关节处于torque/force mode模式则sim.setJointPosition无效。在关节处于torque/force mode的模式时,使用sim.setJointTargetPosition控制关节转动,此时可以设置转动过程的控制参数,包括PID参数和速度限制。

6.机械臂运动至目标点存在漂移的问题
可能的原因:

机械臂的关节工作在非torque/force mode模式,同时开启hybrid模式,此时可能由于重力原因机械臂发生漂移

 

thread script的主循环存在比较耗时的操作
7.vrep同步模式的基本知识
b0-based api:http://www.coppeliarobotics.com/helpFiles/en/b0RemoteApiOverview.htm
child script:http://www.coppeliarobotics.com/helpFiles/en/childScripts.htm
8.vrep机械臂末端抖动的问题
该问题主要原因是:

机械臂安装在移动底盘上,而移动底盘的车轮在仿真时有漂移现象发生,所以导致机械臂一直在抖动。
不同的物理仿真引擎也会导致这种情况出现,使用Bullet 2.78引擎时,抖动现象最弱。
9.vrep存在有的逆解路径无法到达的问题
可能是因为使用伪逆法求逆解得到的路径在奇异点附近运行困难。
但是换成数值解法之后无法得到有效解。
TODO:待解决上述问题。

10.初始仿真时物体属性设置的一些tricks

如上图的场景,想要初始状态是吸盘吸取物体。如果物体属性设置为“Body is dynamic”,仿真开始的一瞬间物体就会掉落下去,下图是物体的初始属性:

查看官方文档后,发现Start in sleep mode和Set to dynamic if gets parent这两个属性很有用,官方解释:


Start in sleep mode:该选项使能时,dynamic物体初始时处于不响应模式(不响应重力作用),直到与其他响应物体发生碰撞
Set to dynamic if gets parent:该模式使能时,只有当该物体设置连接至另一个物体是,才使能该物体为dynamic。。这个属性非常适合和吸盘配合,吸盘吸取物体时就是将被吸取物体设置为吸盘的子物体。
勾选上述两个设置后,仿真开始后物体就不会掉落,移动机械臂吸取物体后,物体将使能dynamic属性,然后就可以进行动力学仿真。

上述方法仍然可能存在如下问题:在吸盘接触物体的一瞬间物体变为"dynamic",吸盘没有吸取到物体时物体就掉落。
解决该问题的小trick:在物体下方添加一个小立方体来支撑物体,将其dynamic属性取消,并设置为透明,设置如下:

 

当吸盘成功吸取物体后移开该立方体,可以重设该立方体的位置即可:示例命令sim.setObjectPosition(190,-1,{0.6,0.6,0.5})

四、基于V-rep模仿OpenAI Gym搭建V-rep强化学习环境
基于V-rep模仿OpenAI Gym搭建V-rep强化学习环境,所实现的代码放置于代码仓库gym_vrep.使用强化学习方法主要来解决双臂机器人在板材安装过程中压紧过程的规划和控制问题。

1.遵照OpenGym环境的格式搭建V-rep强化学习环境
参考示例:

https://zhuanlan.zhihu.com/p/33553076
https://github.com/openai/gym-soccer
https://github.com/apoddar573/Tic-Tac-Toe-Gym_Environment
主要是自定义__init__(),step(),reset(),render(),close()等函数。

2.windows下tensorboard使用
出现tensorboard 无法访问此网站的问题,将tensorboard的网址改成:http://localhost:6006/即可正常访问。

参考:https://blog.csdn.net/wangqi_qiangku/article/details/79835801

3.加速Vrep仿真的方法
将仿真时间步调大。
参考:
http://www.forum.coppeliarobotics.com/viewtopic.php?f=9&t=2247
http://www.coppeliarobotics.com/helpFiles/en/simulation.htm
仿真设置界面:http://www.coppeliarobotics.com/helpFiles/en/simulationPropertiesDialog.htm
关闭图像引擎,使用命令行模式
http://www.forum.coppeliarobotics.com/viewtopic.php?t=7
http://www.forum.coppeliarobotics.com/viewtopic.php?t=6359
可以使用如下命令关闭图像引擎的渲染:simSetBooleanParameter(sim_boolparam_display_enabled,false)
开启单独的图像引擎线程
http://www.coppeliarobotics.com/helpFiles/en/simulation.htm
仿真属性设置:http://www.coppeliarobotics.com/helpFiles/en/simulationPropertiesDialog.htm
附录
1. 贝塞尔曲线的解释
讲解
在线仿真
前端应用
2. 3维绘图软件
inventor:Autodesk公司出品,可以用来绘制机械臂3维模型。

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

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

相关文章

数据库中的增删改查操作、聚合函数、内置函数、分组查询

数据库中的增删改查操作、聚合函数、内置函数、分组查询 CRUD简介Create 新增语法示例单⾏数据全列插⼊单⾏数据指定列插⼊多⾏数据指定列插⼊ Retrieve 检索语法⽰例构造数据 Select全列查询指定列查询查询字段为表达式为查询结果指定别名语法⽰例 结果去重查询 Order by 排序…

Flink转换算子——flatMap/map/filter/keyby/reduce综合案例

需求: 对流数据中的单词进行统计&#xff0c;排除敏感词TMD【腾讯美团滴滴】 此处用到了一个windows版本的软件 netcat&#xff0c;具体用法&#xff0c;先解压&#xff0c;然后在路径中输入cmd,来到黑窗口。 官网地址&#xff1a;netcat 1.11 for Win32/Win64Netcat介绍及安装…

图算法 | 3、图分析与数据科学

图分析(Graph Analytics)在本质上是对图数据的处理与分析&#xff0c;其过程可以概括为图计算。 而图计算的范畴不仅包含数据的计算或分析&#xff0c;还包含元数据管理、模式管理、数据建模、数据清洗、转换、加载、治理、图分析与计算等一系列操作。 或许我们用大数据生命周…

鲸鱼机器人和乐高机器人的比较

鲸鱼机器人和乐高机器人各有其独特的优势和特点&#xff0c;家长在选择时可以根据孩子的年龄、兴趣、经济能力等因素进行综合考虑&#xff0c;选择最适合孩子的教育机器人产品。 优势 鲸鱼机器人 1&#xff09;价格亲民&#xff1a;鲸鱼机器人的产品价格相对乐高更为亲民&…

【Unity】 GamePlay开发:通用的检查点/成就/条件触发系统

特别适用于各种解谜关卡, 成就系统&#xff0c;任务系统&#xff0c;的 通用事件处理 CheckPointHandler.cs随便挂在场景中的某个物体上 (单例模式&#xff0c;场景中只要有一个&#xff09; 1) How To Use CheckPoint Events是一个列表&#xff0c;每个元素是一个组合事件&…

企业OA管理系统:Spring Boot技术实现与案例研究

摘要 随着信息技术在管理上越来越深入而广泛的应用&#xff0c;管理信息系统的实施在技术上已逐步成熟。本文介绍了企业OA管理系统的开发全过程。通过分析企业OA管理系统管理的不足&#xff0c;创建了一个计算机管理企业OA管理系统的方案。文章介绍了企业OA管理系统的系统分析部…

【SQL Server】华中农业大学空间数据库实验报告 实验三 数据操作

1.实验目的 熟悉了解掌握SQL Server软件的基本操作与使用方法&#xff0c;以及通过理论课学习与实验参考书的帮助&#xff0c;熟练掌握使用T-SQL语句和交互式方法对数据表进行插入数据、修改数据、删除数据等等的操作&#xff1b;作为后续实验的基础&#xff0c;根据实验要求重…

基于AXI PCIE IP的FPGA PCIE卡示意图

创作不易&#xff0c;转载请注明出处&#xff1a;https://blog.csdn.net/csdn_gddf102384398/article/details/143926217 上图中&#xff0c;在FPGA PCIE卡示意图内&#xff0c;有2个AXI Master设备&#xff0c;即&#xff1a;PCIE到AXI4-Full-Master桥、AXI CDMA IP&#xff1…

Docker1:认识docker、在Linux中安装docker

欢迎来到“雪碧聊技术”CSDN博客&#xff01; 在这里&#xff0c;您将踏入一个专注于Java开发技术的知识殿堂。无论您是Java编程的初学者&#xff0c;还是具有一定经验的开发者&#xff0c;相信我的博客都能为您提供宝贵的学习资源和实用技巧。作为您的技术向导&#xff0c;我将…

java-排序算法汇总

排序算法&#xff1a; 冒泡排序&#xff08;Bubble Sort&#xff09; 选择排序&#xff08;Selection Sort&#xff09; 插入排序&#xff08;Insertion Sort&#xff09; 快速排序&#xff08;Quick Sort&#xff09; 归并排序&#xff08;Merge Sort&#xff09; 堆排序&…

瑞佑液晶控制芯片RA6807系列介绍 (三)软件代码详解 Part.10(让PNG图片动起来)完结篇

RA6807是RA8876M的缩小版&#xff0c;具备RA8876M的所有功能&#xff0c;只将MCU控制接口进行缩减&#xff0c;仅保留SPI-3和I2C接口&#xff0c;其它功能基本相同。 该芯片最大可控制854x600的分辨率&#xff0c;内建64Mbits显存&#xff0c;多个图层&#xff0c;使用起来相当…

org.apache.log4j的日志记录级别和基础使用Demo

org.apache.log4j的日志记录级别和基础使用Demo&#xff0c;本次案例展示&#xff0c;使用是的maven项目&#xff0c;搭建的一个简单的爬虫案例。里面采用了大家熟悉的日志记录插件&#xff0c;log4j。来自apache公司的开源插件。 package com.qian.test;import org.apache.log…

深度学习实战人脸识别

文章目录 前言一、人脸识别一般过程二、人脸检测主流算法1. MTCNN2. RetinaFace3. CenterFace4. BlazeFace5. YOLO6. SSD7. CascadeCNN 三、人脸识别主流算法1.deepface2.FaceNet3.ArcFace4.VGGFace5.DeepID 四、人脸识别系统实现0.安装教程与资源说明1. 界面采用PyQt5框架2.人…

Qt 字符串、信号槽详解

目录 1.QT中字符串QString的使用 1.1.常见的函数 1.2.总结QT中跟字符&#xff0c;字符串有关的类 1.3.字符串列表StringList 2.信号与槽 1.作用 2.右键转到槽函数 3.关联信号与槽函数(让你的信号跟槽函数之间形成一一对应关系) 4.信号与槽的特点 3.总结 1.继承自QMainWin…

Vscode进行Java开发环境搭建

Vscode进行Java开发环境搭建 搭建Java开发环境(Windows)1.Jdk安装2.VsCode安装3.Java插件4.安装 Spring 插件5.安装 Mybatis 插件5.安装Maven环境6.Jrebel插件7.IntelliJ IDEA Keybindings8. 收尾 VS Code&#xff08;Visual Studio Code&#xff09;是由微软开发的一款免费、开…

响应式数据(v-on、v-if、v-show、v-for、v-bind、v-model、computed、watch)

目录 一、事件绑定指令v-on 二、条件渲染指令v-if 三、v-show 四、遍历指令v-for 1、遍历对象的值 2、遍历对象的值和键&#xff08;先值后键&#xff09; 3、遍历对象的值、键和索引 4、遍历数组的值和索引 五、属性动态化指令v-bind(单向) 【CSS样式的绑定&#…

设计模式之 命令模式

命令模式&#xff08;Command Pattern&#xff09;是行为型设计模式之一&#xff0c;它将请求&#xff08;或命令&#xff09;封装成一个对象&#xff0c;从而使用户能够将请求发送者与请求接收者解耦。通过命令模式&#xff0c;调用操作的对象与执行操作的对象不直接关联&…

微信小程序2-地图显示和地图标记

一、index修改页面&#xff0c;让页面能够显示地图和一个添加标记的按钮。 index.wxml <scroll-view class"scrollarea" scroll-y type"list"><view class"index_container"><map id"map" style"width: 100%; h…

使用eclipse构建SpringBoot项目

我这里用eclipse2018版本做演示&#xff0c;大家有需要的可以下载Eclipse Downloads | The Eclipse Foundation 1.打开eclipse&#xff0c;选择存放代码的位置 2.选择 file >> new >> project >> 选择springboot文件下的 spring starter project 2.这里选择N…

uniapp接入BMapGL百度地图

下面代码兼容安卓APP和H5 百度地图官网&#xff1a;控制台 | 百度地图开放平台 应用类别选择《浏览器端》 /utils/map.js 需要设置你自己的key export function myBMapGL1() {return new Promise(function(resolve, reject) {if (typeof window.initMyBMapGL1 function) {r…