EtherCAT超高速实时运动控制卡XPCIE1032H上位机C#开发(三):EtherCAT总线CSP,CSV,CST模式切换

电子说

1.3w人已加入

描述

XPCIE1032H功能简介

XPCIE1032H是一款基于PCI Express的EtherCAT总线运动控制卡,可选6-64轴运动控制,支持多路高速数字输入输出,可轻松实现多轴同步控制和高速数据传输。

XPCIE1032H运动控制卡集成了强大的运动控制功能,结合MotionRT7运动控制实时软核,解决了高速高精应用中,PC Windows开发的非实时痛点,指令交互速度比传统的PCI/PCIe快10倍。

ethercat

XPCIE1032H支持PWM,PSO功能,板载16进16出通用IO口,其中输出口全部为高速输出口,可配置为4路PWM输出口或者16路高速PSO硬件比较输出口。输入口含有8路高速输入口,可配置为4路高速色标锁存或两路编码器输入。

XPCIE1032H搭配MotionRT7实时内核,使用本地LOCAL接口连接,通过高速的核内交互,可以做到更快速的指令交互,单条指令与多条指令一次性交互时间可以达到3-5us左右。

ethercat

➜XPCIE1032H与MotionRT7实时内核的配合具有以下优势:

ethercat

1.支持多种上位机语言开发,所有系列产品均可调用同一套API函数库;

2.借助核内交互,可以快速调用运动指令,响应时间快至微秒级,比传统PCI/PCIe快10倍;

3.解决传统PCI/PCIe运动控制卡在Windows环境下控制系统的非实时性问题;

4.支持一维/二维/三维PSO(高速硬件位置比较输出),适用于视觉飞拍、精密点胶和激光能量控制等应用;

5.提供高速输入接口,便于实现位置锁存;

6.支持EtherCAT总线和脉冲输出混合联动、混合插补。

ethercat

➜使用XPCIE1032H和MotionRT7进行项目开发时,通常需要进行以下步骤:

1.安装驱动程序,识别XPCIE1032H;

2.打开并执行文件“MotionRT710.exe”,配置参数和运行运动控制实时内核;

3.使用ZDevelop软件连接到控制器,进行参数监控。连接时请使用PCI/LOCAL方式,并确保ZDevelop软件版本在3.10以上;

4.完成控制程序开发,通过LOCAL链接方式连接到运动控制卡,实现实时运动控制。

ethercat

➜与传统PCI/PCIe卡和PLC的测试数据结果对比:

平均值 C++ LOCAL C# LOCAL 传统PCI/PCIe卡接口交互 PLC网口通讯交互
1w次单条读取交互周期 4.70us 5.3us 64us 500us-10ms
10w次单条读取交互周期 3.90us 5.7us 65us 500us-10ms
1w次多条读取交互周期 6.20us 8.85us 472us 500us-10ms
10w次多条读取交互周期 5.50us 8.37us 471us 500us-10ms

我们可以从测试对比结果看出,XPCIE1032H运动控制卡配合实时运动控制内核MotionRT7,在LOCAL链接(核内交互)的方式下,指令交互的效率是非常稳定,当测试数量从1w增加到10w时,单条指令交互时间与多条指令交互时间波动不大,非常适用于高速高精的应用。

XPCIE1032H控制卡安装

关闭计算机电源。

打开计算机机箱,选择一条空闲的XPCIE卡槽,用螺丝刀卸下相应的挡板条。

将运动控制卡插入该槽,拧紧挡板条上的固定螺丝。

XPCIE1032H驱动安装与建立连接参考往期文章EtherCAT超高速实时运动控制卡XPCIE1032H上位机C#开发(一):驱动安装与建立连接。

一、新建C#项目(VS2022)

ethercat

到正运动技术官网的下载中心选择需要的平台库文件。

ethercat

解压下载的安装包找到 “Zmcaux.cs” , “zauxdll.dll” , “zmotion.dll” 放入到项目文件中。

1、“Zmcaux.cs”放在项目根目录文件中,与bin目录同级。

ethercat

2、“zauxdll.dll”,“zmotion.dll”放在bin -> Debug。

ethercat

用vs打开新建的项目文件,在右边的解决方案资源管理器中点击显示所有,选中项目,右键“添加”->“现有项”,选中zmcaux.cs文件添加进在项目中。

ethercatethercat

双击Form1.cs里面的Form1,出现代码编辑界面,在文件开头写入using cszmcaux,并声明控制器句柄g_handle。

ethercat

二、相关PC函数介绍

相关PC函数介绍详情可参考“ZMotion PC函数库编程手册 V2.1.1”。

ethercatethercatethercatethercatethercatethercatethercatethercatethercat

其他基本轴参数相关PC函数:

ethercat

在form设计界面找到需要用到的控件拖拽到窗体中进行UI界面设计,设计效果图如下。

ethercat

三、相关程序以及设计思路

本例程以XPLCIE1032H搭载在MotionRT7实时内核上,通过EthereCAT总线口接一个松下伺服驱动器为节点0为例子,如下图:

ethercat

例程配套的basic程序(总线初始化脚本文件)如下,用户可根据实际需求配置总线节点数量,驱动器pdo等参数(例程中pdo列表配置的是18,该列表中同时包含607A-周期位置、60FF-周期速度、6071-周期力矩的数字字典,因此可以支持总线三种模式的切换)。

关于pdo列表的配置可以打开zbasic编程手册,搜索DRIVE_PROFILE指令。

 

'ECAT总线初始化
GLOBAL CONST BUS_TYPE = 0           '总线类型。可用于上位机区分当前总线类型
GLOBAL CONST Bus_Slot  = 0          '槽位号0(单总线控制器缺省0)
GLOBAL CONST Bus_AxisStart = 0      '总线轴起始轴号
GLOBAL CONST Bus_NodeNum   = 1      '总线配置节点数量,用于判断实际检测到的从站数量是否一致
GLOBAL MAX_AXISNUM                  '最大轴数
MAX_AXISNUM = SYS_ZFEATURE(0)
GLOBAL Bus_InitStatus               '总线初始化完成状态
Bus_InitStatus = -1
GLOBAL Bus_TotalAxisnum             '检查扫描的总轴数
DELAY(3000)        
'延时3S等待驱动器上电,不同驱动器自身上电时间不同,具体根据驱动器调整延时
?"总线通讯周期:",SERVO_PERIOD,"us"
Ecat_Init()        '初始化ECAT总线 
END
GLOBAL SUB Ecat_Init()
   local Node_Num,Temp_Axis,Drive_Vender,Drive_Device,Drive_Alias
   RAPIDSTOP(2)
   for i=0 to SYS_ZFEATURE(0) - 1          '初始化还原轴类型          
       AXIS_ENABLE(i) = 0
       ATYPE(i) = 0  
       DELAY(20)
   next
   Bus_InitStatus = -1
   Bus_TotalAxisnum = 0  
   SLOT_STOP(Bus_Slot)        
   DELAY(200)
   SLOT_SCAN(Bus_Slot)                  '扫描总线
   if return then 
       ?"总线扫描成功","连接从站设备数:"NODE_COUNT(Bus_Slot)
       '判断总线检测数量是否为实际接线数量
       if NODE_COUNT(Bus_Slot) <      > Bus_NodeNum then    
           ?"扫描节点数量与程序配置数量不一致!" ,"配置数量:"Bus_NodeNum,"检测数量:"NODE_COUNT(Bus_Slot)
           Bus_InitStatus = 0    '初始化失败。报警提示
       endif   
       '开始映射轴号
       for Node_Num = 0 to NODE_COUNT(Bus_Slot)-1          '遍历扫描到的所有从站节点
           Drive_Vender = NODE_INFO(Bus_Slot,Node_Num,0)      '读取驱动器厂商
           Drive_Device = NODE_INFO(Bus_Slot,Node_Num,1)      '读取设备编号
           Drive_Alias = NODE_INFO(Bus_Slot,Node_Num,3)        '读取设备拨码ID
           if NODE_AXIS_COUNT(Bus_Slot,Node_Num) <      > 0 then      '判断当前节点是否有电机
               for j = 0 to NODE_AXIS_COUNT(Bus_Slot,Node_Num) - 1  '根据节点带的电机数量循环配置轴参数(针对一拖多驱动器)
                   Temp_Axis = Bus_AxisStart + Bus_TotalAxisnum  '轴号按NODE顺序分配
                   BASE(Temp_Axis)
                   AXIS_ADDRESS = Bus_TotalAxisnum+1      '映射轴号
                   ATYPE = 65    '设置控制模式 65-位置 66-速度 67-转矩 
                   DRIVE_PROFILE = 18    '驱动器PDO可设置为模式18
                   DISABLE_GROUP(Temp_Axis)        '每轴单独分组
                   Bus_TotalAxisnum = Bus_TotalAxisnum+1  '总轴数+1
               next
           endif
       next    
       ?"轴号映射完成","连接总轴数:"Bus_TotalAxisnum    
       WA 200
       SLOT_START(Bus_Slot)                  '启动总线
       if return then     
           WDOG = 1                    '使能总开关
           ?"开始清除驱动器错误"
           for i = Bus_AxisStart to Bus_AxisStart + Bus_TotalAxisnum - 1 
               BASE(i)
               DRIVE_CLEAR(0)
               DELAY 50
               ?"驱动器错误清除完成"
               DATUM(0)              '清除控制器轴状态错误"
               WA 100  
               '"轴使能"
               AXIS_ENABLE = 1
           next
           Bus_InitStatus  = 1
           ?"轴使能完成"
           ?"总线开启成功"      
       else
           ?"总线开启失败"
           Bus_InitStatus = 0
       endif  
   else
       ?"总线扫描失败"
       Bus_InitStatus = 0
   endif
ENDSUB

 

1、通过LOCAL链接的方式链接到运动控制卡。

 

private void Connect_Button_Click(object sender, EventArgs e)
{
   if (g_handle == (IntPtr)0)
   {
       DisConnect_Button_Click(sender, e);
   }
   //MotionRT7通过local连接的方式连接到控制器
   zmcaux.ZAux_FastOpen(5, "local", 1000, out g_handle);
   if (g_handle != (IntPtr)0)
   {
       this.Text = "已链接";
       timer1.Enabled = true;      //连接成功后开启定时器
       Connect_Button.BackColor = Color.Green;
       MessageBox.Show("链接成功");
   }
   else
   {
       MessageBox.Show("链接失败,请选择正确的LOCAL!");
   }
}

 

2、选择编辑好的BAS脚本文件下载到控制器ram(掉电不保存)。

注意:下载完成之后会自动执行bas程序进行总线初始化。

 

private void BasDownLoad_Button_Click(object sender, EventArgs e)
{
   if (g_handle == (IntPtr)0)
   {
       MessageBox.Show("未链接到控制器!", "提示");
   }
   else
   {
       int tmpret = 0;
       string strFilePath;
       //显示一个标准对话框,提示用户打开想要选择的bas文件
       OpenFileDialog openFileDialog1 = new OpenFileDialog();
       openFileDialog1.InitialDirectory = "\";
       openFileDialog1.Filter = "配置文件(*.bas)|*.bas";
       openFileDialog1.RestoreDirectory = true;
       openFileDialog1.FilterIndex = 1;
       //打开配置文件
       if (openFileDialog1.ShowDialog() == DialogResult.OK)            
       {
           strFilePath = openFileDialog1.FileName;
           //下载到RAM,下载完成之后会自动运行一次bas程序,执行总线初始化
           tmpret = zmcaux.ZAux_BasDown(g_handle, strFilePath, 0);
           if (tmpret != 0)
           {
               MessageBox.Show("文件下载失败!", "提示");
           }
           else
           {
               DownLoadFlag = true;    //更新文件下载标志
               MessageBox.Show("文件下载成功!", "提示");
           }
       }
   }
}

 

3、初始化程序执行后,通过ZAux_Direct_GetUserVar函数接口,可以读取basic程序自定义的变量—在此案例,获取的是总线初始化映射轴数量、总线初始化起始轴号和总线初始化完成状态。

 

GLOBAL CONST Bus_AxisStart       '总线轴起始轴号
GLOBAL Bus_InitStatus            '总线初始化完成状态
GLOBAL  Bus_TotalAxisnum         '检查扫描的总轴数
使用定时器,对总线轴数量,总线起始轴号,初始化状态等总线初始化信息进行更新。
private void Update_EcatInitMessage()
{
   //读取basic程序自定义变量--总线初始化的总轴数
   zmcaux.ZAux_Direct_GetUserVar(g_handle, "Bus_TotalAxisnum", ref EcatAxisNum);
   //读取basic程序自定义变量--总线初始化的总线起始轴号
   zmcaux.ZAux_Direct_GetUserVar(g_handle, "Bus_AxisStart", ref EcatStartAxisNum);
   //读取basic程序自定义变量--总线初始化的初始化状态
   zmcaux.ZAux_Direct_GetUserVar(g_handle, "Bus_InitStatus", ref EcatInitStatus);
   //刷新界面总线初始化信息
   if (EcatInitStatus == 1 && DownLoadFlag == true)
   {
       BusAxisNum_Label.Text = "总线轴数量 : " + EcatAxisNum.ToString();
       BusStartAxis_Label.Text = "总线起始轴 : " + EcatStartAxisNum.ToString();
       InitState_Label.Text = "初始化状态 : 成功";
   }
   else if(EcatInitStatus == 0 && DownLoadFlag == true)
   {
       BusAxisNum_Label.Text = "总线轴数量 : " + EcatAxisNum.ToString();
       BusStartAxis_Label.Text = "总线起始轴 : " + EcatStartAxisNum.ToString();
       InitState_Label.Text = "初始化状态 : 失败";
   }
   else
   {
       InitState_Label.Text = "初始化状态 : 未完成";
   }
}

 

4、定时器中加入ZAux_Direct_GetUnits等函数接口对轴的脉冲当量、运行速度、加速度、减速度、轴类型、DPOS、MPOS、轴状态和轴的运动状态进行实时的监控并反馈。

 

private void Update_AxisPara()
{
   int CurAxisAtype = 0;           //当前轴类型
   int CurAxisIdle = 0;            //当前轴运动完成标志
   int CurAxisStatus = 0;          //当前轴状态
   float CurAxisDpos = 0;          //当前轴规划位置(DPOS)
   float CurAxisMpos = 0;          //当前轴反馈位置(MPOS)
   float CurAxisUnits = 0;         //当前轴脉冲当量
   float CurAxisSpeed = 0;         //当前轴运行速度
   float CurAxisAccel = 0;         //当前轴加速度
   float CurAxisDecel = 0;         //当前轴减速度
   if (DownLoadFlag == true)
   {
       //更新当前运动的轴
       MoveAxis = Convert.ToInt32(AxisNum_Value.Text);
       //读取当前轴的脉冲当量
       zmcaux.ZAux_Direct_GetUnits(g_handle, MoveAxis, ref CurAxisUnits);
       //读取当前轴的运行速度
       zmcaux.ZAux_Direct_GetSpeed(g_handle, MoveAxis, ref CurAxisSpeed);
       //读取当前轴的加速度
       zmcaux.ZAux_Direct_GetAccel(g_handle, MoveAxis, ref CurAxisAccel);
       //读取当前轴的减速度
       zmcaux.ZAux_Direct_GetDecel(g_handle, MoveAxis, ref CurAxisDecel);
       //读取当前轴的轴类型
       zmcaux.ZAux_Direct_GetAtype(g_handle, MoveAxis, ref CurAxisAtype);
       //读取当前轴的规划位置(DPOS)
       zmcaux.ZAux_Direct_GetDpos(g_handle, MoveAxis, ref CurAxisDpos);
       //读取当前轴的反馈位置(MPOS)
       zmcaux.ZAux_Direct_GetMpos(g_handle, MoveAxis, ref CurAxisMpos);
       //读取当前轴是否运动完成
       zmcaux.ZAux_Direct_GetIfIdle(g_handle, MoveAxis, ref CurAxisIdle);
       //读取当前轴的轴状态
       zmcaux.ZAux_Direct_GetAxisStatus(g_handle, MoveAxis, ref CurAxisStatus);
       //刷新界面的轴参数信息
       DPOS_Label.Text = "DPOS位置: " + CurAxisDpos;
       MPOS_Label.Text = "MPOS位置: " + CurAxisMpos;
       AxisState_Label.Text = "轴 状 态: " + CurAxisStatus;
       //运动结束信息刷新
       if (CurAxisIdle == 0)
       {
           Idle_Label.Text = "运动状态:  运动中";
       }
       else
       {
           Idle_Label.Text = "运动状态:  停止中";
       }
       //轴类型信息刷新
       if (CurAxisAtype == 65)
       {
           AxisAtype_Label.Text = "轴 类 型: 65(CSP)";
       }
       else if (CurAxisAtype == 66)
       {
           AxisAtype_Label.Text = "轴 类 型: 66(CSV)";
       }
       else if (CurAxisAtype == 67)
       {
           AxisAtype_Label.Text = "轴 类 型: 67(CST)";
       }
       else
       {
           AxisAtype_Label.Text = "轴 类 型: " + CurAxisAtype;
       }
   }  
}

 

5、当PDO包含607A时,ATYPE可设置为65,周期位置模式,此时使用运动指令控制电机运动。

通过文本控件的TextChanged(更改text属性引发的事件)事件,设置轴的脉冲当量、运行速度、加速度、减速度参数。

 

//脉冲当量变化
private void Units_Value_TextChanged(object sender, EventArgs e)
{
   //设置轴的脉冲当量
   zmcaux.ZAux_Direct_SetUnits(g_handle, MoveAxis, Convert.ToSingle(Units_Value.Text));
}
//运行速度变化
private void Speed_Value_TextChanged(object sender, EventArgs e)
{
   //设置轴的运行速度
   zmcaux.ZAux_Direct_SetSpeed(g_handle, MoveAxis, Convert.ToSingle(Speed_Value.Text));
}
//加速度变化
private void Accel_Value_TextChanged(object sender, EventArgs e)
{
   //设置轴的加速度
   zmcaux.ZAux_Direct_SetAccel(g_handle, MoveAxis, Convert.ToSingle(Accel_Value.Text));
}
//减速度变化
private void Decel_Value_TextChanged(object sender, EventArgs e)
{
   //设置轴的减速度
   zmcaux.ZAux_Direct_SetDecel(g_handle, MoveAxis, Convert.ToSingle(Decel_Value.Text));
}
轴的基本参数设置完成后,可以通过按钮控件的MouseDown(鼠标在组件上方并按下时发生)事件,调用单轴持续运动的函数ZAux_Direct_Single_Vmove指令,函数第三个参数设置为1(正向)使电机往正方向持续运动;
通过按钮控件的MouseUp(鼠标在组件上方并释放时发生)事件,调用单轴停止运动的函数指令ZAux_Direct_Single_Cancel,使电机运动停止。
同理,负方向运动只需要将ZAux_Direct_Single_Vmove函数指令的第三个参数设置为-1(负向)使电机往负方向持续运动。
//正向持续
private void PosiTive_Button_MouseDown(object sender, MouseEventArgs e)
{
   //单轴持续运动--正向
   zmcaux.ZAux_Direct_Single_Vmove(g_handle, MoveAxis, 1);
}
//正向停止
private void PosiTive_Button_MouseUp(object sender, MouseEventArgs e)
{
   //单轴停止运动
   zmcaux.ZAux_Direct_Single_Cancel(g_handle, MoveAxis, 2);
}
//负向持续
private void NegaTive_Button_MouseDown(object sender, MouseEventArgs e)
{
   //单轴持续运动--负向
   zmcaux.ZAux_Direct_Single_Vmove(g_handle, MoveAxis, -1);
}
//负向停止
private void NegaTive_Button_MouseUp(object sender, MouseEventArgs e)
{
   //单轴停止运动
   zmcaux.ZAux_Direct_Single_Cancel(g_handle, MoveAxis, 2);
}

 

6、当PDO包含60FF时,ATYPE可设置为66,周期速度模式,限制最大力矩。使用DAC指令控制电机以设置值的速度运行,速度单位有两个,脉冲数/S和R/MIN,有驱动器确定,使用时先给较小的数值,观察电机速度情况,再加大。

特别地,将DAC值置为0,电机停止转动;DAC值为负,电机负向转动。

 

private void SpeedValue_Config_Button_Click(object sender, EventArgs e)
{
   int ret1, ret2;            
   //设置周期速度模式的DAC值
   ret1 = zmcaux.ZAux_Direct_SetDAC(g_handle, (uint)MoveAxis, Convert.ToSingle(CSV_Value.Text));
   //周期速度模式限制转矩
   ret2 = zmcaux.ZAux_BusCmd_SetMaxDriveTorque(g_handle, (uint)MoveAxis, Convert.ToInt32(TorqueLimit_Value.Text));
   if (ret1 == 0 && ret2 == 0)
   {
       MessageBox.Show("设置成功!", "提示");
   }
   else
   {
       MessageBox.Show("设置失败!", "提示");
   }
}

 

7、当PDO包含6071时,ATYPE可设置为67,周期力矩模式,限制最大速度。此时使用DAC指令控制电机以设置值的力矩运行,DAC值范围0-1000,对应0-100%的6071设置值,比如DAC=10,此时电机力矩=1%的6071值。

特别地,将DAC值置为0,电机停止转动;DAC值为负,电机负向转动。

CST周期力矩模式时,力矩给定要从小到大给定,驱动器内部要做好速度限制,防止飞车出现安全事故。

 

private void CST_Set_Button_Click(object sender, EventArgs e)
{
   if (g_handle == (IntPtr)0)
   {
       MessageBox.Show("未链接到控制器!", "提示");
       return;
   }
   int ret1,ret2,ret3;
   //周期力矩模式设置总线轴最大转矩
   ret1 = zmcaux.ZAux_BusCmd_SetMaxDriveTorque(g_handle, (uint)MoveAxis, Convert.ToInt32(MaxTorque_Value.Text));
   //周期力矩模式设置总线轴目标转矩
   ret2 = zmcaux.ZAux_Direct_SetDAC(g_handle, (uint)MoveAxis, Convert.ToSingle(TargetTorque_Value.Text));
   //周期力矩模式的速度限制--汇川驱动器的最大限速数字字典是607Fh
   ret3 = zmcaux.ZAux_BusCmd_SDOWriteAxis(g_handle, (uint)MoveAxis, 0x607F, 0, 7, Convert.ToInt32(SpeedLimit_Value.Text));
   if (ret1 == 0 && ret2 == 0 && ret3 == 0)
   {
       MessageBox.Show("设置成功!", "提示");
   }
   else
   {
       MessageBox.Show("设置失败!", "提示");
   }
}

 

8、通过单选框控件的CheckedChanged(checked属性更改时发生的事件)事件,对切换的模式进行轴类型的设置,并将DAC置0,防止实际加工情况时事故的发生。

速度模式下限制最大力矩,力矩模式下限制最大速度一般都是通过SDO读写驱动器参数的。

 

//周期位置模式
private void CSP_RadioButton_CheckedChanged(object sender, EventArgs e)
{
   if (g_handle == (IntPtr)0)
   {
       MessageBox.Show("未链接到控制器!", "提示");
       return;
   }
   //将轴类型设置成65-周期位置模式
   zmcaux.ZAux_Direct_SetAtype(g_handle, MoveAxis, 65);
   //将DAC置0,防止切换成周期速度模式或周期力矩模式时出现事故
   zmcaux.ZAux_Direct_SetDAC(g_handle, (uint)MoveAxis, 0);
}
//周期速度模式
private void CSV_RadioButton_CheckedChanged(object sender, EventArgs e)
{
   if (g_handle == (IntPtr)0)
   {
       MessageBox.Show("未链接到控制器!", "提示");
       return;
   }
   //将轴类型设置成66-周期速度模式
   zmcaux.ZAux_Direct_SetAtype(g_handle, MoveAxis, 66);
   //将DAC置0,防止切换成周期力矩模式时出现事故
   zmcaux.ZAux_Direct_SetDAC(g_handle, (uint)MoveAxis, 0);
}
//周期力矩模式
private void CST_RadioButton_CheckedChanged(object sender, EventArgs e)
{
   if (g_handle == (IntPtr)0)
   {
       MessageBox.Show("未链接到控制器!", "提示");
       return;
   }
   //将轴类型设置成67-周期力矩模式
   zmcaux.ZAux_Direct_SetAtype(g_handle, MoveAxis, 67);
   //将DAC置0,防止切换成周期速度模式时出现事故
   zmcaux.ZAux_Direct_SetDAC(g_handle, (uint)MoveAxis, 0);
}

 

四、运行效果

(1)周期位置模式运行结果。

ethercat

(2)周期速度模式运行结果。

ethercat

(3)周期力矩模式运行结果。

ethercat

本次,正运动技术EtherCAT超高速实时运动控制卡XPCIE1032H上位机C#开发(三):EtherCAT总线CSP,CSV,CST模式切换,就分享到这里。

源代码下载地址以及更多精彩内容请关注“正运动小助手”公众号,需要相关开发环境与例程代码,请咨询正运动技术销售工程师。

本文由正运动技术原创,欢迎大家转载,共同学习,一起提高中国智能制造水平。文章版权归正运动技术所有,如有转载请注明文章来源。

审核编辑 黄宇

打开APP阅读更多精彩内容
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉

全部0条评论

快来发表一下你的评论吧 !

×
20
完善资料,
赚取积分