ArcGIS JSAPI 高级教程 - ArcGIS Maps SDK for JavaScript - 自定义范围后处理效果(优化版)

news2025/1/12 9:00:24

ArcGIS JSAPI 高级教程 - ArcGIS Maps SDK for JavaScript - 自定义范围后处理效果(优化版)

    • 核心代码
    • 完整代码
    • 在线示例

在这里插入图片描述

ArcGIS Maps SDK for JavaScript 从 4.29 开始增加 RenderNode 类,可以添加数据以及操作 FBO(ManagedFBO)

通过操作 FBO,可以通过后处理实现很多效果,官方提供了几个示例,感兴趣可以看看。

本文介绍一下通过 FBO,实现自定义范围后处理效果(自定义三角形范围)。

本文(优化版)与之前文章的区别在于:优化版可以支持贴地的效果,实用性更强,详情见示例图和在线示例。

另外,本文以三角形范围为例,可以在此基础上扩展为多边形范围

本文包括核心代码、完整代码以及在线示例


核心代码

首先介绍一下原理:通过地图构建三角形数据,转为 WebGL 内部坐标,即世界坐标;

片元着色器中,通过深度纹理重建世界坐标。

根据当前 uv 的世界坐标以及三角形顶点数据,判断是否在三角形内,三角形内外显示不同颜色。


#version 300 es

precision mediump float;
out mediump vec4 fragColor;

in vec2 uv;

// 颜色纹理
uniform sampler2D colorTex;

// 渲染深度纹理
uniform sampler2D depthTex;

// 相机矩阵
uniform mat4 u_viewMatrix;
uniform mat4 u_projectionMatrix;
uniform mat4 u_inverseProjectionMatrix;
// 相机远近点
uniform vec2 nearFar;

// 三角形顶点
uniform vec3[3] u_triangle_out;

// 线性化深度
float linearizeDepth(float depth) {
    float depthNdc = depth * 2.0 - 1.0;
    return (2.0 * nearFar[0] * nearFar[1]) / (depthNdc * (nearFar[1] - nearFar[0]) - (nearFar[1] + nearFar[0]));
}

// 获取深度值
float linearDepth(vec2 uv) {
    ivec2 iuv = ivec2(uv * vec2(textureSize(depthTex, 0)));
    return texelFetch(depthTex, iuv, 0).r;
}

// 深度值获取坐标
vec4 getPositionByDepth(vec2 uv) {

    // 获取深度值
    float depth = linearDepth(uv);

    // 将深度值转换为视图空间中的Z值
    // 这通常涉及到将非线性深度值转换为线性深度值
    float viewZ = linearizeDepth(depth);

    // 计算裁剪空间中的W值
    // 这通常用于从NDC(标准化设备坐标)转换为裁剪坐标
    float clipW = u_projectionMatrix[2][3] * viewZ + u_projectionMatrix[3][3];

    // 将纹理坐标和深度值转换为NDC坐标
    // NDC坐标范围是[-1, 1]
    vec3 ndcPosition = vec3(uv, depth) * 2.0 - 1.0;

    // 将NDC坐标转换为裁剪坐标
    // 通过乘以裁剪空间中的W值来实现
    vec4 clipPosition = vec4(ndcPosition, 1.0) * clipW;

    // 将裁剪坐标变换回视图坐标
    // 通过乘以投影矩阵的逆矩阵来实现
    vec4 viewPos = u_inverseProjectionMatrix * clipPosition;

    // 进行透视除法,将视图坐标转换为齐次坐标
    viewPos /= viewPos.w;

    // 返回视图空间中的位置
    return viewPos;
}

// 判断两个向量是否指向同一方向
bool SameSide(vec3 A, vec3 B, vec3 C, vec3 P)
{
    vec3 AB = B - A;
    vec3 AC = C - A;
    vec3 AP = P - A;

    vec3 v1 = cross(AB, AC);
    vec3 v2 = cross(AB, AP);

    // Normalize the cross products to ensure consistent direction
    v1 = normalize(v1);
    v2 = normalize(v2);

    // V1和v2应该指向同一个方向
    return dot(v1, v2) >= 0.0;
}

// 判断点在三角形内
bool isPointInTriangle(vec3 A, vec3 B, vec3 C, vec3 P)
{
    return SameSide(A, B, C, P) &&
    SameSide(B, C, A, P) &&
    SameSide(C, A, B, P);
}

void main() {

    vec4 color = texture(colorTex, uv);

    // 重建世界坐标
    vec4 localPosition = getPositionByDepth(uv);

    // 转换三角形顶点数据
    vec4 temp1 = u_viewMatrix * vec4(u_triangle_out[0], 1.0);
    vec4 temp2 = u_viewMatrix * vec4(u_triangle_out[1], 1.0);
    vec4 temp3 = u_viewMatrix * vec4(u_triangle_out[2], 1.0);

    if (gl_FrontFacing == true){
        // 三角形范围
        if (!isPointInTriangle(
            temp1.xyz / temp1.w,
            temp2.xyz / temp2.w,
            temp3.xyz / temp3.w,
            localPosition.xyz
        )) {
            fragColor = color;
        } else {
            fragColor = color * 2.0;
        }
    }
}


完整代码


<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="utf-8"/>
    <meta name="viewport" content="initial-scale=1, maximum-scale=1,user-scalable=no"/>
    <title>Custom RenderNode - 自定义范围后处理(优化版) | Sample | ArcGIS Maps SDK for JavaScript 4.29</title>

    <link rel="stylesheet" href="https://openlayers.vip/arcgis_api/4.30/esri/themes/light/main.css"/>
    <script src="https://openlayers.vip/arcgis_api/4.30/init.js"></script>
    <script src="https://openlayers.vip/examples/resources/renderCommon.js"></script>
    <script type="module" src="https://js.arcgis.com/calcite-components/2.5.1/calcite.esm.js"></script>
    <link rel="stylesheet" type="text/css" href="https://js.arcgis.com/calcite-components/2.5.1/calcite.css"/>

    <script>
        var _hmt = _hmt || [];
        (function () {
            var hm = document.createElement("script");
            hm.src = "https://hm.baidu.com/hm.js?f80a36f14f8a73bb0f82e0fdbcee3058";
            var s = document.getElementsByTagName("script")[0];
            s.parentNode.insertBefore(hm, s);
        })();
    </script>
    <style>
        html,
        body,
        #viewDiv {
            padding: 0;
            margin: 0;
            height: 100%;
            width: 100%;
        }
    </style>
    <script>
        require(["esri/Map", "esri/views/SceneView", "esri/views/3d/webgl/RenderNode",
            "esri/Graphic", "esri/views/3d/webgl",
            "esri/geometry/SpatialReference",
            "esri/widgets/Home",
            "esri/layers/IntegratedMesh3DTilesLayer",

        ], function (
            Map,
            SceneView,
            RenderNode,
            Graphic,
            webgl,
            SpatialReference,
            Home,
            IntegratedMesh3DTilesLayer,
        ) {

            const {map, view} = initMap({Map, SceneView, Home});

            // 3dtile 数据
            const layer = new IntegratedMesh3DTilesLayer({
                url: "http://openlayers.vip/cesium/3dtile/xianggang_1.1/tileset.json",
                title: "Utrecht Integrated Mesh 3D Tiles"
            });

            view.map.add(layer);

            // 创建多边形
            const polygon = {
                type: "polygon", // autocasts as new Polygon()
                hasZ: false,
                rings: [
                    [114.17572049692221, 22.29277650792162, 0],
                    [114.1723157633122, 22.299970323140712, 0],
                    [114.1658997800559, 22.29321842654308, 0],
                    [114.17572049692221, 22.29277650792162, 0]
                ]
            };

            const fillSymbol = {
                type: "simple-fill", // autocasts as new SimpleFillSymbol()
                color: [227, 139, 79, 0.8],
                outline: {
                    // autocasts as new SimpleLineSymbol()
                    color: [255, 255, 255],
                    width: 1
                }
            };

            // Add the geometry and symbol to a new graphic
            const polygonGraphic = new Graphic({
                geometry: polygon,
                visible: false,
                symbol: fillSymbol
            });

            // Add the graphics to the view's graphics layer
            view.graphics.addMany([polygonGraphic]);

            // 获取多边形坐标
            const points = polygonGraphic.geometry.rings[0].flat();

            // 世界坐标
            let localOriginRender;

            view.when(() => {

                // 定位
                layer.when(function () {
                    view.extent = layer.fullExtent;
                });

                // 经纬度坐标转为世界坐标
                localOriginRender = webgl.toRenderCoordinates(
                    view,
                    points,
                    0,
                    SpatialReference.WGS84,
                    new Float32Array(points.length - 3),
                    0,
                    (points.length - 3) / 3,
                );

                // Derive a new subclass from RenderNode called LuminanceRenderNode
                const LuminanceRenderNode = RenderNode.createSubclass({
                    constructor: function () {
                        // consumes and produces define the location of the the render node in the render pipeline
                        this.consumes = {required: ["composite-color"]};
                        this.produces = "composite-color";
                    },
                    // Ensure resources are cleaned up when render node is removed
                    destroy() {
                        this.shaderProgram && this.gl?.deleteProgram(this.shaderProgram);
                        this.positionBuffer && this.gl?.deleteBuffer(this.positionBuffer);
                        this.vao && this.gl?.deleteVertexArray(this.vao);
                    },
                    properties: {
                        // Define getter and setter for class member enabled
                        enabled: {
                            get: function () {
                                return this.produces != null;
                            },
                            set: function (value) {
                                // Setting produces to null disables the render node
                                this.produces = value ? "composite-color" : null;
                                this.requestRender();
                            }
                        }
                    },

                    render(inputs) {
                        // The field input contains all available framebuffer objects
                        // We need color texture from the composite render target
                        const input = inputs.find(({name}) => name === "composite-color");
                        const color = input.getTexture();
                        // Acquire the composite framebuffer object, and bind framebuffer as current target
                        const output = this.acquireOutputFramebuffer();

                        const gl = this.gl;
                        const depth = input.getTexture(gl.DEPTH_STENCIL_ATTACHMENT);

                        // Clear newly acquired framebuffer
                        gl.clearColor(0, 0, 0, 1);
                        gl.colorMask(true, true, true, true);
                        gl.clear(gl.COLOR_BUFFER_BIT);

                        // 激活透明
                        activeOpacity(gl);

                        // 初始化着色器
                        this.ensureShader(gl);
                        // 初始化屏幕数据
                        this.ensureScreenSpacePass(gl);

                        // 绑定着色器参数
                        gl.useProgram(this.shaderProgram);
                        gl.uniform2fv(this.nearFarUniformLocation, [this.camera.near, this.camera.far]);

                        // 激活一号纹理
                        gl.activeTexture(gl.TEXTURE0);
                        // 绑定一号纹理
                        gl.bindTexture(gl.TEXTURE_2D, color.glName);
                        // 传入着色器
                        gl.uniform1i(this.textureUniformLocation, 0);

                        // 激活三号纹理
                        // 绑定深度纹理
                        gl.activeTexture(gl.TEXTURE2);
                        gl.bindTexture(gl.TEXTURE_2D, depth.glName);
                        gl.uniform1i(this.depthTexUniformLocation, 2);

                        // 传入三角形顶点
                        gl.uniform3fv(this.textureUniformTriangleExtent,
                            new Float32Array(localOriginRender));

                        // 激活相机矩阵
                        activeMatrix(this);

                        // Issue the render call for a screen space render pass
                        gl.bindVertexArray(this.vao);

                        // 绘制
                        gl.drawArrays(gl.TRIANGLES, 0, 3);

                        // use depth from input on output framebuffer
                        output.attachDepth(input.getAttachment(gl.DEPTH_STENCIL_ATTACHMENT));
                        this.requestRender();

                        return output;
                    },

                    // 着色器程序
                    shaderProgram: null,
                    // 纹理
                    textureUniformLocation: null,
                    // 顶点位置
                    positionLocation: null,
                    // 顶点数组
                    vao: null,
                    // 顶点缓冲区
                    positionBuffer: null,
                    // Setup screen space filling triangle
                    ensureScreenSpacePass(gl) {
                        if (this.vao) {
                            return;
                        }

                        this.vao = gl.createVertexArray();
                        gl.bindVertexArray(this.vao);
                        this.positionBuffer = gl.createBuffer();
                        gl.bindBuffer(gl.ARRAY_BUFFER, this.positionBuffer);
                        const vertices = new Float32Array([-1.0, -1.0, 3.0, -1.0, -1.0, 3.0]);
                        gl.bufferData(gl.ARRAY_BUFFER, vertices, gl.STATIC_DRAW);

                        gl.vertexAttribPointer(this.positionLocation, 2, gl.FLOAT, false, 0, 0);
                        gl.enableVertexAttribArray(this.positionLocation);

                        gl.bindVertexArray(null);
                    },

                    // Setup custom shader programs
                    ensureShader(gl) {
                        if (this.shaderProgram != null) {
                            return;
                        }
                        // The vertex shader program
                        const vshader = `#version 300 es

                                // 绘制顶点
                                in vec2 position;
                                // uv 取样
                                out vec2 uv;

                                void main() {
                                    // 绘制顶点
                                    gl_Position = vec4(position, 0.0, 1.0);
                                    // uv 调整中心
                                    uv = position * 0.5 + vec2(0.5);
                                }
                    `;

                        // The fragment shader program applying a greyscsale conversion
                        const fshader = `#version 300 es

                                precision mediump float;
                                out mediump vec4 fragColor;

                                in vec2 uv;

                                // 颜色纹理
                                uniform sampler2D colorTex;

                                // 渲染深度纹理
                                uniform sampler2D depthTex;

                                // 相机矩阵
                                uniform mat4 u_viewMatrix;
                                uniform mat4 u_projectionMatrix;
                                uniform mat4 u_inverseProjectionMatrix;
                                // 相机远近点
                                uniform vec2 nearFar;

                                // 三角形顶点
                                uniform vec3[3] u_triangle_out;

                                // 线性化深度
                                float linearizeDepth(float depth) {
                                    float depthNdc = depth * 2.0 - 1.0;
                                    return (2.0 * nearFar[0] * nearFar[1]) / (depthNdc * (nearFar[1] - nearFar[0]) - (nearFar[1] + nearFar[0]));
                                }

                                // 获取深度值
                                float linearDepth(vec2 uv) {
                                    ivec2 iuv = ivec2(uv * vec2(textureSize(depthTex, 0)));
                                    return texelFetch(depthTex, iuv, 0).r;
                                }

                                // 深度值获取坐标
                                vec4 getPositionByDepth(vec2 uv) {

                                    // 获取深度值
                                    float depth = linearDepth(uv);

                                    // 将深度值转换为视图空间中的Z值
                                    // 这通常涉及到将非线性深度值转换为线性深度值
                                    float viewZ = linearizeDepth(depth);

                                    // 计算裁剪空间中的W值
                                    // 这通常用于从NDC(标准化设备坐标)转换为裁剪坐标
                                    float clipW = u_projectionMatrix[2][3] * viewZ + u_projectionMatrix[3][3];

                                    // 将纹理坐标和深度值转换为NDC坐标
                                    // NDC坐标范围是[-1, 1]
                                    vec3 ndcPosition = vec3(uv, depth) * 2.0 - 1.0;

                                    // 将NDC坐标转换为裁剪坐标
                                    // 通过乘以裁剪空间中的W值来实现
                                    vec4 clipPosition = vec4(ndcPosition, 1.0) * clipW;

                                    // 将裁剪坐标变换回视图坐标
                                    // 通过乘以投影矩阵的逆矩阵来实现
                                    vec4 viewPos = u_inverseProjectionMatrix * clipPosition;

                                    // 进行透视除法,将视图坐标转换为齐次坐标
                                    viewPos /= viewPos.w;

                                    // 返回视图空间中的位置
                                    return viewPos;
                                }

                                // 判断两个向量是否指向同一方向
                                bool SameSide(vec3 A, vec3 B, vec3 C, vec3 P)
                                {
                                    vec3 AB = B - A;
                                    vec3 AC = C - A;
                                    vec3 AP = P - A;

                                    vec3 v1 = cross(AB, AC);
                                    vec3 v2 = cross(AB, AP);

                                    // Normalize the cross products to ensure consistent direction
                                    v1 = normalize(v1);
                                    v2 = normalize(v2);

                                    // V1和v2应该指向同一个方向
                                    return dot(v1, v2) >= 0.0;
                                }

                                // 判断点在三角形内
                                bool isPointInTriangle(vec3 A, vec3 B, vec3 C, vec3 P)
                                {
                                    return SameSide(A, B, C, P) &&
                                    SameSide(B, C, A, P) &&
                                    SameSide(C, A, B, P);
                                }

                                void main() {

                                    vec4 color = texture(colorTex, uv);

                                    // 重建世界坐标
                                    vec4 localPosition = getPositionByDepth(uv);

                                    // 转换三角形顶点数据
                                    vec4 temp1 = u_viewMatrix * vec4(u_triangle_out[0], 1.0);
                                    vec4 temp2 = u_viewMatrix * vec4(u_triangle_out[1], 1.0);
                                    vec4 temp3 = u_viewMatrix * vec4(u_triangle_out[2], 1.0);

                                    if (gl_FrontFacing == true){
                                        // 三角形范围
                                        if (!isPointInTriangle(
                                            temp1.xyz / temp1.w,
                                            temp2.xyz / temp2.w,
                                            temp3.xyz / temp3.w,
                                            localPosition.xyz
                                        )) {
                                            fragColor = color;
                                        } else {
                                            fragColor = color * 2.0;
                                        }
                                    }
                                }

                   `;

                        this.shaderProgram = initWebgl2Shaders(gl, vshader, fshader);

                        this.textureUniformLocation = gl.getUniformLocation(this.shaderProgram, "colorTex");
                        this.depthTexUniformLocation = gl.getUniformLocation(this.shaderProgram, "depthTex");
                        this.nearFarUniformLocation = gl.getUniformLocation(this.shaderProgram, "nearFar");

                        // 三角形顶点位置
                        this.textureUniformTriangleExtent = gl.getUniformLocation(this.shaderProgram, "u_triangle_out");

                        this.positionLocation = gl.getAttribLocation(this.shaderProgram, "position");
                    }
                });

                // Initializes the new custom render node and connects to SceneView
                const luminanceRenderNode = new LuminanceRenderNode({view});

                // Toggle button to enable/disable the custom render node
                const renderNodeToggle = document.getElementById("renderNodeToggle");
                renderNodeToggle.addEventListener("calciteSwitchChange", () => {
                    luminanceRenderNode.enabled = !luminanceRenderNode.enabled;
                });
            });
        });
    </script>
</head>
<body>
<calcite-block open heading="Toggle Render Node" id="renderNodeUI">
    <calcite-label layout="inline">
        Color
        <calcite-switch id="renderNodeToggle" checked></calcite-switch>
        Grayscale
    </calcite-label>
</calcite-block>
<div id="viewDiv"></div>
</body>
</html>



在这里插入图片描述


在线示例

ArcGIS Maps SDK for JavaScript 在线示例:自定义范围后处理效果(优化版)

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

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

相关文章

微信公众号扫码登录

开发前准备 微信公众平台微信公众平台&#xff0c;给个人、企业和组织提供业务服务与用户管理能力的全新服务平台。https://mp.weixin.qq.com/申请注册个服务号&#xff0c;并且需要微信认证&#xff0c;注意个人公众号无法开通这个功能&#xff0c;因为个人就不给做微信认证。…

Python | Leetcode Python题解之第350题两个数组的交集II

题目&#xff1a; 题解&#xff1a; class Solution:def intersect(self, nums1: List[int], nums2: List[int]) -> List[int]:nums1.sort()nums2.sort()length1, length2 len(nums1), len(nums2)intersection list()index1 index2 0while index1 < length1 and ind…

LLM 中 100K 上下文窗口背后的秘密

最近有几则关于新型大型语言模型 (LLM) 的公告&#xff0c;这些模型可以使用极大的上下文窗口&#xff0c;例如65K 个标记&#xff08;MosaicML 的MPT-7B-StoryWriter-65k &#xff09;甚至100K 个标记&#xff08; Antropic 的引入 100K 上下文窗口&#xff09;。在 Palm-2技术…

C++ | Leetcode C++题解之第354题俄罗斯套娃信封问题

题目&#xff1a; 题解&#xff1a; class Solution { public:int maxEnvelopes(vector<vector<int>>& envelopes) {if (envelopes.empty()) {return 0;}int n envelopes.size();sort(envelopes.begin(), envelopes.end(), [](const auto& e1, const aut…

一招制胜!Spring Boot、Prometheus和Grafana三剑合璧,称霸监控领域!

1. 添加Prometheus和Actuator依赖 在pom.xml中添加Spring Boot Actuator和Micrometer Prometheus依赖&#xff1a; <dependencies> <!--监控功能Actuator--><dependency><groupId>org.springframework.boot</groupId><artifactId>spring…

光伏电站气象站:提升电站效益,智能监控

随着全球对可持续发展和清洁能源需求的日益增长&#xff0c;光伏发电作为一种清洁、可再生的能源形式&#xff0c;正逐步成为能源结构转型的重要支柱。然而&#xff0c;光伏系统的发电效率直接受到气候条件的影响&#xff0c;如光照强度、温度、湿度、风速及风向等。因此&#…

网上商城购物系统

TOC springboot0752网上商城购物系统 第1章 绪论 1.1背景及意义 随着社会的快速发展&#xff0c;计算机的影响是全面且深入的。人们生活水平的不断提高&#xff0c;日常生活中人们对网上商城购物系统方面的要求也在不断提高&#xff0c;购物的人数更是不断增加&#xff0c;…

Kubernetes--深入Pod

前言&#xff1a;本博客仅作记录学习使用&#xff0c;部分图片出自网络&#xff0c;如有侵犯您的权益&#xff0c;请联系删除 出自B站博主教程笔记&#xff1a; ​​​​​​​完整版Kubernetes&#xff08;K8S&#xff09;全套入门微服务实战项目&#xff0c;带你一站式深入…

什么是BOM,有哪些分类?

一、什么是BOM&#xff1f; BOM是物料清单的缩写&#xff0c;也称为产品结构表或产品结构树。 BOM的作用主要是通过计算机辅助企业生产管理&#xff0c;使计算机能够识别企业所制造的产品构成和所有要涉及的物料。 在制造业中&#xff0c;BOM是一份详细记录制造某个产品时所…

“AI+Security”系列第2期(五):大模型自身安全

近日&#xff0c;由安全极客、Wisemodel 社区与 InForSec 网络安全研究国际学术论坛联合主办的 “AISecurity” 系列第二期 —— 对抗&#xff01;大模型自身安全的攻防博弈线上活动如期举行。 在圆桌环节&#xff0c;君同未来创始人兼 CEO 韩蒙、前阿里云高级安全专家郑瀚、C…

聚星文社——绘唐科技Ai推文软件

聚星文社——绘唐科技Ai推文软件 聚星文社--绘唐科技Ai推文软件https://iimenvrieak.feishu.cn/docx/ZhRNdEWT6oGdCwxdhOPcdds7nof AI推文软件是一种利用人工智能技术帮助用户自动生成推文内容的工具。 该软件会分析用户提供的相关信息和目标群体&#xff0c; 然后使用机器学习…

机器学习:决策树之回归树的原理

目录 前言 一、回归树样例 二、回归树原理 三、原理实现步骤 1.选择特征和分裂点 2.递归分裂 3.生成叶子节点 四、进行预测 总结 前言 决策树回归&#xff08;Decision Tree Regression&#xff09;是一种用于预测连续目标变量的回归模型。与分类决策树类似&#xff0…

RISC-V中的acquire和release

RISC-V中&#xff0c;有两类原子操作指令&#xff1a; ⚫ 内存原子操作&#xff08;AMO&#xff09; ⚫ 加载保留/条件存储&#xff08;load reserved / store conditional&#xff09; 此处我们先看下LR和SC指令&#xff1b; 引入的原因 编程语言的开发者会假定体系结构提…

docker-compose示例:nacos单机部署

前面咱们完成了docker基本环境搭建&#xff0c;下面就趁热打铁来练习下nacos的单机部署。 参考官方文档&#xff1a;Nacos Docker 快速开始。考虑到官方搭建教程过于精炼&#xff0c;笔者把搭建过程分享给大家。 文章目录 下载最新部署源码解决网络导致的sql文件下不下来docke…

保姆级-C#与Halcon的窗体界面展示阈值分割图像教程(机器视觉保姆级教程)

经历上一篇《零基础小白实现C#调用halcon dll的过程&#xff0c;并测试程序证明C#halcon联合开发成功》的发布已经过去三天啦&#xff0c; 零基础小白实现C#调用halcon dll的过程&#xff0c;并测试程序证明C#halcon联合开发成功_添加halcondotnet.dll-CSDN博客 在友友的催更下…

树链剖分学习笔记

前言 树链剖分这个东西呢&#xff0c;简而言之就是把一些树上的操作转换成线性的问题。那看起来平平无奇的树链剖分&#xff0c;为什么很多人就是喜欢使用他呢&#xff0c;那想必肯定是有原因的&#xff0c;我们先卖个关子&#xff0c;先看一下树链剖分怎么写&#xff0c;原理…

DMA(hal库)

DMA&#xff08;直接存储器存取&#xff09; DMA&#xff08;Direct Memory Access&#xff0c;直接内存访问&#xff09;是一种允许外设或内存之间直接传输数据的技术&#xff0c;而无需 CPU 参与。这能显著提高数据传输效率并减少 CPU 的负担。 使用 DMA 通常涉及以下步骤&…

图片转PDF怎么转?教你3种快捷方便的jpg转pdf方法

图片文件以及PDF文档已经是我们工作当中不可或缺的一部分&#xff0c;我们在一些商务合作的场景下经常需要把拍摄下来的合同、企划书、画册等图片内容转换为PDF格式后再发送&#xff0c;这样能够极大程度的保证文件的安全性&#xff0c;那么图片应该如何转换成PDF文件呢?今天来…

如何制作统信UOS启动盘?

如何制作统信UOS启动盘&#xff1f; 一、下载UOS系统安装镜像二、在UOS系统环境下制作启动盘步骤一&#xff1a;准备U盘步骤二&#xff1a;打开启动盘制作工具步骤三&#xff1a;选择ISO镜像文件步骤四&#xff1a;选择安装介质并格式化步骤五&#xff1a;等待制作完成 三、在W…

(LLM) 很笨

大型语言模型 (LLM) 并非你所想的那样。你被骗了。LLM 很笨&#xff0c;非常笨。事实上&#xff0c;它们更接近数据库&#xff0c;而不是人类。 这就是为什么人工智能仍然处于征服智能的第一步……如果有的话。 终极煤气灯效应 建立前沿人工智能的成本很高。前沿人工智能需要大…