几何向量:向量到平面投影和LookAt

news2024/11/15 23:40:20

      在研究所保密开发完后回来隔离两波,已经接近四五个月没碰外网电脑了,可以说是活成了原始人。
      因为某些开发细节原因,需要实现向量投影和LookAt功能,记录一下。
      首先实现向量到平面投影,如下:
在这里插入图片描述

      前面我们聊过点向量平面之间的关系,好像写过好几篇文章,这里就从简。其实计算投影无非就是计算P在平面G上的投影P0,计算如下:
请添加图片描述      我们根据平面G方程得到ABCD分量,然后求出系数k,带入X0、Y0、Z0代数式就能达到P0坐标了,如下:

using System.Collections;
using System.Collections.Generic;
using UnityEngine;

public class PALAVectorProjection : MonoBehaviour
{
    public Transform ground0;
    public Transform ground1;
    public Transform ground2;

    public Transform from;
    public Transform to;

    void Start()
    {

    }

    void Update()
    {
        Vector3 g0 = ground0.position;
        Vector3 g1 = ground1.position;
        Vector3 g2 = ground2.position;

        Debug.DrawLine(g0, g1, Color.black);
        Debug.DrawLine(g1, g2, Color.black);
        Debug.DrawLine(g2, g0, Color.black);

        Vector3 f = from.position;
        Vector3 t = to.position;

        Debug.DrawLine(f, t, Color.yellow);

        Vector3 f0 = GetProjectPoint(f, g0, g1, g2);
        Vector3 t0 = GetProjectPoint(t, g0, g1, g2);

        Debug.DrawLine(f0, t0, Color.white);
    }

    /// <summary>
    /// 获取向量到平面投影向量
    /// </summary>
    /// <param name="f"></param>
    /// <param name="t"></param>
    /// <param name="g0"></param>
    /// <param name="g1"></param>
    /// <param name="g2"></param>
    /// <returns></returns>
    private Vector3 GetProjectVector(Vector3 f, Vector3 t, Vector3 g0, Vector3 g1, Vector3 g2)
    {
        Vector3 f0 = GetProjectPoint(f, g0, g1, g2);
        Vector3 t0 = GetProjectPoint(t, g0, g1, g2);
        Vector3 vproj = t0 - f0;
        return vproj;
    }

    /// <summary>
    /// 计算点到平面投影
    /// </summary>
    /// <param name="p"></param>
    /// <param name="g0"></param>
    /// <param name="g1"></param>
    /// <param name="g2"></param>
    /// <returns></returns>
    private Vector3 GetProjectPoint(Vector3 p, Vector3 g0, Vector3 g1, Vector3 g2)
    {
        Vector3 gv1 = g1 - g0;
        Vector3 gv2 = g2 - g0;
        Vector3 n = Vector3.Cross(gv1, gv2).normalized;

        //构建平面参数
        float A = n.x, B = n.y, C = n.z;
        float D = -A * g0.x - B * g0.y - C * g0.z;

        //构建投影参数
        float X = p.x, Y = p.y, Z = p.z;
        float k = (A * X + B * Y + C * Z + D) / (A * A + B * B + C * C);

        float X0 = X - k * A;
        float Y0 = Y - k * B;
        float Z0 = Z - k * C;

        return new Vector3(X0, Y0, Z0);
    }
}

      效果如下:
在这里插入图片描述
      白色的线段是黄色线段在平面G上的投影。
      继续,接下来要实现一下LookAt功能,首先看一下unity的API,如下:

using System.Collections;
using System.Collections.Generic;
using UnityEngine;

public class PALATestLookAt : MonoBehaviour
{
    public Transform from;
    public Transform to;

    void Start()
    {

    }

    void Update()
    {
        Debug.DrawLine(from.position, to.position, Color.blue);
        Debug.DrawLine(from.position, from.position + from.right, Color.red);
        Debug.DrawLine(from.position, from.position + from.up, Color.green);
        from.LookAt(to);
    }
}

      效果如下:
在这里插入图片描述
      现在我自己实现一下,如下:
在这里插入图片描述

      朝向计算就是半角向量的180度旋转计算,设Z轴上模长等于ft模长的向量ft0绕半角向量ft1旋转180度,就变成了向量ft
      实现一下:

using System.Collections;
using System.Collections.Generic;
using UnityEngine;

public class PALAAxisLookAt : MonoBehaviour
{
    public Transform from;
    public Transform to;

    void Start()
    {

    }

    void Update()
    {
        Debug.DrawLine(from.position, to.position, Color.blue);
        Debug.DrawLine(from.position, from.position + from.right, Color.red);
        Debug.DrawLine(from.position, from.position + from.up, Color.green);

        GetAxisLookAt();
    }
    /// <summary>
    /// 绕中轴旋转180度
    /// </summary>
    private void GetAxisLookAt()
    {
        Vector3 f = from.position;
        Vector3 t = to.position;

        Vector3 ft = t - f;

        float ftlen = Vector3.Distance(f, t);
        Vector3 ft0 = Vector3.forward * ftlen;

        Vector3 ft1 = ft + ft0;

        from.eulerAngles = Vector3.zero;

        //绕ft1轴旋转180度
        from.RotateAround(f, ft1, 180);
    }
}

      效果如下:
在这里插入图片描述
      这是一种LookAt的实现方法。
      继续探索,现实中存在的朝向计算,坦克就是很好的例子,如下:
在这里插入图片描述

      坦克的炮台可以想象为绕世界坐标Y轴旋转,炮筒想象成绕本地坐标X轴旋转,稍微绘制一下,如下:
在这里插入图片描述

      首先求得ft在平面xfz上的投影ft0,ft’(fz)与投影ft0绕世界坐标Y轴夹角θy度就是“炮台”的旋转度数,然后ft0与ft绕本地坐标X轴夹角θx度就是“炮筒”的旋转度数,旋转以左手定则为准,那么接下来代码实现:

using System.Collections;
using System.Collections.Generic;
using System.Reflection;
using UnityEngine;

public class PALAProjectionLookAt : MonoBehaviour
{
    public Transform from;
    public Transform to;

    void Start()
    {
        Application.targetFrameRate = 60;
    }

    void Update()
    {
        Vector3 f = from.position;
        Vector3 t = to.position;
        Vector3 lx = f + from.right;
        Vector3 ly = f + from.up;
        Vector3 lz = f + from.forward;

        Debug.DrawLine(f, lx, Color.red);
        Debug.DrawLine(f, ly, Color.green);
        Debug.DrawLine(f, t, Color.black);

        GetProjectLookAt();
    }

    /// <summary>
    /// 世界坐标Y轴投影旋转
    /// 本地坐标X轴旋转
    /// </summary> 
    private void GetProjectLookAt()
    {
        Vector3 f = from.position;
        Vector3 t = to.position;
        Vector3 x = f + Vector3.right;
        Vector3 y = f + Vector3.up;
        Vector3 z = f + Vector3.forward;

        //重置旋转
        from.eulerAngles = Vector3.zero;

        //绕世界Y轴旋转
        Vector3 yaxis = Vector3.up;
        Vector3 ft0 = GetProjectVector(f, t, f, x, z);
        Vector3 fz = Vector3.forward;
        float θy = GetVectorAxisAngle(fz, ft0, yaxis);
        from.RotateAround(f, yaxis, θy);
        //绕本地X轴旋转
        Vector3 xaixs = from.right;
        Vector3 ft = t - f;
        float θx = GetVectorAxisAngle(ft0, ft, xaixs);
        from.RotateAround(f, xaixs, θx);

#if UNITY_EDITOR
        Debug.LogFormat("PALAProjectionLookAt θx = {0} θy = {1}", θx, θy);
#endif
    }

    /// <summary>
    /// 获取绕axis轴f到t的角度
    /// </summary>
    /// <param name="f"></param>
    /// <param name="t"></param>
    /// <param name="axis"></param>
    /// <returns></returns>
    private float GetVectorAxisAngle(Vector3 f, Vector3 t, Vector3 axis)
    {
        float deg = Vector3.SignedAngle(f, t, axis);
        return deg;
    }

    /// <summary>
    /// 计算f(不变)的投影向量
    /// </summary>
    /// <param name="f"></param>
    /// <param name="t"></param>
    /// <param name="g0"></param>
    /// <param name="g1"></param>
    /// <param name="g2"></param>
    /// <returns></returns>
    private Vector3 GetProjectVector(Vector3 f, Vector3 t, Vector3 g0, Vector3 g1, Vector3 g2)
    {
        Vector3 t0 = GetProjectPoint(t, g0, g1, g2);
        Vector3 ft0 = t0 - f;
        return ft0;
    }

    /// <summary>
    /// 计算点到平面投影
    /// </summary>
    /// <param name="p"></param>
    /// <param name="g0"></param>
    /// <param name="g1"></param>
    /// <param name="g2"></param>
    /// <returns></returns>
    private Vector3 GetProjectPoint(Vector3 p, Vector3 g0, Vector3 g1, Vector3 g2)
    {
        Vector3 gv1 = g1 - g0;
        Vector3 gv2 = g2 - g0;
        Vector3 n = Vector3.Cross(gv1, gv2).normalized;

        //构建平面参数
        float A = n.x, B = n.y, C = n.z;
        float D = -A * g0.x - B * g0.y - C * g0.z;

        //构建投影参数
        float X = p.x, Y = p.y, Z = p.z;
        float k = (A * X + B * Y + C * Z + D) / (A * A + B * B + C * C);

        float X0 = X - k * A;
        float Y0 = Y - k * B;
        float Z0 = Z - k * C;

        return new Vector3(X0, Y0, Z0);
    }
}

      效果如下:
在这里插入图片描述
      这也是一种LookAt的实现方法。
      自己实现LookAt的好处是可以自由控制朝向轴,可能API不一定适合开发需求。
      好,以后有时间继续。

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

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

相关文章

第二证券|鲍威尔发声:释放重磅信号,美股大涨!中概股狂涨

大家早上好&#xff01;昨夜今晨又有许多大事产生&#xff1a;美联储主席鲍威尔证明&#xff0c;12月开端或许放缓加息&#xff1b;微软涨逾1100亿美元&#xff0c;美股进入技能型牛市&#xff1b;小鹏轿车昨日暴升近50%&#xff0c;中概股11月涨逾40%&#xff1b;法、德不满美…

java计算机毕业设计ssm企业日常事务管理系统sl5xl(附源码、数据库)

java计算机毕业设计ssm企业日常事务管理系统sl5xl&#xff08;附源码、数据库&#xff09; 项目运行 环境配置&#xff1a; Jdk1.8 Tomcat8.5 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#…

全栈性能测试教程之性能测试相关知识(二) Jmeter的应用

性能测试相关知识&#xff08;二&#xff09; Jmeter的应 1、性能测试的方法 1.1验收负载测试&#xff1a; 在QA的环境模拟生产运行的业务压力和使用场景组合&#xff0c;测试系统的性能是否满足生产环境的性能诉求。 1.2负载测试 在被测系统上持续不断的增加压力&#xff…

易点易动RFID固定资产管理系统助力企业年终固定资产大盘点

固定资产作为资产构成的重要组成部分&#xff0c;也是企业完成生产经营的物质保障&#xff0c;对企业的发展起着重要的作用。越来越多的企业者开始重视企业内部的固定资产管理&#xff0c;从而会定期对固定资产进行盘点&#xff0c;以保证固定资产账实一致、账账相符。每逢年底…

粒子群算法和鲸鱼算法的比较(Matlab代码实现)

目录 1 粒子群优化算法 2 鲸鱼优化算法 3 粒子群算法和鲸鱼算法比较 4 Matlab代码实现 1 粒子群优化算法 粒子群优化算法(PSO&#xff1a;Particle swarm optimization) 是一种进化计算技术&#xff08;evolutionary computation&#xff09;。源于对鸟群捕食的行为研究…

GitHub限时开源36小时的阿里Java架构师学习手册,上线即标星35k+

前言 今年受大环境影响面试于往年相比难得多&#xff0c;对程序员要求越来越高&#xff01;环境我们无法改变能改变的就是自己&#xff0c;努力提升技术&#xff01; 我在GitHub无意见看见115k的Java教程&#xff0c;感觉还不错&#xff0c;给大家看看&#xff01;文档总共分为…

Cloud Mail JavaScript管理邮件

Cloud Mail JavaScript管理邮件 使用流行的云服务发送、接收和管理邮件。 云邮件使用流行的云服务简化了邮件的发送、接收和管理&#xff0c;包括Amazon SES、Microsoft 365(Outlook Mail)和Gmail。还支持包括OAuth、TLS 1.3和TLS 1.2在内的现代身份验证和安全选项。 云邮件功…

前后端验证码交互完整流程

本文章基于vueelement-uispringbootredis讲解&#xff0c;其他的都是工具&#xff0c;可以直接拿来用&#xff0c;不懂redis没关系&#xff08;因为本文只用了简单的存取&#xff09;&#xff0c;但前面三个要懂 如果你只想看前端或者后端的代码逻辑&#xff0c;本文章同样适用…

提升代码可读性,减少if-else的几个小技巧

前言&#x1f481;‍♂️ 相信大家或多或少都接触过拥有庞大 if else 的项目代码吧&#xff0c;多重嵌套的 if else 在维护的时候真的让人很恼火&#x1f621;&#xff0c;有时候一个 bug 排查下来&#xff0c;严重感觉身体被掏空&#x1f63f;。 本文并未有消灭或歧视 if el…

2022年测试行业的新变化

清晨的第一缕阳光&#xff0c;打开手机&#xff0c;熟悉的数字映入眼帘&#xff0c;洗漱完毕&#xff0c;戴好口罩&#xff0c;新的一天开始了 看似平静的表面&#xff0c;却到处暗潮汹涌 偶然间在朋友圈看到一则招聘广告&#xff0c;Base上海&#xff0c;两年以上经验&#…

python接口自动化测试框架

本文总结分享介绍接口测试框架开发&#xff0c;环境使用python3selenium3unittestddtrequests测试框架及ddt数据驱动&#xff0c;采用Excel管理测试用例等集成测试数据功能&#xff0c;以及使用HTMLTestRunner来生成测试报告&#xff0c;目前有开源的poman、Jmeter等接口测试工…

文件的上传和下载

一、node实现文件上传 1、FormData对象&#xff1a;以对象的方式来表示页面中的表单&#xff0c;又称为表单对象。以key-value的方式来保存数据&#xff0c;XMLHttpRequest对 象可以轻松的表单对象发送的服务器端 ​ &#xff08;1&#xff09;是一个构造函数&#xff1a;ne…

力扣hot100——第4天:19删除链表的倒数第N个节点、20有效的括号、21合并两个有序链表

文章目录1.19删除链表的倒数第N个节点【代码随想录已刷】2.20有效的括号【代码随想录已刷】3.21合并两个有序链表3.1.题目3.2.题解1.19删除链表的倒数第N个节点【代码随想录已刷】 参考&#xff1a;力扣题目链接&#xff1b;自己的博客解答 2.20有效的括号【代码随想录已刷】…

whistle监听方法

视频教程 程序员抓包神器&#xff0c;快速定位线上bug_哔哩哔哩_bilibili whistle官网 关于whistle GitBook 安装whistle后&#xff0c; 复制启动网址 系统代理也设置完毕后&#xff0c;在浏览器打开http://192.168.4.238:8899,即可进行抓包 拦截js&#xff0c;并执行自定…

10年的老测试告诉你八大测试用例设计方法

一&#xff1a;等价类划分法 1:有效等价类: 2:无效等价类: 案例:比如一个登陆输入框,规定只能输入中文,同时长度为6-10。 通过等价类设计测试用例: 测试用例中重要的三步: 输入 操作 预计结果 如果与预期结果不符合就是bug。 有效等价类: 输入:输入长度为6的中文,输入的为…

微信小程序接口请求多文件+参数上传、单文件+参数上传(formData形式) 微信小程序实现formData格式传参(亲测有效)

01.引入所需formData js文件 1.文件链接 链接: https://pan.baidu.com/s/1BDxx0-1KMAnkceXb45L5rg 提取码: 6ibp 2.引入使用 const FormData require(../../../../utils/formData.js)formData.js与mimeMap.js请确保在同一层级 02.formData参数 1.参数设置 let data {a…

Vue3 Composition API(案例)

前言&#xff1a;如果你是从vue2转到vue3的一份子&#xff0c;那么你重点学一下核心内容Composition API 。vue3的更新使代码写起来更加清晰&#xff0c;而且更接近于原生开发&#xff0c;对TS支持友好,现在我们来学习一下API。 如果你也想学一下TS那么请看这里 带你学习语法T…

力扣(LeetCode)1769. 移动所有球到每个盒子所需的最小操作数(C++)

暴力循环 直观模拟&#xff0c;对于某个固定的盒子&#xff0c;可以遍历所有盒子&#xff0c;∑\sum∑ 遍历的盒子里的球数 \times 遍历的盒子到固定的盒子的距离&#xff0c;得移动所有球到固定盒子的最小操作数。依次固定所有盒子&#xff0c;遍历&#xff0c;得到答案。 c…

嵌入式Linux 开发经验:platform_driver_register 的使用方法

前言 嵌入式Linux 设备驱动开发时&#xff0c;经常遇到平台驱动 platform_driver_register 的注册&#xff0c;最近深入了看了驱动开发为何使用平台驱动 开发一个设备驱动时&#xff0c;为了实现 设备的 打开、关闭、控制等操作&#xff0c;可以注册为 Linux misc 设备&#x…

Ra-08透传固件应用

目录1、功能介绍2、硬件接线3、固件烧录4、应用说明指令说明应用示例5、联系我们1、功能介绍 Ra-08透传固件主要功能有&#xff0c;设置发送或者接收模式&#xff0c;配置各个射频参数&#xff0c;设置本地地址与发送的目标地址&#xff0c;设置进入睡眠模式等。 2、硬件接线…