3dtiles平移旋转工具制作

3dtiles平移旋转缩放原理及可视化工具实现

背景

平时工作中,通过cesium平台来搭建一个演示场景是很常见的事情。一般来说,演示场景不需要多完善的功能,但是需要一批三维模型搭建,如厂房、电力设备、园区等。在实际搭建过程中,就会面临一个尴尬的问题,那就是模型定位,常规操作中,我们一般用三维模型的中心点对应一个经纬度坐标,以此转换成3dtiles格式,但是给定的经纬度坐标一般是模糊的大致位置,甚至有的场景不需要准确的坐标,只需要你找个合适的场景把各个模型搭建起来,这就不得不对模型进行微调位置,以适应场景。

  • cesium中有对模型进行调整的代码,事实上,如果模型定位的位置不是我们想要的位置,可以通过修改该3dtiles的矩阵,来改变它,如平移:

    /**基于本地的ENU坐标系的偏移,也就是垂直于地表向上为Z,东为X,北为Y* @param tileset Cesium3DTileset* @param dx x轴偏移量。单位:米* @param dy y轴偏移量。单位:米* @param dz z轴偏移量。单位:米*/
    function translate(tileset: Cesium3DTileset, dx: number, dy: number, dz: number) {if (dx === 0 && dy === 0 && dz === 0) return// 对于3DTileset,我们需要的结果是一个模型矩阵,那么平移就是计算一个世界坐标下的平移矩阵。// 获取中心点const origin = tileset.boundingSphere.center// 以该点建立ENU坐标系const toWorldMatrix = Transforms.eastNorthUpToFixedFrame(origin)// 该坐标系下平移后的位置const translatePosition = new Cartesian3(dx, dy, dz)// 获取平移后位置的世界坐标const worldPosition = Matrix4.multiplyByPoint(toWorldMatrix, translatePosition, new Cartesian3())// 计算世界坐标下的各个平移量const offset = Cartesian3.subtract(worldPosition, origin, new Cartesian3())// 从世界坐标下的平移量计算世界坐标的平移矩阵const translateMatrix = Matrix4.fromTranslation(offset)// 应用平移矩阵。这里应该与原本的模型矩阵点乘,而不是直接赋值tileset.modelMatrix = Matrix4.multiply(translateMatrix, tileset.modelMatrix, new Matrix4())
    }
    
    • 但是实际开发中,我们把一个模型准确的放在我们想要的位置上,或者调整其与底图对齐,需要多次进行一点一点的矫正,也就是说需要多次调用该函数,调整xyz平移参数,这样做十分麻烦,因此最好能做一个可视化控件,能够直接通过拖住的形式来调整模型,最后得出模型调整后的位置矩阵,然后在新一轮代码中,直接将模型的位置改为该位置矩阵,即可完成模型的调整。

      在这里插入图片描述

功能开发

可视化控件

可视化控件,即我们看到的拖拽箭头,当平移时,是一个带箭头的三维坐标系的坐标轴,当是旋转时,是三个互相垂直的圆环;而这个可视化的开发,依靠cesium完全可以实现,因为cesium本身支持绘制图形,并且可以监听到鼠标是否划过和点击图形。

平移控件的显示
  • 首先控件的显示需要思考的问题是,控件需要多大,控件的位置在哪里,显然,控件的大小需要根据模型的大小来制定,而控件的位置肯定就是模型的位置,因此,很自然我们需要创建一个函数,来在创建控件前获取模型的位置和根据模型范围制定控件大小。

     /*** 初始化参数和清理工具*/private initParam(): TransformOption {this.removeAllTools();const b3dm = this._b3dm;const viewer = this._viewer;const length = b3dm.boundingSphere.radius / 3;const originalMatrix = this._b3dm.root.transform.clone()const ps = new Cesium.Cartesian3();Cesium.Matrix4.getTranslation(originalMatrix,ps)let pos = CoordTransform.transformCartesianToWGS84(viewer,ps)this._params = {...this._params,tx: pos.lng,ty: pos.lat,tz: pos.alt ,};return {originDegree:pos,length,};}
    

    _params记录了6个参数,分别是tx,ty,tz和rx,ry,rz,它们可以用来记录当前模型的位置和姿态变化

    • 有了初始的位置参数和范围之后,我们就可以构建坐标系,首先是坐标轴,在构建坐标轴的时候,需要考虑到,坐标轴是三个正交的方向,但是每一个方向指向哪里?比如X轴指向哪里?

      • 事实上,X轴指向任意一个方向,我们都能轻易的构建出一个三个方向正交的坐标系,但是显然,不符合我们的操作习惯。

      • 首先,先做一个假设,如果是根据Cesium的世界坐标系的方向来建立坐标轴,是否可行?显然是不行的,假设一个模型处于地球表面,我们的习惯,依然是想按照朝东为X轴,朝北为Y轴,朝上为Z轴来移动物体,这种习惯和我们的经纬度的习惯是符合的,东西走向为经度走向,南北走向为维度走向,那么我们让物体按X轴移动,就是沿着经度走,按Y轴移动,就是沿着维度走

        在这里插入图片描述

      • 而我们的Cesium的世界坐标系显然不是这种情况,如图所示,绿色坐标就是我们刚才说的坐标系,而蓝色坐标系,而是Cesium的世界坐标系,显然如果我们按照世界坐标系的方向建立坐标轴,当我们移动某个方向的时候,都会偏离出地球。

      在这里插入图片描述

  • 清楚了这一点,我们就可以按照图上绿色坐标系的形式建立坐标系,我们姑且称为局部坐标系,我们已经知道这个坐标系的原点的坐标了,也知道坐标系的范围了(可以理解为每个轴的长度),那么只需要求出每个轴的终点坐标就可以了,基于此,我们再创建一个getTransPosition函数,用来求终点坐标。

    	const { originDegree, length } = this.initParam();const translateCartesian = new Cesium.Cartesian3(length, length, length);const ps = CoordTransform.transformWGS84ToCartesian(this._viewer,originDegree)const targetDegree = this.getTransPosition(ps, translateCartesian);
    

    我们不需要求出每个轴的终点坐标,事实上只需要求出 translateCartesian这个向量的坐标就行了,然后每个轴的终点坐标,相当于向量坐标的分量,基于此,我们完善 getTransPosition函数

    /*** 根据平移距离获取目标点* @param originPosition - 原始位置(笛卡尔坐标)* @param translateCartesian - 平移向量(笛卡尔坐标)* @return 平移后的位置(经纬度坐标)*/private getTransPosition(originPosition: Cesium.Cartesian3,translateCartesian: Cesium.Cartesian3): { lng: number; lat: number; alt: number } {// 东-北-上参考系构造出4*4的矩阵const transform = Cesium.Transforms.eastNorthUpToFixedFrame(originPosition);//构造平移矩阵const m = new Cesium.Matrix4();Cesium.Matrix4.setTranslation(Cesium.Matrix4.IDENTITY,translateCartesian,m);//将当前位置矩阵乘以平移矩阵得到平移后的位置矩阵const modelMatrix = Cesium.Matrix4.multiply(transform, m, new Cesium.Matrix4());const finalPosition = new Cesium.Cartesian3();//从位置矩阵中获取坐标信息Cesium.Matrix4.getTranslation(modelMatrix, finalPosition);//转换为地理坐标系return CoordTransform.transformCartesianToWGS84(this._viewer,finalPosition);}
    

    这个思路很简单,首先根据原始位置创建一个局部坐标系,然后求出在局部坐标系下从(0,0,0)点平移到向量终点的平移矩阵,而局部坐标系下原点转到世界坐标系下也是一个矩阵,两个矩阵相乘即可求出向量终点在世界坐标系下的位置,也就是一个笛卡尔坐标,有了笛卡尔坐标,也就最终能算出向量终点的经纬度坐标

  • 我们可以分析一下向量终点的经纬度坐标,这里简称终点坐标,假设为(lng1,lat1,alt1),而我们原点的坐标是(lng0,lat0,alt0),那么很显然,我们能够得出,X轴的长度范围为 (lng0,lng1 ),其他轴也是如此。

  • 因此,我们获取到了坐标系的起点,终点,长度等信息,自然而然,下一步可以建立坐标系了 this.initLineArrow(originDegree, targetDegree, length);

     /*** 绘制坐标轴* @param originDegree -原始坐标(经纬度)* @param targetDegree -目标坐标(经纬度)* @param length - 坐标轴长度*/private initLineArrow(originDegree: { lng: number; lat: number; alt: number },targetDegree: { lng: number; lat: number; alt: number },length: number): void {const arrows = new Cesium.PolylineCollection();//x轴(红色)const xPos = [originDegree.lng,originDegree.lat,originDegree.alt,targetDegree.lng,originDegree.lat,originDegree.alt,];this.drawArrow(arrows, "model_edit_xArrow", xPos, Cesium.Color.RED);//y轴(绿色)const yPos = [originDegree.lng,originDegree.lat,originDegree.alt,originDegree.lng,targetDegree.lat,originDegree.alt,];this.drawArrow(arrows, "model_edit_yArrow", yPos, Cesium.Color.GREEN);//z轴(蓝色)const zPos = [originDegree.lng,originDegree.lat,originDegree.alt,originDegree.lng,originDegree.lat,targetDegree.alt,];this.drawArrow(arrows, "model_edit_zArrow", zPos, Cesium.Color.BLUE);this._coordArrows = this._viewer.scene.primitives.add(arrows);if(this._coordArrows){this._coordArrows._name = "CoordAxis";}}
    
  • 这里创建的x,y,z轴坐标明显就是两个点的连线,X轴是横跨经线的直线,Y轴是横跨纬线的直线,Z轴则是垂直地面的直线,且三个轴长度相等。然后,我们将每个轴坐标传入 drawArrow 函数,这里是最终实现坐标轴的代码,我们这里可以做一个思考,如果给每个坐标轴的终点,加上箭头,我们需要做两个操作,首先把坐标轴画出来,其次在该轴末尾画一个箭头,这样操作显然比较麻烦,尤其是画箭头,cesium有一个api已经实现了这个操作,只需要给出坐标,就能实现一条直线并且末尾带箭头,让我们完善这个代码

    /*** 绘制箭头* @param arrows - PolylineCollection集合* @param name - 箭头名称* @param positions - 箭头位置数组* @param color - 箭头颜色*/private drawArrow(arrows: Cesium.PolylineCollection,name: string,positions: number[],color: Cesium.Color): void {const arrow = arrows.add({positions: Cesium.Cartesian3.fromDegreesArrayHeights(positions),width: this._defaultWidth,material: Cesium.Material.fromType(Cesium.Material.PolylineArrowType, {color: color,}),}) as EditablePolyline;arrow._name = name;}
    

    在这里插入图片描述

至此,我们成功创建了坐标系轴,并且每个坐标系轴都赋予一个name属性,整个坐标系有一个整体name属性

在这里插入图片描述

旋转控件的实现

旋转控件的实现就比较简单,大致思路就是首先确定圆环的圆心(原点),其次插值求出一个圆环点,其次根据圆环点借助cesium中的api创建一个圆环,最后通过旋转把其它两个圆环创建出来,我们一步步剖析这个操作

  • 根据最开始求得的模型的位置坐标和范围创建圆环坐标点,这里以原始坐标为圆心,以范围为半径,进行插值求点

     public editRotation(): void {const { originDegree, length } = this.initParam();this.createCircle(originDegree.lng,originDegree.lat,originDegree.alt,length);}
    
    /*** 创建旋转圆环*/private createCircle(lon: number,lat: number,height: number,radius: number): void {const positions: Cesium.Cartesian3[] = [];//生成圆形点位for (let i = 0; i <= 360; i += 3) {const sin = Math.sin(Cesium.Math.toRadians(i));const cos = Math.cos(Cesium.Math.toRadians(i));positions.push(new Cesium.Cartesian3(radius * cos, radius * sin, 0));}const matrix = Cesium.Transforms.eastNorthUpToFixedFrame(Cesium.Cartesian3.fromDegrees(lon, lat, height));//创建三个方向的旋转圆环this.createAxisCircles(positions, matrix);}
    

    这里同样以模型坐标为原点建立局部坐标系,然后插值计算了局部坐标系为圆心的圆的各个插值点坐标,接下来就是创建圆环

  • 由于插值点是在XY平面的点,所以它构建的圆环可以刚好垂直Z轴,作为Z轴的旋转圆环,假设我们同时创建三个这样的圆环,只需要让其中一个圆环向Y轴方向旋转90度,即可得出Y轴旋转圆环,向X轴方向旋转90度,即可得出X轴旋转圆环,这里的旋转不能简单的只是旋转90度,要知道这个圆环是在局部坐标系下,要想变换成局部坐标系下的旋转90度,需要变换到世界坐标系下,再乘以旋转矩阵

    /*** 创建三个方向的旋转圆环*/private createAxisCircles(positions: Cesium.Cartesian3[],matrix: Cesium.Matrix4): void {//Z轴圆环const zCircle = this.createAxisSphere("model_edit_zCircle",positions,matrix,Cesium.Color.BLUE);this._viewer.scene.primitives.add(zCircle);//X轴圆环const yCircle = this.createAxisSphere("model_edit_yCircle",positions,matrix,Cesium.Color.RED);this._viewer.scene.primitives.add(yCircle);const yRotation = Cesium.Matrix4.fromRotationTranslation(Cesium.Matrix3.fromRotationY(Cesium.Math.toRadians(90)));Cesium.Matrix4.multiply((yCircle.geometryInstances as Cesium.GeometryInstance).modelMatrix,yRotation,(yCircle.geometryInstances as Cesium.GeometryInstance).modelMatrix);//Y轴圆环const xCircle = this.createAxisSphere("model_edit_xCircle",positions,matrix,Cesium.Color.GREEN);this._viewer.scene.primitives.add(xCircle);const xRotation = Cesium.Matrix4.fromRotationTranslation(Cesium.Matrix3.fromRotationX(Cesium.Math.toRadians(90)));Cesium.Matrix4.multiply((xCircle.geometryInstances as Cesium.GeometryInstance).modelMatrix,xRotation,(xCircle.geometryInstances as Cesium.GeometryInstance).modelMatrix);}
    
  • 而创建球体也很简单,当有了插值点后,直接借助cesium中的api进行画线即可,插值点越密,圆形越光滑

     /*** 创建坐标轴球体*/private createAxisSphere(name: string,positions: Cesium.Cartesian3[],matrix: Cesium.Matrix4,color: Cesium.Color): Cesium.Primitive {const primitive = new Cesium.Primitive({geometryInstances: new Cesium.GeometryInstance({id: name,geometry: new Cesium.PolylineGeometry({positions,width: 5,}),attributes: {color: Cesium.ColorGeometryInstanceAttribute.fromColor(color),},}),releaseGeometryInstances: false,appearance: new Cesium.PolylineColorAppearance({translucent: false,}),modelMatrix: matrix,}) as EditablePrimitive;primitive._name = name;this._coordCircle.push(primitive);return primitive;}
    

    在这里插入图片描述

控件的销毁

我们实现了控件的创建,很显然让切换控件时,需要销毁当前控件,我们之前已经记录了每个控件,因此当销毁时,只需要拿到该控件,在primitives集合中去除即可

 /*** 移除所有工具*/private removeAllTools(): void {this.removeCoordArrows();this.removeCoordCircle();}/*** 移除坐标箭头*/private removeCoordArrows(): void {if (this._coordArrows) {this._viewer.scene.primitives.remove(this._coordArrows);this._coordArrows = undefined;}}/*** 移除坐标圆环*/private removeCoordCircle(): void {this._coordCircle.forEach((element) => {this._viewer.scene.primitives.remove(element);});this._coordCircle = [];}
}

实现控件拖动

不管是移动模型还是旋转模型,我们操作的逻辑都是需要鼠标左键按下,然后检测鼠标是否在控件上,如果是,则检测是否是否滑动,计算滑动的距离应用到模型变换上,然后检测鼠标左键抬起,整个过程结束。我们逐步分解一下

  • 首先需要监听鼠标按下命令,如果按下后,再检测是否悬停在控件上,如果是,那就要锁定相机,避免鼠标移动地图拖动,然后让悬停的控件变粗,检测悬停位置的鼠标经纬度,如果在地球范围内,在继续进行后续操作

    /*** 初始化鼠标事件(移动,按下,抬起)*/private initEvent(): void {const viewer = this._viewer;this._handler = new Cesium.ScreenSpaceEventHandler(viewer.scene.canvas);this._handler.setInputAction((event:Cesium.ScreenSpaceEventHandler.PositionedEvent) => {const pick = viewer.scene.pick(event.position);if (pick?.primitive?._name &&pick.primitive._name.indexOf("model_edit") !== -1) {//锁定相机viewer.scene.screenSpaceCameraController.enableRotate = false;this._currentPick = pick.primitive as EditablePrimitive;this._currentPick.width = 25;const downPos = viewer.scene.camera.pickEllipsoid(event.position,viewer.scene.globe.ellipsoid);let _tx = 0,_ty = 0,_tz = 0;let _rx = 0,_ry = 0,_rz = 0;//防止点击到地球之外报错if (downPos && Cesium.defined(downPos)) {const downDegree = CoordTransform.transformCartesianToWGS84(viewer,downPos);//鼠标移动事件//  剩余操作}, Cesium.ScreenSpaceEventType.LEFT_DOWN);}
    

    然后在剩余操作里完善鼠标移动事件,其实不管是平移操作还是旋转操作,本质上都是鼠标移动,当是移动操作的时候,我们可以轻易的计算X轴移动距离和Y轴移动距离,只需要鼠标移动的末尾的经纬度减去最初鼠标点击的经纬度,即可算出差来,作为X轴移动的距离和Y轴移动的距离,但是移动Z轴的时候,我们发现鼠标依然是向上拖动的,也就是鼠标在竖向操作,类似于Y轴的移动逻辑,此时只需要求出鼠标滑动的距离。乘以一个系数,就可以得出Z轴移动的距离了。

      //鼠标移动事件this._handler.setInputAction((movement: Cesium.ScreenSpaceEventHandler.MotionEvent) => {if (!this._currentPick) return;  // 增加空值检查const endPos = viewer.scene.camera.pickEllipsoid(movement.endPosition,viewer.scene.globe.ellipsoid);if (endPos && Cesium.defined(endPos)) {const endDegree = CoordTransform.transformCartesianToWGS84(viewer,endPos);const _yPix = movement.endPosition.y - event.position.y;const _xPix = movement.endPosition.x - event.position.x;//根据当前选中控制器更新相应变量switch (this._currentPick._name) {case "model_edit_xArrow":_tx = endDegree.lng - downDegree.lng;break;case "model_edit_yArrow":_ty = endDegree.lat - downDegree.lat;break;case "model_edit_zArrow":_tz = -this._dStep * _yPix;break;case "model_edit_xCircle":_rx = this._rStep * _yPix;break;case "model_edit_yCircle":_ry = this._rStep * _xPix;break;case "model_edit_zCircle":_rz = this._rStep * _xPix;break;}this.updateModel(this._params, _tx, _ty, _tz, _rx, _ry, _rz);}}, Cesium.ScreenSpaceEventType.MOUSE_MOVE);
    

    这里的movement.endPosition是屏幕坐标系,他是个二维坐标,用来判断鼠标横向滑动的距离和竖向滑动的距离,然后我们再来分析旋转的逻辑,也就是 _rx,_ry,_rz,让旋转X轴时,其实是鼠标竖向滑动,Y和Z轴则是横向滑动,我们分别计算一个滑动系数,最后更新模型

    /*** 更新模型位置*/private updateModel(params: EditParams,tx: number,ty: number,tz: number,rx: number,ry: number,rz: number): void {//创建旋转矩阵const rotationX = Cesium.Matrix4.fromRotationTranslation(Cesium.Matrix3.fromRotationX(Cesium.Math.toRadians(params.rx + rx)));const rotationY = Cesium.Matrix4.fromRotationTranslation(Cesium.Matrix3.fromRotationY(Cesium.Math.toRadians(params.ry + ry)));const rotationZ = Cesium.Matrix4.fromRotationTranslation(Cesium.Matrix3.fromRotationZ(Cesium.Math.toRadians(params.rz + rz)));let position = Cesium.Cartesian3.fromDegrees(params.tx + tx,params.ty + ty,params.tz + tz )let matrix = Cesium.Transforms.eastNorthUpToFixedFrame(position)//旋转、平移矩阵相乘Cesium.Matrix4.multiply(matrix,rotationX,matrix)Cesium.Matrix4.multiply(matrix,rotationY,matrix)Cesium.Matrix4.multiply(matrix,rotationZ,matrix)//更新模型变换this._b3dm.root.transform = matrix;// //更新平移指示器if (this._coordArrows) {this.updateLineArrow(this._b3dm);}}
    

更新模型的操作很简单,分别求出旋转矩阵和平移矩阵,最后替换掉原有的矩阵,模型就发生变换了,这里需要注意的是,当模型发生平移后,可视化控件也需要跟着发生变换,此时需要更新可视化控件的位置,原理也很简单,重新求一下模型的位置,然后创建可视化控件即可

/*** 更新箭头指示器* @param b3dm - 3DTiles模型*/private updateLineArrow(b3dm: Cesium.Cesium3DTileset): void {//移除当前的箭头指示器this.removeCoordArrows();const viewer = this._viewer;//计算长度(使用包围球半径的1/3作为指示器长度)const length = b3dm.boundingSphere.radius / 3;const originalMatrix = this._b3dm.root.transform.clone()const ps = new Cesium.Cartesian3();Cesium.Matrix4.getTranslation(originalMatrix,ps)let originDegree = CoordTransform.transformCartesianToWGS84(viewer,ps)//创建平移向量(三个方向等长)const translateCartesian = new Cesium.Cartesian3(length, length, length);//深拷贝中心点位置const originPos = JSON.parse(JSON.stringify(ps));//计算目标点位置const targetDegree = this.getTransPosition(originPos, translateCartesian);//重新初始化箭头指示器this.initLineArrow(originDegree, targetDegree, length);}
  • 最后是当鼠标滑动结束后,伴随而来的是鼠标抬起事件,在这个事件中需要记录模型变换后的姿态,重新解锁相机,移除掉鼠标移动的监听
//鼠标抬起事件this._handler.setInputAction(() => {if (!this._currentPick) return;  // 增加空值检查viewer.scene.screenSpaceCameraController.enableRotate = true;this._currentPick.width = this._defaultWidth;this._currentPick = undefined;//更新最新参数this._params.tx += _tx;this._params.ty += _ty;this._params.tz += _tz;this._params.rx += _rx;this._params.ry += _ry;this._params.rz += _rz;//移除事件监听this._handler.removeInputAction(Cesium.ScreenSpaceEventType.MOUSE_MOVE);this._handler.removeInputAction(Cesium.ScreenSpaceEventType.LEFT_UP);}, Cesium.ScreenSpaceEventType.LEFT_UP);}

以上就是所有流程的原理,主要是需要弄明白cesium世界坐标系和局部坐标系之间的变换以及鼠标移动对应的是哪个轴的操作。

最新参数
this._params.tx += _tx;
this._params.ty += _ty;
this._params.tz += _tz;
this._params.rx += _rx;
this._params.ry += _ry;
this._params.rz += _rz;

        //移除事件监听this._handler.removeInputAction(Cesium.ScreenSpaceEventType.MOUSE_MOVE);this._handler.removeInputAction(Cesium.ScreenSpaceEventType.LEFT_UP);}, Cesium.ScreenSpaceEventType.LEFT_UP);}

> 以上就是所有流程的原理,主要是需要弄明白cesium世界坐标系和局部坐标系之间的变换以及鼠标移动对应的是哪个轴的操作。

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

我是如何从 0 到 1 找到 Web3 工作的?

作者&#xff1a;Lotus的人生实验 关于我花了一个月的时间&#xff0c;从 0 到 1 学习 Web3 相关的知识和编程知识。然后找到了一个 Web3 创业公司实习的远程工作。 &#x1f447;&#x1f447;&#x1f447; 我的背景: 计算机科班&#xff0c;学历还可以(大厂门槛水平) 毕业工…

进程状态(R|S|D|t|T|X|Z)、僵尸进程及孤儿进程

文章目录 一.进程状态进程排队状态&#xff1a;运行、阻塞、挂起 二.Linux下的进程状态R 运行状态&#xff08;running&#xff09;S 睡眠状态&#xff08;sleeping)D 磁盘休眠状态&#xff08;Disk sleep&#xff09;t 停止、暂停状态(tracing stopped)T 停止、暂停状态(stopp…

为什么要将PDF转换为CSV?CSV是Excel吗?

在企业和数据管理的日常工作中&#xff0c;PDF文件和CSV文件承担着各自的任务。PDF通常用于传输和展示静态的文档&#xff0c;而CSV因其简洁、易操作的特性&#xff0c;广泛应用于数据存储和交换。如果需要从PDF中提取、分析或处理数据&#xff0c;转换为CSV格式可能是一个高效…

Starlink卫星动力学系统仿真建模第十讲-基于SMC和四元数的卫星姿态控制示例及Python实现

基于四元数与滑模控制的卫星姿态控制 一、基本原理 1. 四元数姿态表示 四元数运动学方程&#xff1a; 3. 滑模控制设计 二、代码实现&#xff08;Python&#xff09; 1. 四元数运算工具 import numpy as npdef quat_mult(q1, q2):"""四元数乘法""…

CSS—引入方式、选择器、复合选择器、文字控制属性、CSS特性

目录 CSS 1.引入方式 2.选择器 3.复合选择器 4.文字控制属性 5.CSS特性 CSS 层叠样式表&#xff0c;是一种样式表语言&#xff0c;用来描述HTML文档的呈现 书写时一般按照顺序&#xff1a;盒子模型属性—>文字样式—>圆角、阴影等修饰属性 1.引入方式 引入方式方…

OpenHarmony-4.基于dayu800 GPIO 实践(2)

基于dayu800 GPIO 进行开发 1.DAYU800开发板硬件接口 LicheePi 4A 板载 2x10pin 插针&#xff0c;其中有 16 个原生 IO&#xff0c;包括 6 个普通 IO&#xff0c;3 对串口&#xff0c;一个 SPI。TH1520 SOC 具有4个GPIO bank&#xff0c;每个bank最大有32个IO&#xff1a;  …

win11 24h2 远程桌面 频繁断开 已失去连接 2025

一、现象 Windows11自升级2025年2月补丁后版本号为系统版本是26100.3194&#xff0c;远程桌面频繁断开连接&#xff0c;尝试连接&#xff0c;尤其在连接旧的server2012 二、临时解决方案 目前经测试&#xff0c;在组策略中&#xff0c;远程桌面连接客户端&#xff0c;关闭客户…

rust学习笔记6-数组练习704. 二分查找

上次说到rust所有权看看它和其他语言比有什么优势&#xff0c;就以python为例 # Python3 def test():a [1, 3, -4, 7, 9]print(a[4])b a # 所有权没有发生转移del b[4]print(a[4]) # 由于b做了删除&#xff0c;导致a再度访问报数组越界if __name__ __main__:test() 运行结…

Windows安装NVIDIA显卡CUDAD调用GPU,适用于部署deepseek r1

显卡、显卡驱动、CUDA之间的关系 显卡&#xff1a;&#xff08;GPU&#xff09;&#xff0c;主流是NVIDIA的GPU&#xff0c;因为深度学习本身需要大量计算。GPU的并行计算能力&#xff0c;在过去几年里恰当地满足了深度学习的需求。AMD的GPU基本没有什么支持&#xff0c;可以不…

基于无人机遥感的烟株提取和计数研究

一.研究的背景、目的和意义 1.研究背景及意义 烟草作为我国重要的经济作物之一&#xff0c;其种植面积和产量的准确统计对于烟草产业的发展和管理至关重要。传统的人工烟株计数方法存在效率低、误差大、难以覆盖大面积烟田等问题&#xff0c;已无法满足现代烟草种植管理的需求…

《深度学习实战》第3集:循环神经网络(RNN)与序列建模

第3集&#xff1a;循环神经网络&#xff08;RNN&#xff09;与序列建模 引言 在深度学习领域&#xff0c;处理序列数据&#xff08;如文本、语音、时间序列等&#xff09;是一个重要的研究方向。传统的全连接网络和卷积神经网络&#xff08;CNN&#xff09;难以直接捕捉序列中…

【前沿探索篇七】【DeepSeek自动驾驶:端到端决策网络】

第一章 自动驾驶的"感官革命":多模态神经交响乐团 1.1 传感器矩阵的量子纠缠 我们把8路摄像头+4D毫米波雷达+128线激光雷达的融合称为"传感器交响乐",其数据融合公式可以简化为: def sensor_fusion(cam, radar, lidar):# 像素级特征提取 (ResNet-152…

可狱可囚的爬虫系列课程 13:Requests使用代理IP

一、什么是代理 IP 代理 IP&#xff08;Proxy IP&#xff09;是一个充当“中间人”的服务器IP地址&#xff0c;用于代替用户设备&#xff08;如电脑、手机等&#xff09;直接与目标网站或服务通信。用户通过代理IP访问互联网时&#xff0c;目标网站看到的是代理服务器的IP地址&…

https:原理

目录 1.数据的加密 1.1对称加密 1.2非对称加密 2.数据指纹 2.1数据指纹实际的应用 3.数据加密的方式 3.1只使用对称加密 3.2只使用非对称加密 3.3双方都使用对称加密 3.4非对称加密和对称加密一起使用 4.中间人攻击 5.CA证书 5.1什么是CA证书 CA证书的验证 6.https的原理 1.数据…

Github项目管理之 其余分支同步main分支

文章目录 方法&#xff1a;通过 Pull Request 同步分支1. **创建一个从 main 到目标分支的 Pull Request**2. **合并 Pull Request** 注意事项总结 在 GitHub 网页上&#xff0c;你可以通过 Pull Request 的方式将一个分支&#xff08;例如 main 分支&#xff09;的修改同步到…

Aseprite绘画流程案例(5)——花盆

1.最终图片效果 参考素材来源于&#xff1a;手绘像素画第三课&#xff1a;像素画盆花示范&#xff08;无参考图&#xff09;_哔哩哔哩_bilibili 2.流程 1.新建画布40X27的画布&#xff0c;打开显示网格&#xff0c;背景色为白色 2.画出梯形的盆 3.给盆进行亮暗对比上色 4.添…

【模板】csdn markdown语法演示

这里写自定义目录标题 欢迎使用Markdown编辑器新的改变功能快捷键合理的创建标题&#xff0c;有助于目录的生成如何改变文本的样式插入链接与图片如何插入一段漂亮的代码片生成一个适合你的列表创建一个表格设定内容居中、居左、居右SmartyPants 创建一个自定义列表如何创建一个…

【Python系列】PYTHONUNBUFFERED=1的作用

&#x1f49d;&#x1f49d;&#x1f49d;欢迎来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐:kwan 的首页,持续学…

Adobe After Effects的动画制作

作者&#xff1a;余佳琪 目录 一、 前言 二、 可动骨骼的选择 三、 运动曲线的设置 四、 图层的选定与应用 五、 插件的应用&#xff08;阴影&#xff0c;高光&#xff0c;特效&#xff09; 六、 导出 一、 前言 在当今世界&#x…

可狱可囚的爬虫系列课程 14:10 秒钟编写一个 requests 爬虫

一、前言 当重复性的工作频繁发生时&#xff0c;各种奇奇怪怪提高效率的想法就开始萌芽了。当重复代码的模块化封装已经不能满足要求的时候&#xff0c;更高效的方式就被揭开了神秘的面纱。本文基于这样的想法&#xff0c;来和大家探讨如何 10 秒钟编写一个 requests 爬虫程序。…