当前位置: 首页 > news >正文

OSG开发笔记(三十三):同时观察物体不同角度的多视图从相机技术

​若该文为原创文章,未经允许不得转载
本文章博客地址:https://blog.csdn.net/qq21497936/article/details/143932273
各位读者,知识无穷而人力有穷,要么改需求,要么找专业人士,要么自己研究

长沙红胖子Qt(长沙创微智科)博文大全:开发技术集合(包含Qt实用技术、树莓派、三维、OpenCV、OpenGL、ffmpeg、OSG、单片机、软硬结合等等)持续更新中…

OSG开发专栏(点击传送门)

上一篇:《OSG开发笔记(三十二):深入理解相机视口、制作支持与主视图同步变换旋转的相机HUD》
下一篇:《OSG开发笔记(三十四): OsgUtil::Simplifier:简化几何体,提升显示性能和渲染效率》


前言

  前面的相机hud可以单独显示图形,继续深入研究相机hud,技术就是子视图了,实现该功能的直接技术是从相机技术。
  本篇描述osg从相机技术


Demo

  请添加图片描述

  请添加图片描述


相机视口的关键调用

是否清除颜色深度缓存(清除)

pCamera->setClearMask(GL_DEPTH_BUFFER_BIT);

  如果不清除颜色缓存,渲染的窗口中若无内容,则将其他窗口渲染的内容显示到当前窗口。

设置渲染顺序(最后渲染)

// 设置POST渲染顺序(最后渲染)
pCamera->setRenderOrder(osg::Camera::POST_RENDER);

  后渲染的优先级比较高(最后显示,显示优先级最高)。

设置是否接受事件(不接受)

// 设置为不接收事件,始终得不到焦点
pCamera->setAllowEventFocus(false);

设置视口大小

// 视口就是引擎三维的区域,但是注意区别于屏幕的坐标系(屏幕是左上为0,0,而三维引擎是左下为0,0)
pSlaveFrontCamera->setViewport(0,0,rect().width() / 4,rect().height() / 4);

设置从相机故过程

步骤一:新建相机

osg::ref_ptr<osg::Camera> pSlaveFrontCamera = new osg::Camera;

步骤二:设置上下文

pSlaveFrontCamera->setGraphicsContext(_pViewer->getWindow());

步骤三:设置视图区域

// 视口就是引擎三维的区域,但是注意区别于屏幕的坐标系(屏幕是左上为0,0,而三维引擎是左下为0,0)
pSlaveFrontCamera->setViewport(0,0,rect().width() / 4,rect().height() / 4);

步骤四:设置渲染顺序

pSlaveFrontCamera->setRenderOrder(osg::Camera::POST_RENDER);

步骤五:关键步骤添加从相机

  第二个参数是缩放矩阵,第三个参数是旋转矩阵

_pViewer->addSlave(pSlaveFrontCamera,osg::Matrix(),osg::Matrix::rotate(osg::DegreesToRadians(0.0), 0.0, 0.0, 0.0),true);

Demo关键源码

osg::ref_ptr<osg::Node> OsgWidget::getMulViewCameraNode()
{// 隐藏整个demo全局的按钮面板(没用到按键的直接隐藏,不影响此Demo){ui->groupBox_pannel->setVisible(false);ui->label_cursor->setVisible(false);ui->label_cursor_2->setVisible(false);ui->label_msg->setVisible(false);ui->label_state->setVisible(false);}osg::ref_ptr<osg::Group> pGroup = new osg::Group;// 绘制盒体(立方体、长方体){osg::ref_ptr<osg::Geode> pGeode = new osg::Geode;// 创建专门指明精细度的类osg::TessellationHints,并设置对应精细度osg::ref_ptr<osg::TessellationHints> pHints = new osg::TessellationHints;pHints->setDetailRatio(0.5);// 绘制几何类型(几何体)qreal width = 5.0f;// 函数1pGeode->addDrawable(new osg::ShapeDrawable(new osg::Box(osg::Vec3(0, 0, 0), width), pHints));
#if 1// 设置关闭光照:OFF,同时旋转都能看到了(光照关闭,法向量不起作用){osg::StateSet *pStateSet = pGeode->getOrCreateStateSet();pStateSet->setMode(GL_LIGHTING, osg::StateAttribute::ON);
//            pStateSet->setMode(GL_LIGHTING, osg::StateAttribute::OFF);}
#endifpGroup->addChild(pGeode);}// 创建多视口相机{
#if 0// 这里改用了自己窗口已经创建的,这块废掉了,但是保留,基本的核心思想是一样的osg::ref_ptr<osg::GraphicsContext::WindowingSystemInterface> pWindowingSystemInterface= osg::GraphicsContext::getWindowingSystemInterface();if(!pWindowingSystemInterface.get()){LOG << "if(!pWindowingSystemInterface.get())";return pGroup.get();}unsigned int width = 0;unsigned int height = 0;pWindowingSystemInterface->getScreenResolution(osg::GraphicsContext::ScreenIdentifier(0),width,height);osg::ref_ptr<osg::GraphicsContext::Traits> pTraits = new osg::GraphicsContext::Traits;{pTraits->x = 0;pTraits->y = 0;pTraits->width = width;pTraits->height = height;pTraits->windowDecoration = false;pTraits->doubleBuffer = true;pTraits->sharedContext = 0;}LOG << pTraits->x << pTraits->y << pTraits->width << pTraits->height;osg::ref_ptr<osg::GraphicsContext> pGraphicsContext = osg::GraphicsContext::createGraphicsContext(pTraits.get());if(!pGraphicsContext->valid()){LOG << "if(!pGraphicsContext->valid())";return pGroup.get();}
#endifdouble angle = 15.0f;#if 1// 前osg::ref_ptr<osg::Camera> pSlaveFrontCamera = new osg::Camera;pSlaveFrontCamera->setGraphicsContext(_pViewer->getWindow());// 视口就是引擎三维的区域,但是注意区别于屏幕的坐标系(屏幕是左上为0,0,而三维引擎是左下为0,0)pSlaveFrontCamera->setViewport(0,0,rect().width() / 4,rect().height() / 4);pSlaveFrontCamera->setRenderOrder(osg::Camera::POST_RENDER);_pViewer->addSlave(pSlaveFrontCamera,osg::Matrix(),osg::Matrix::rotate(osg::DegreesToRadians(0.0), 0.0, 0.0, 0.0),true);
#endif#if 1// 后osg::ref_ptr<osg::Camera> pSlaveBehindCamera = new osg::Camera;pSlaveBehindCamera->setGraphicsContext(_pViewer->getWindow());// 视口就是引擎三维的区域,但是注意区别于屏幕的坐标系(屏幕是左上为0,0,而三维引擎是左下为0,0)pSlaveBehindCamera->setViewport(0,rect().width() / 4 * 3,rect().width() / 4,rect().height() / 4);pSlaveBehindCamera->setRenderOrder(osg::Camera::POST_RENDER);_pViewer->addSlave(pSlaveBehindCamera,osg::Matrix::translate(0, 0, 0),osg::Matrix::rotate(osg::DegreesToRadians(30), 1.0, 0.0, 0.0),true);
#endif#if 0// 左
//        osg::ref_ptr<osg::Camera> pSlaveLeftCamera = new osg::Camera;
//        pSlaveLeftCamera->setGraphicsContext(_pViewer->getWindow());osg::ref_ptr<HudRotateCamera> pSlaveLeftCamera = new HudRotateCamera;pSlaveLeftCamera->setGraphicsContext(_pViewer->getWindow());pSlaveLeftCamera->setMasterCamera(_pViewer->getCamera());pSlaveLeftCamera->setViewport(0,rect().height() / 8 * 3,rect().width()  / 4,rect().height() / 4);pSlaveLeftCamera->setRenderOrder(osg::Camera::POST_RENDER);#if 0_pViewer->addSlave(pSlaveLeftCamera,osg::Matrix(),osg::Matrix::rotate(osg::DegreesToRadians(angle), 0.0, 0.0, 1.0),true);
#endif
#if 0// 设置相机位置,观察目标点和方向osg::Vec3f vec3Eye = osg::Vec3f(100, 100, 0);osg::Vec3f vec3Center = osg::Vec3f(0, 0, 0);osg::Vec3f vec3Up = osg::Vec3f(0, 1, 0);pSlaveLeftCamera->setViewMatrixAsLookAt(vec3Eye, vec3Center, vec3Up);_pViewer->addSlave(pSlaveLeftCamera);
#endif
#if 1// 设置slave相机的位置和方向osg::ref_ptr<osg::MatrixTransform> transform = new osg::MatrixTransform();// 设置平移矩阵,根据需要进行调整osg::Matrix matrix;matrix.makeTranslate(0, 0, 0); // 这里的x, y, z是你想要平移到的位置transform->setMatrix(matrix);transform->addChild(pGroup);_pViewer->addSlave(pSlaveLeftCamera,osg::Matrix(),osg::Matrix::rotate(osg::DegreesToRadians(15.0f), 0.0, 1.0, 1.0),true);pSlaveLeftCamera->setProjectionMatrixAsPerspective(100.0f, 1.0, 1.0f, 100.0f);
#endif#endif#if 0// 右osg::ref_ptr<osg::Camera> pSlaveRightCamera = new osg::Camera;pSlaveRightCamera->setGraphicsContext(_pViewer->getWindow());pSlaveRightCamera->setViewport(rect().width()  / 4 * 3,rect().height() / 8 * 3,rect().width()  / 4,rect().height() / 4);pSlaveRightCamera->setRenderOrder(osg::Camera::POST_RENDER);_pViewer->addSlave(pSlaveRightCamera,osg::Matrix(),osg::Matrix::rotate(osg::DegreesToRadians(-angle), 0.0, 0.0, 1.0),true);
#endif#if 0// 上osg::ref_ptr<osg::Camera> pSlaveUpCamera = new osg::Camera;pSlaveUpCamera->setGraphicsContext(_pViewer->getWindow());pSlaveUpCamera->setViewport(rect().width()  / 8 * 3,0,rect().width()  / 4,rect().height() / 4);pSlaveUpCamera->setRenderOrder(osg::Camera::POST_RENDER);_pViewer->addSlave(pSlaveUpCamera,osg::Matrix(),osg::Matrix::rotate(osg::DegreesToRadians(angle), 1.0, 0.0, 0.0),true);
#endif#if 0// 下osg::ref_ptr<osg::Camera> pSlaveDownCamera = new osg::Camera;pSlaveDownCamera->setGraphicsContext(_pViewer->getWindow());pSlaveDownCamera->setViewport(rect().height() / 8 * 3,rect().height() / 8 * 6,rect().width() / 4,rect().height() / 4);pSlaveDownCamera->setRenderOrder(osg::Camera::POST_RENDER);_pViewer->addSlave(pSlaveDownCamera,osg::Matrix(),osg::Matrix::rotate(osg::DegreesToRadians(-angle), 1.0, 0.0, 0.0),true);
#endif}return pGroup.get();
}

工程模板v1.36.0

  在这里插入图片描述


入坑

入坑一:设置相机就崩溃

问题

  在这里插入图片描述

解决过程

  定位到不设置相机就不崩溃,然后这里是笔者自己造的Qt与OSG结合的,使用了位置,这里也可以查看实际打印的创建的区域坐标和大小,确实也是不对:
  在这里插入图片描述

  那直接把场景里面的gc赋值给他测试,是可以的,修改的地方有点多,因为这个Qt+OSG是笔者根据源码原理进行调整渲染的,与直接编译出来的qt+osg还是有点区别,总之一句话,就是Qt渲染窗口里面已经有这个osg::ref_ptrosg::GraphicsContext了,不用去额外建立了:
  删除以下代码:
  在这里插入图片描述

  然后再调整相机代码,还有从Qt渲染窗口里面增加拿到这个内容上下文的函数就好了。

解决

  新增获取函数,原本不能获取
  在这里插入图片描述

  在这里插入图片描述

  这里实际大小为:
  在这里插入图片描述

  所以外面代码,直接用窗口的宽高好了(笔者是铺满的):这里是要缩小放前面,那就是改为4/1吧:
  在这里插入图片描述

  在这里插入图片描述

入坑二:左视图没有

问题

  左视图应该显示,但是没显示
  在这里插入图片描述

解决过程

  改成一样的:
  在这里插入图片描述

  然后不偏移试试:
  在这里插入图片描述

  偏移一个小角度试试:
  在这里插入图片描述

  所以是Y轴的中心不对,但是我们也没有改,测试绕x轴:
  在这里插入图片描述

  然后绕z轴,发现就z轴没有偏移:
  在这里插入图片描述

  在这里插入图片描述

  在这里插入图片描述

  尝试单独设置添加相机的视口是无效的:
  在这里插入图片描述

  在这里插入图片描述

  尝试单独修改同步旋转的相机去修改视口,也是无效:
  在这里插入图片描述

  继续尝试:
  在这里插入图片描述

  是否与内置相机的视口有关系,测试也无关:
  在这里插入图片描述

  在这里插入图片描述

解决(主技术方向未解决)

  从原始从相机技术方面暂时没有解决,因为也尝试了更改矩阵、修改相机视角观看位置都没什么变化。
  可以确认的是,应该是相机旋转的中心不对,并不是场景中心不对,所以鼠标拽托中间还是在旋转,而视角旋转则x和y轴存在偏移。X和y存在偏移就是左右和里外,若是与屏幕有关也是上下和左右,所以这里这么分析推断也不对。
  代码全部放出,读者有兴趣可以提供协助,一起探讨。

规避解决方法

  直接在相机中修改偏移旋转,然后当作结点加入,是可以解决,而且还不能是从相机,需要addChild进入:
  在这里插入图片描述

  这时候拉伸有问题:
  在这里插入图片描述

  变形了:
  在这里插入图片描述

  在这里插入图片描述

  终于外挂一个东西解决:
  在这里插入图片描述

  在这里插入图片描述

  但是鼠标中键按下偏移中心点,会都向右,理论上反向180°的y轴应该向左,但是还是向右,因为是场景偏移,我们规避是对场景下的相机进行旋转,所以实际是移动场景相机了,相机里面的正反对外无效。

官方从相机示例(也存在问题,怀疑是osg3.4.0源码bug)

  为了再次深入论证是否代码问题,笔者又用官方的demo实现:

#include <osg/Camera>
#include <osg/Group>
#include <osg/Geode>
#include <osg/ShapeDrawable>
#include <osgViewer/Viewer>
#include <osg/GraphicsContext>int main(int argc, char *argv[])
{osg::ref_ptr<osg::Group> pGroup = new osg::Group;// 绘制盒体(立方体、长方体){osg::ref_ptr<osg::Geode> pGeode = new osg::Geode;// 创建专门指明精细度的类osg::TessellationHints,并设置对应精细度osg::ref_ptr<osg::TessellationHints> pHints = new osg::TessellationHints;pHints->setDetailRatio(0.5);// 绘制几何类型(几何体)double width = 5.0f;// 函数1pGeode->addDrawable(new osg::ShapeDrawable(new osg::Box(osg::Vec3(0, 0, 0), width), pHints));// 设置关闭光照:OFF,同时旋转都能看到了(光照关闭,法向量不起作用){osg::StateSet *pStateSet = pGeode->getOrCreateStateSet();pStateSet->setMode(GL_LIGHTING, osg::StateAttribute::ON);}pGroup->addChild(pGeode);}osg::GraphicsContext::WindowingSystemInterface * wsi = osg::GraphicsContext::getWindowingSystemInterface();if(!wsi){return 0;}osg::ref_ptr<osg::GraphicsContext::Traits > traits = new osg::GraphicsContext::Traits;traits->x = 0;traits->y = 0;traits->width = 800;traits->height = 600;traits->windowDecoration = false;traits->doubleBuffer = true;traits->sharedContext = 0;osg::ref_ptr<osg::GraphicsContext> gc = osg::GraphicsContext::createGraphicsContext(traits);if(!gc.valid()){return 0;}osg::ref_ptr<osgViewer::Viewer> viewer = new osgViewer::Viewer;osg::ref_ptr<osg::Camera> master = new osg::Camera;master->setGraphicsContext(gc);master->setViewport(0, 0, 800, 600);viewer->addSlave(master.get());osg::ref_ptr<osg::Camera> leftcam = new osg::Camera;leftcam->setGraphicsContext(gc);leftcam->setViewport(0, 0, 800 / 2, 600 / 2);leftcam->setRenderOrder(osg::Camera::POST_RENDER);viewer->addSlave(leftcam.get(),osg::Matrix(),osg::Matrix::rotate(osg::DegreesToRadians(15.0), 0.0, 1.0, 0.0),true);viewer->setSceneData(pGroup);viewer->run();return 0;
}

  在这里插入图片描述

  改进后相机代码:

    osg::ref_ptr<osgViewer::Viewer> viewer = new osgViewer::Viewer;osg::ref_ptr<osg::Camera> master = new osg::Camera;master->setGraphicsContext(gc);master->setViewport(0, 0, 800, 800);viewer->addSlave(master.get());{osg::ref_ptr<osg::Camera> leftcam = new osg::Camera;leftcam->setGraphicsContext(gc);leftcam->setViewport(0, 0, 100, 100);leftcam->setRenderOrder(osg::Camera::POST_RENDER);viewer->addSlave(leftcam.get(),osg::Matrix(),osg::Matrix::rotate(osg::DegreesToRadians(15.0), 1.0, 0.0, 0.0),true);}{osg::ref_ptr<osg::Camera> leftcam = new osg::Camera;leftcam->setGraphicsContext(gc);leftcam->setViewport(100, 0, 100, 100);leftcam->setRenderOrder(osg::Camera::POST_RENDER);viewer->addSlave(leftcam.get(),osg::Matrix(),osg::Matrix::rotate(osg::DegreesToRadians(15.0), 0.0, 1.0, 0.0),true);}{osg::ref_ptr<osg::Camera> leftcam = new osg::Camera;leftcam->setGraphicsContext(gc);leftcam->setViewport(200, 0, 100, 100);leftcam->setRenderOrder(osg::Camera::POST_RENDER);viewer->addSlave(leftcam.get(),osg::Matrix(),osg::Matrix::rotate(osg::DegreesToRadians(15.0), 0.0, 0.0, 1.0),true);}{osg::ref_ptr<osg::Camera> leftcam = new osg::Camera;leftcam->setGraphicsContext(gc);leftcam->setViewport(300, 0, 100, 100);leftcam->setRenderOrder(osg::Camera::POST_RENDER);viewer->addSlave(leftcam.get(),osg::Matrix(),osg::Matrix::rotate(osg::DegreesToRadians(180.0), 0.0, 0.0, 1.0),true);}

  所以是osg3.4.0的源码这块就有问题:
  在这里插入图片描述


上一篇:《OSG开发笔记(三十二):深入理解相机视口、制作支持与主视图同步变换旋转的相机HUD》
下一篇:《OSG开发笔记(三十四): OsgUtil::Simplifier:简化几何体,提升显示性能和渲染效率》


本文章博客地址:https://blog.csdn.net/qq21497936/article/details/143932273

相关文章:

OSG开发笔记(三十三):同时观察物体不同角度的多视图从相机技术

​若该文为原创文章&#xff0c;未经允许不得转载 本文章博客地址&#xff1a;https://blog.csdn.net/qq21497936/article/details/143932273 各位读者&#xff0c;知识无穷而人力有穷&#xff0c;要么改需求&#xff0c;要么找专业人士&#xff0c;要么自己研究 长沙红胖子Qt…...

模糊逻辑学习 | 模糊推理 | 模糊逻辑控制

注&#xff1a;本文为几位功夫博主关于 “模糊逻辑学习 / 推理 / 控制” 的相关几篇文章合辑。 初学模糊逻辑控制&#xff08;Fuzzy Logic Control&#xff09; ziqian__ 已于 2022-08-19 20:30:25 修改 一、前言 模糊逻辑控制&#xff08;Fuzzy Logic Control&#xff09;是…...

【JavaEE】Servlet:表白墙

文章目录 一、前端二、前置知识三、代码1、后端2、前端3、总结 四、存入数据库1、引入 mysql 的依赖&#xff0c;mysql 驱动包2、创建数据库数据表3、调整上述后端代码3.1 封装数据库操作&#xff0c;和数据库建立连接3.2 调整后端代码 一、前端 <!DOCTYPE html> <ht…...

C++特殊类设计(不能被拷贝的类、只能在堆上创建对象的类、不能被继承的类、单例模式)

C特殊类设计 在实际应用中&#xff0c;可能需要设计一些特殊的类对象&#xff0c;如不能被拷贝的类、只能在堆上创建对象的类、只能在栈上创建对象的类、不能被继承的类、只能创建一个对象的类&#xff08;单例模式&#xff09;。 1. 不能被拷贝的类 拷贝只会发生在两个场景…...

【小白学机器学习34】用python进行基础的数据统计 mean,var,std,median,mode ,四分位数等

目录 1 用 numpy 快速求数组的各种统计量&#xff1a;mean, var, std 1.1 数据准备 1.2 直接用np的公式求解 1.3 注意问题 1.4 用print() 输出内容&#xff0c;显示效果 2 为了验证公式的后背&#xff0c;下面是详细的展开公式的求法 2.1 均值mean的详细 2.2 方差var的…...

安装 Docker(使用国内源)

一、安装Docker-ce 1、下载阿里云的repo源 [rootlocalhost ~]# yum install yum-utils -y && yum-config-manager --add-repo https://mirrors.aliyun.com/docker-ce/linux/centos/docker-ce.repo && yum makecache # 尝试列出 docker-ce 的版本 [rootlocalh…...

Ajax学习笔记,第一节:语法基础

Ajax学习笔记&#xff0c;第一节&#xff1a;语法基础 一、概念 1、什么是Ajax 使用浏览器的 XMLHttpRequest 对象 与服务器通信2、什么是axios Axios是一个基于Promise的JavaScript库&#xff0c;支持在浏览器和Node.js环境中使用。相较于Ajax&#xff0c;Axios提供了更多…...

《用Python画蔡徐坤:艺术与编程的结合》

简介 大家好&#xff01;今天带来一篇有趣的Python编程项目&#xff0c;用代码画出知名偶像蔡徐坤的形象。这个项目使用了Python的turtle库&#xff0c;通过简单的几何图形和精心设计的代码来展示艺术与编程的结合。 以下是完整的代码和效果介绍&#xff0c;快来试试看吧&…...

Unity中动态生成贴图并保存成png图片实现

实现原理&#xff1a; 要生成长x宽y的贴图&#xff0c;就是生成x*y个像素填充到贴图中&#xff0c;如下图&#xff1a; 如果要改变局部颜色&#xff0c;就是从x1到x2(x1<x2),y1到y2(y1<y2)这个范围做处理&#xff0c; 或者要想做圆形就是计算距某个点&#xff08;x1,y1&…...

Mac配置maven环境及在IDEA中配置Maven

Mac配置maven环境及在IDEA中配置Maven 1. 介绍 Maven是一款广泛用于Java等JVM语言项目的工具&#xff0c;它以项目对象模型&#xff08;POM&#xff09;为基础进行项目管理&#xff0c;通过POM文件来定义项目信息和依赖关系。同时&#xff0c;它也是构建自动化工具&#xff0…...

Reactor 模式的理论与实践

1. 引言 1.1 什么是 Reactor 模式&#xff1f; Reactor 模式是一种用于处理高性能 I/O 的设计模式&#xff0c;专注于通过非阻塞 I/O 和事件驱动机制实现高并发性能。它的核心思想是将 I/O 操作的事件分离出来&#xff0c;通过事件分发器&#xff08;Reactor&#xff09;将事…...

vim 一次注释多行 的几种方法

在 Vim 中一次注释多行是一个常见操作。可以使用以下方法根据你的具体需求选择合适的方式&#xff1a; 方法 1&#xff1a;手动插入注释符 进入正常模式&#xff1a; 按 Esc 确保进入正常模式。 选择需要注释的多行&#xff1a; 移动到第一行&#xff0c;按下 Ctrlv 进入可视块…...

问题记录-Java后端

问题记录 目录 问题记录1.多数据源使用事务注意事项&#xff1f;2.mybatis执行MySQL的存储过程&#xff1f;3.springBoot加载不到nacos配置中心的配置问题4.服务器产生大量close_wait情况 1.多数据源使用事务注意事项&#xff1f; 问题&#xff1a;在springBoot项目中多表处理数…...

李春葆《数据结构》-课后习题代码题

一&#xff1a;假设不带权有向图采用邻接矩阵 g 存储&#xff0c;设计实现以下功能的算法&#xff1a; &#xff08;1&#xff09;求出图中每个顶点的入度。 代码&#xff1a; void indegree(MatGraph g){int i,j,n;printf("各个顶点的入度&#xff1a;\n");for(i…...

51c~C语言~合集2

我自己的原文哦~ https://blog.51cto.com/whaosoft/12652943 一、嵌入式开发中的C语言编译器 如果你和一个优秀的程序员共事&#xff0c;你会发现他对他使用的工具非常熟悉&#xff0c;就像一个画家了解他的画具一样。----比尔.盖茨1 不能简单的认为是个工具 嵌入式程序开发…...

【Python】构建事件驱动架构:用Python实现实时应用的高效系统

解锁Python编程的无限可能:《奇妙的Python》带你漫游代码世界 事件驱动架构(Event-Driven Architecture,EDA)是一种基于事件流动进行系统设计的模式,广泛应用于游戏开发、实时监控和分布式系统中。它通过解耦事件的生产者和消费者,提升系统的可扩展性和灵活性。本文章从…...

Git(一)基本使用

目录 一、使用git -v 查看安装git版本 二、使用mkdir 创建一个文件&#xff0c;并使用 git init 在该目录下创建一个本地仓库&#xff0c; 三、通过git clone命令接入线上仓库 四、使用git status查看仓库状态信息 五、利用echo写入一个文件 并使用cat进行查看 【Linux】e…...

HarmonyOS应用开发者基础认证,Next版本发布后最新题库(10月8日更新题库未收录)

笔者会尽量找到答案的出处&#xff0c;力求答案准确无误。有些题目答案可能有错&#xff0c;也有一些笔者实在找不到出处&#xff0c;也不知道答案的&#xff0c;如果读者发现错误或有补充建议&#xff0c;欢迎评论或私信笔者。您的每一条反馈都是宝贵的&#xff0c;能够帮助笔…...

【PGCCC】Postgresql BRIN 索引原理

前言 postgresql 提供了块级索引&#xff08;简称 BRIN&#xff09;&#xff0c;主要适用于类似时序数据之类的&#xff0c;有着天然的顺序&#xff0c;而且都是添加写的场景。相比于 btree 索引&#xff0c;它的体积小得多&#xff0c;非常适用于大数据量的场景。 原理 pos…...

腾讯云 AI 代码助手:产品研发过程的思考和方法论

一、文章摘要 本文将详细阐述 腾讯云 AI 代码助手的历史发展形态与产品整体架构&#xff0c;并从技术、研发方法论的角度分别阐述了产品的研发过程。 全文阅读约 5&#xff5e;8 分钟。 二、产品布局 AI 代码助手产品经历了三个时代的发展 第一代诸如 Eclipse、Jetbrains、V…...

Matlab 深度学习 PINN测试与学习

PINN 与传统神经网络的区别 与传统神经网络的不同之处在于&#xff0c;PINN 能够以微分方程形式纳入有关问题的先验专业知识。这些附加信息使 PINN 能够在给定的测量数据之外作出更准确的预测。此外&#xff0c;额外的物理知识还能在存在含噪测量数据的情况下对预测解进行正则…...

【Angular】async详解

在 Angular 中&#xff0c;async 关键字用于定义异步函数&#xff0c;通常与 await 一起使用来处理 Promise。这使得异步代码看起来更像同步代码&#xff0c;从而更容易理解和维护。 基本用法 定义异步函数&#xff1a;使用 async 关键字。等待 Promise 解析&#xff1a;使用…...

抖音SEO矩阵系统:开发技术分享

市场环境剖析 短视频SEO矩阵系统是一种策略&#xff0c;旨在通过不同平台上的多个账号建立联系&#xff0c;整合同一品牌下的各平台粉丝流量。该系统通过遵循每个平台的规则和内容要求&#xff0c;输出企业和品牌形象&#xff0c;以矩阵形式增强粉丝基础并提升商业价值。抖音作…...

SpringBoot集成minio,并实现文件上传

SpringBoot集成minio 什么是minioSpringBoot集成minio1、引入minio依赖2、配置Minio相关参数3、在代码里读取自定义的minio配置4、在minio配置类里,注册ConfigurationProperties实现文件上传到minio1、利用SpringMVC实现接口的异常全局处理2、返回文件路径给前端3、返回文件流…...

centos为用户赋予sudo权限

在CentOS系统中&#xff0c;要为用户test赋予sudo权限&#xff0c;你需要按照以下步骤操作&#xff1a; 确保sudo包已安装&#xff1a; 如果系统中没有安装sudo&#xff0c;你可以通过yum&#xff08;CentOS 7及以下&#xff09;或dnf&#xff08;CentOS 8及以上&#xff09;来…...

SAP 零售方案 CAR 系统的介绍与研究

前言 当今时代&#xff0c;零售业务是充满活力和活力的业务领域之一。每天&#xff0c;由于销售运营和客户行为&#xff0c;它都会生成大量数据。因此&#xff0c;公司迫切需要管理数据并从中检索见解。它将帮助公司朝着正确的方向发展他们的业务。 这就是为什么公司用来处理…...

Android Framework AudioFlinge 面试题及参考答案

目录 请解释什么是 AudioFlinger? AudioFlinger 在 Android 系统中的位置是什么? AudioFlinger 的主要职责有哪些? AudioFlinger 如何管理音频流? 在 AudioFlinger 中,什么是音频会话? 请简述 AudioFlinger 的工作流程。 AudioFlinger 是如何与硬件交互的? 在 A…...

嵌入式系统与单片机工作原理详解

随着现代科技的发展&#xff0c;嵌入式系统已经深入到我们日常生活中的方方面面。无论是智能家居、汽车电子&#xff0c;还是工业控制、医疗设备&#xff0c;都离不开嵌入式系统的支持。而单片机作为嵌入式系统的核心组件&#xff0c;是实现这些功能的关键之一。本文将详细介绍…...

Diving into the STM32 HAL-----Timers笔记

嵌入式设备会按时间执行某些活动。对于真正简单且不准确的延迟&#xff0c;繁忙的循环可以执行任务&#xff0c;但是使用 CPU 内核执行与时间相关的活动从来都不是一个聪明的解决方案。因此&#xff0c;所有微控制器都提供专用的硬件外设&#xff1a;定时器。定时器不仅是时基生…...

对比 MyBatis 批处理 BATCH 模式与 INSERT INTO ... SELECT ... UNION ALL 进行批量插入

前言 在开发中&#xff0c;我们经常需要批量插入大量数据。不同的批量插入方法有不同的优缺点&#xff0c;适用于不同的场景。本文将详细对比两种常见的批量插入方法&#xff1a; MyBatis 的批处理模式。使用 INSERT INTO ... SELECT ... UNION ALL 进行批量插入。 MyBatis …...