【解决视觉引导多个位置需要标定多个位置的问题】

news2025/1/13 13:43:26

** 以下只针对2D定位,就是只有X、Y、Rz三个自由度的情况。**

假设一种情况,当视觉给机器人做引导任务时,零件有多个,分布在料框里,视觉需要走多个位置去拍,那么只需要对第一个位置确定拍照位,确定试抓位,其余的拍照位置通过只平移XY,零件特征偏移归0的方式确定,这种方式当两垛零件方向不一样的时候就不得不单独走9点。
为了解决这个问题,提高调试效率,现在分享一种通过计算来偷懒的方法。
通过这种方式,只需要9点标定一次,然后通过告知机器人当前拍照位坐标,计算零件在Base下的坐标。

1. 数学原理:

坐标系ABC分别代表机器人基座、Tool、相机。
已知的有B_A:示教器上能看到。
已知的有C_B: 物体在相机坐标系位置。
求:C_A的位置= C_A = B_A * C_B
简单来说就是只要知道C_A和C_B相乘就能算出来C_A,矩阵公式我就不画了,相信大家都见过。。。

在这里插入图片描述

2. 直接本主题,上代码:

namespace Math_Matrix
{
    /// <summary>
    /// 已知B_A坐标系关系,C_B坐标关系,求C_A
    /// </summary>
    public static class Position
    {
        /// <summary>
        /// 机器人工具中心在Base的平移X坐标
        /// </summary>
        public static double ToolCenterInBaseX { get; set; }

        /// <summary>
        /// 机器人工具中心在Base的平移Y坐标
        /// </summary>
        public static double ToolCenterInBaseY { get; set; }

        /// <summary>
        /// 机器人工具中心在Base的旋转Z坐标
        /// </summary>
        public static double ToolCenterInBaseRz { get; set; }


        /// <summary>
        /// 物体在工具坐标系下的平移X坐标
        /// </summary>
        public static double ObjectInToolCenterX { get; set; }

        /// <summary>
        /// 物体在工具坐标系下的平移Y坐标
        /// </summary>
        public static double ObjectInToolCenterY { get; set; }

        /// <summary>
        /// 物体在工具坐标系下的旋转Z坐标
        /// </summary>
        public static double ObjectInToolCenterRz { get; set; }

        /// <summary>
        /// 计算物体在机器人底座的位置
        /// </summary>
        /// <returns>坐标XYR</returns>
        public static XYR CalculaterObjectInBaseXYR()
        {
            Matrix<double> OB = CalculateTransformationMatrix();

            double r00 = OB.At(0, 0);
            double r10 = OB.At(1, 0);

            double Tx = OB.At(0, 3);
            double Ty = OB.At(1, 3);
            double Rz = Math.Atan2(r10, r00) * 180 / Math.PI; 

            return new XYR(Tx, Ty, Rz);
        }

        /// <summary>
        /// 矩阵相乘
        /// </summary>
        /// <returns>结果矩阵</returns>
        private static Matrix<double> CalculateTransformationMatrix()
        {
            Matrix<double> TW = DenseMatrix.OfArray(new double[,] {
                                                        { Math.Cos(ToolCenterInBaseRz*Math.PI/180), -Math.Sin(ToolCenterInBaseRz*Math.PI/180),0, ToolCenterInBaseX},
                                                        { Math.Sin(ToolCenterInBaseRz*Math.PI/180), Math.Cos(ToolCenterInBaseRz*Math.PI/180),0 ,ToolCenterInBaseY},
                                                        { 0, 0, 1 ,0},
                                                        { 0, 0, 0,1 }});

            Matrix<double> OT = DenseMatrix.OfArray(new double[,] {
                                                        { Math.Cos(ObjectInToolCenterRz*Math.PI/180), -Math.Sin(ObjectInToolCenterRz*Math.PI/180), 0,ObjectInToolCenterX},
                                                        { Math.Sin(ObjectInToolCenterRz*Math.PI/180), Math.Cos(ObjectInToolCenterRz*Math.PI/180),0, ObjectInToolCenterY},
                                                        { 0, 0, 1,0 },
                                                        { 0, 0, 0,1 }});
            return TW * OT;
        }
    }

    /// <summary>
    /// 记录坐标结果
    /// </summary>
    public struct XYR
    {
        /// <summary>
        /// 平移X坐标
        /// </summary>
        public double X { get; }

        /// <summary>
        /// 平移Y坐标
        /// </summary>
        public double Y { get; }

        /// <summary>
        /// Z轴旋转角度
        /// </summary>
        public double Rotation { get; }

        /// <summary>
        /// 构造的时候传入坐标值
        /// </summary>
        /// <param name="x">X</param>
        /// <param name="y">Y</param>
        /// <param name="rotation">Rz</param>
        public XYR(double x, double y, double rotation)
        {
            X = x;
            Y = y;
            Rotation = rotation;
        }
    }
}

运行验证图上红色ABC坐标系:

  static void Main(string[] args)
        {
            Position.ToolCenterInBaseX = 6;
            Position.ToolCenterInBaseY = 8;
            Position.ToolCenterInBaseRz = -40;

            Position.ObjectInToolCenterX = 8.8;
            Position.ObjectInToolCenterY = 9.8;
            Position.ObjectInToolCenterRz = 150;

            XYR pos = Position.CalculaterObjectInBaseXYR();

            Console.WriteLine(pos.X);
            Console.WriteLine(pos.Y);
            Console.WriteLine(pos.Rotation);

            Console.ReadKey();
        }

结果:
在这里插入图片描述

运行验证图上红色A B’ C坐标系:

static void Main(string[] args)
        {
            Position.ToolCenterInBaseX = 13;
            Position.ToolCenterInBaseY = 6;
            Position.ToolCenterInBaseRz = 0;

            Position.ObjectInToolCenterX = 6;
            Position.ObjectInToolCenterY = 4;
            Position.ObjectInToolCenterRz = 110;

            XYR pos = Position.CalculaterObjectInBaseXYR();

            Console.WriteLine(pos.X);
            Console.WriteLine(pos.Y);
            Console.WriteLine(pos.Rotation);

            Console.ReadKey();
        }

结果:
在这里插入图片描述

结尾发个广告,基于YOLO V8写的软件发布B站了,看到帮忙点个赞,过些天我会发源码给好朋友们!
链接:基于PySide6开发的YOLO V8视觉检测系统

拜了个拜。。。

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

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

相关文章

力扣6:N字形变化

代码&#xff1a; class Solution { public:string convert(string s, int numRows){int lens.size();if(numRows1){return s;}int d2*numRows-2;int count0;string ret;//第一行&#xff01;for(int i0;i<len;id){rets[i];}//第k行&#xff01;for(int i1;i<numRows-1;…

智能优化算法应用:基于教与学算法无线传感器网络(WSN)覆盖优化 - 附代码

智能优化算法应用&#xff1a;基于教与学算法无线传感器网络(WSN)覆盖优化 - 附代码 文章目录 智能优化算法应用&#xff1a;基于教与学算法无线传感器网络(WSN)覆盖优化 - 附代码1.无线传感网络节点模型2.覆盖数学模型及分析3.教与学算法4.实验参数设定5.算法结果6.参考文献7.…

超越GPT-4,拥有联网能力,Kimi-Chat大模型已免费使用,国内直接访问

目前ChatGPT的所有免费用户都已可以使用带有语音功能的ChatGPT。 人吧&#xff0c;总是贪婪的&#xff0c;我还想要ChatGPT Plus用户独享的“联网”功能。 目前对于ChatGPT来说&#xff0c;不想交钱&#xff0c;别拥有“联网”能力了&#xff0c;于是我找到了一个后起之秀&…

【差旅游记】新疆哈密回王府印象

哈喽&#xff0c;你好啊&#xff0c;我是雷工&#xff01; 2023年11月4号&#xff0c;那天的风的确挺大&#xff0c;逛完哈密博物馆考虑要不要去旁边的哈密回王府逛逛。想着来都来了&#xff0c;虽然网上评价不太好&#xff0c;还是去溜达一圈吧&#xff0c;于是决定自己去转转…

为啥网络安全那么缺人,但很多人却找不到工作?

文章目录 一、学校的偏向于学术二、学的东西太基础三、不上班行不行 为什么网络安全的人才缺口那么大&#xff0c;但是大学毕业能找到网安工作的人却很少&#xff0c;就连招聘都没有其他岗位多&#xff1f; 明明央视都说了网络安全的人才缺口还有300多万&#xff0c;现在找不到…

C++ 用ifstream读文件

输入流的继承关系: C++ 使用标准库类来处理面向流的输入和输出: iostream 处理控制台 IOfstream 处理命名文件 IOstringstream 完成内存 string 的 IO每个IO 对象都维护一组条件状态 flags (eofbit, failbit and badbit),用来指出此对象上是否可以进行 IO 操作。如果遇到错误…

vue实战——登录【详解】(含自适配全屏背景,记住账号--支持多账号,显隐密码切换,登录状态保持)

效果预览 技术要点——自适配全屏背景 https://blog.csdn.net/weixin_41192489/article/details/119992992 技术要点——密码输入框 自定义图标切换显示隐藏 https://blog.csdn.net/weixin_41192489/article/details/133940676 技术要点——记住账号&#xff08;支持多账号&…

「江鸟中原」有关HarmonyOS-ArkTS的Http通信请求

一、Http简介 HTTP&#xff08;Hypertext Transfer Protocol&#xff09;是一种用于在Web应用程序之间进行通信的协议&#xff0c;通过运输层的TCP协议建立连接、传输数据。Http通信数据以报文的形式进行传输。Http的一次事务包括一个请求和一个响应。 Http通信是基于客户端-服…

进程等待讲解

今日为大家分享有关进程等待的知识&#xff01;希望读完本文&#xff0c;大家能有一定的收获&#xff01; 正文开始&#xff01; 进程等待的引进 既然我们今天要讲进程等待这个概念&#xff01;那么只有我们把下面这三个方面搞明白&#xff0c;才能真正的了解进程等待&#x…

形象建设、生意经营、用户运营,汽车品牌如何在小红书一举多得?

随着小红书在多领域的持续成长&#xff0c;现在来小红书看汽车的用户&#xff0c;需求逐渐多元化与专业化。近1年的时间&#xff0c;有超过1亿人在小红书「主动搜索」过汽车内容&#xff0c;大家已经不仅限于玩车、用车&#xff0c;更是扩展到了百科全书式的看、选、买、学各个…

Python3 selenium 设置元素等待的三种方法

为什么要设置元素等待&#xff1f; 当你的网络慢的时候&#xff0c;打开网页慢&#xff0c;网页都没完全打开&#xff0c;代码已经在执行了&#xff0c;但是没找到你定位的元素&#xff0c;此时python会报错。 当你的浏览器或电脑反应慢&#xff0c;网页没完全打开&#xff0c;…

12、模块化编程

模块化编程 1、传统方式编程&#xff1a;所有的函数均放在main.c里&#xff0c;若使用的模块比较多&#xff0c;则一个文件内会有很多的代码&#xff0c;不利于代码的组织和管理&#xff0c;而且很影响便朝着的思路 2、模块化编程&#xff1a;把各个模块的代码放在不同的.c文件…

java--单继承、Object

java是单继承的&#xff0c;java中的类不支持多继承&#xff0c;但是支持多层继承。 反证法&#xff1a; 如果一个类同时继承两个类&#xff0c;然后两个类中都有同样的一个方法&#xff0c;哪当我创建这个类里的方法&#xff0c;是调用哪父类的方法 所以java中的类不支持多继…

PostgreSQL + SQL Server = WiltonDB

WiltonDB 是一个基于 PostgreSQL 的开源数据库&#xff0c;通过 Babelfish 插件支持 Microsoft SQL Server 协议以及 T-SQL 语句。 Babelfish 是亚马逊提供的一个开源项目&#xff0c;使得 PostgreSQL 数据库同时具有 Microsoft SQL Server 数据查询和处理的能力。Babelfish 可…

11、动态数码管显示

数码管驱动方式 1、单片机直接扫描&#xff1a;硬件设备简单&#xff0c;但会消耗大量的单片机CPU时间 2、专用驱动芯片&#xff1a;内部自带显存、扫描电路&#xff0c;单片机只需告诉他显示什么即可 #include <REGX52.H> //数组代表显示亮灯的内容0、1、2、3、4、5、…

单车模型及其线性化

文章目录 1 单车模型2 线性化3 实现效果4 参考资料 1 单车模型 这里讨论的是以后轴为中心的单车运动学模型&#xff0c;由下式表达&#xff1a; S ˙ [ x ˙ y ˙ ψ ˙ ] [ v c o s ( ψ ) v s i n ( ψ ) v t a n ( ψ ) L ] \dot S \begin{bmatrix} \dot x\\ \dot y\\ \d…

C++ : 初始化列表 类对象作为类成员

传统方式初始化 C 提供了初始化列表语法&#xff0c;用来初始化属性 初始化列表 语法&#xff1a; 构造函数()&#xff1a;属性1(值1), 属性2&#xff08;值2&#xff09;... {} class Person { public://传统方式初始化 Person(int a, int b, int c) {m_A a;m_B b;m_C c…

python之pyqt专栏6-信号与槽2

上一篇python之pyqt专栏5-信号与槽1-CSDN博客&#xff0c;我们通过信号与槽实现了点击Button&#xff0c;改变Label的文本内容。可以知道 信号是在类中定义的&#xff0c;是类的属性 槽函数是信号通过connect连接的任意成员函数&#xff0c;当信号发生时&#xff0c;执行与信号…

FFmpeg架构全面分析

一、简介 它的官网为&#xff1a;https://ffmpeg.org/&#xff0c;由Fabrice Bellard&#xff08;法国著名程序员Born in 1972&#xff09;于2000年发起创建的开源项目。该人是个牛人&#xff0c;在很多领域都有很大的贡献。 FFmpeg是多媒体领域的万能工具。只要涉及音视频领…

Selenium-Unittest单元测试框架

1、Unittest介绍 为什么要学习单元测试框架 测试用例的组织与运行需要单元测试框架的参与&#xff0c;从而满足不同测试场景的需要&#xff0c;单元测试框架提供了丰富的比较方法&#xff1a;实际结果与预期结果的对比测试结果 单元测试框架提供了丰富的日志&#xff1a;给出测…