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

vue项目中使用antvX6(可拖拽,vue3)

参考 先知demons 这位大佬的这篇文章:https://blog.csdn.net/wzy_PROTEIN/article/details/136305034?utm_medium=distribute.pc_relevant.none-task-blog-2defaultbaidujs_baidulandingword~default-0-136305034-blog-136032209.235v43pc_blog_bottom_relevance_base2&spm=1001.2101.3001.4242.1&utm_relevant_index=3
以下是我的流程+思路+页面展示(页面较粗糙)
在这里插入图片描述

  • 左侧:折叠菜单+选项卡
  • 右侧:画布
<template><div class="dashboard-container"><!-- 顶部操作按钮 --><div class="butClass"><el-button type="primary" @click="save">保存</el-button></div><!-- 主内容区 --><div class="antvBox"><!-- 左侧节点菜单列表 --><el-tabs type="border-card" class="tabsBoxs"><el-tab-pane label="产品"><el-collapse v-model="activeNames" accordion><!-- 产品、场景、菜单分组 --><el-collapse-item title="产品" name="product"><div class="productClass"><!-- 动态渲染节点项,支持拖拽 --><divdraggable="true"@dragend="handleDragEnd($event, item, activeNames)"v-for="item in productData.data":key="item.name":class="getShapeClass(item.type)">{{ item.name }}</div></div></el-collapse-item><!-- 场景、菜单分组结构类似 --></el-collapse></el-tab-pane></el-tabs><!-- 右侧画布区域 --><div class="canvas-card"><div id="container" /> <!-- AntV X6 画布容器 --></div></div></div>
</template>
  • 我写的是css绘制图形拖拽去画布后生成节点(我的节点内容是列表内容渲染的,而且需要多层级)折叠面板只能展开一级,拖拽后自动关闭当前层级,打开下一级,需要修改的可以在handleDragEnd里修改

<script setup lang="ts">import { ref, onMounted } from 'vue';import { Graph } from '@antv/x6';// --------------------------- 状态管理 ---------------------------// 折叠面板当前展开的分组名称(手风琴模式)const activeNames = ref<string>('product');// 当前选中的节点(用于节点选中状态管理)const curSelectNode = ref(null);//  产品节点数据const productData = {data: [{ name: '开始', type: 'start', id: 1 },{ name: '过程', type: 'process', id: 2 },{ name: '可选过程', type: 'optProcess', id: 3 },{ name: '决策', type: 'decisionMaking', id: 4 },{ name: '数据', type: 'nodeData', id: 6 },{ name: '连接', type: 'connect', id: 5 },],};// 场景节点数据const sceneData = {data: [{ name: '场景A', type: 'process', id: 11 },{ name: '场景B', type: 'start', id: 12 },{ name: '场景C', type: 'optProcess', id: 13 },{ name: '场景D', type: 'decisionMaking', id: 14 },{ name: '场景E', type: 'nodeData', id: 15 },{ name: '场景F', type: 'connect', id: 16 },],};// 菜单节点数据const menuData = {data: [{ name: '菜单A', type: 'process', id: 21 },{ name: '菜单B', type: 'start', id: 22 },{ name: '菜单C', type: 'optProcess', id: 23 },{ name: '菜单D', type: 'decisionMaking', id: 24 },{ name: '菜单E', type: 'nodeData', id: 25 },{ name: '菜单F', type: 'connect', id: 26 },],};// X6 图形实例const graph = ref(null);/*** 保存当前画布状态*/const save = () => {// 输出画布完整JSON结构(包含节点、连线、布局等信息)console.log(graph.value.toJSON(), 'graph');// 输出所有节点实例(可用于数据持久化或调试)console.log(graph.value.getNodes(), 'node');};// --------------------------- 节点创建逻辑 ---------------------------/*** 向画布添加节点的核心方法* @param x - 节点在画布中的X坐标(基于容器左上角)* @param y - 节点在画布中的Y坐标(基于容器左上角)* @param item - 节点数据(包含type/name/id等信息)*/const addHandleNode = (x, y, item) => {// 基础样式(所有节点共用的默认配置)const baseStyle = {label: item.name, // 节点文本内容attrs: {// 图形和文本样式body: {// 节点图形样式fill: '#eff4ff', // 填充色stroke: '#6397ff', // 边框色strokeWidth: 1, // 边框宽度},label: {// 节点文本样式text: item.name, // 文本内容(与label重复,AntV X6要求)fill: 'black', // 文本颜色fontSize: 12, // 字体大小},},width: 80, // 节点默认宽度(可被类型配置覆盖)height: 40, // 节点默认高度(可被类型配置覆盖)// 统一添加连接桩配置ports: {groups: {top: {position: 'top' /* 顶部端口 */,attrs: {circle: {r: 4,magnet: true,stroke: '#5F95FF',strokeWidth: 1,fill: '#fff',},},},bottom: {position: 'bottom' /* 底部端口 */,attrs: {circle: {r: 4,magnet: true,stroke: '#5F95FF',strokeWidth: 1,fill: '#fff',},},},left: {position: 'left' /* 左侧端口 */,attrs: {circle: {r: 4,magnet: true,stroke: '#5F95FF',strokeWidth: 1,fill: '#fff',},},},right: {position: 'right' /* 右侧端口 */,attrs: {circle: {r: 4,magnet: true,stroke: '#5F95FF',strokeWidth: 1,fill: '#fff',},},},},items: [// 端口实例(启用四个方向的端口){ group: 'top', id: 'top' },{ group: 'bottom', id: 'bottom' },{ group: 'left', id: 'left' },{ group: 'right', id: 'right' },],},};// 按节点类型配置差异化样式(形状、尺寸、特殊属性)const shapeConfig = {// 开始start: {shape: 'rect', // 形状:矩形(通过圆角模拟椭圆)label: item.name,width: 80, // 宽度height: 40, // 高度attrs: {body: {fill: '#eff4ff', // 填充颜色stroke: '#6397ff', // 边框颜色strokeWidth: 1, // 边框宽度rx: 20, // 圆角半径(实现胶囊形)ry: 26,},label: {textWrap: {// 文本换行设置ellipsis: true, // 允许省略width: -10, // 文本宽度调整},},},},// 过程process: {shape: 'rect', // 普通矩形节点label: item.name,width: 80,height: 40,attrs: {body: {fill: '#eff4ff',stroke: '#6397ff',strokeWidth: 1,},label: {textWrap: {ellipsis: true,width: -10,},},},},// 数据节点(多边形)nodeData: {shape: 'polygon', // 形状:多边形label: item.name,width: 90,height: 40,attrs: {body: {fill: '#eff4ff',stroke: '#6397ff',strokeWidth: 1,refPoints: '10,0 40,0 30,20 0,20', // 多边形顶点坐标},label: {textWrap: {ellipsis: true,width: -10,},},},},// 决策节点(菱形)decisionMaking: {shape: 'polygon', // 形状:多边形label: item.name,width: 90,height: 60,attrs: {body: {fill: '#eff4ff',stroke: '#6397ff',strokeWidth: 1,},},// 菱形顶点坐标(相对于节点中心)points: [[-40, 0],[0, -30],[40, 0],[0, 30],],},// 连接connect: {shape: 'circle',width: 60,height: 60,attrs: {body: {fill: '#eff4ff',stroke: '#6397ff',strokeWidth: 1,},},},// 可选过程optProcess: {shape: 'rect',label: item.name,width: 80,height: 40,attrs: {body: {fill: '#eff4ff',stroke: '#6397ff',strokeWidth: 1,rx: 6,ry: 6,},label: {textWrap: {ellipsis: true,width: -10,},},},ports: {groups: {/* ... */},items: [/* ... */],visible: true, // 默认可见},},};// 坐标校验(防止节点超出画布边界)x = Math.max(0, Math.min(x, graph.value.options.width - 80));y = Math.max(0, Math.min(y, graph.value.options.height - 40));// 创建节点并添加到画布(合并基础样式和类型特定样式)graph.value.addNode({id: item.id, // 节点唯一IDx: x, // 坐标y: y,...baseStyle, // 基础样式...shapeConfig[item.type], // 合并基础样式和类型特定配置});};// --------------------------- 拖拽事件处理 ---------------------------/*** 处理节点拖拽结束事件(计算坐标并创建节点)* @param e - 拖拽事件对象* @param item - 被拖拽的节点数据* @param name - 当前折叠面板分组名称(用于自动切换分组)*/const handleDragEnd = (e, item, name) => {// 获取画布容器及其位置信息const container = document.getElementById('container');const rect = container.getBoundingClientRect();// 计算节点在画布中的坐标(基于容器左上角)const x = e.clientX - rect.left; // X坐标 = 鼠标X坐标 - 容器左边界距离视口左边界的距离const y = e.clientY - rect.top; // Y坐标同理// 校验节点类型是否合法(防止非法类型节点被创建)const validTypes = ['start', 'process', 'optProcess', 'decisionMaking', 'nodeData', 'connect'];if (!validTypes.includes(item.type)) {return;}// 添加节点到画布addHandleNode(x, y, item);if (name == 'product') {activeNames.value = 'scene';} else if (name == 'scene') {activeNames.value = 'menu';}// addHandleNode(e.pageX - 500, e.pageY - 200, item);};// 添加统一显示控制const showPorts = (show: boolean) => {const ports = graph.value?.container?.querySelectorAll('.x6-port-body');ports?.forEach((port) => {port.style.visibility = show ? 'visible' : 'hidden';});};// --------------------------- 节点交互事件 ---------------------------/*** 初始化节点相关交互事件(鼠标进入/离开、点击等)*/const nodeAddEvent = () => {const container = document.getElementById('container');// 鼠标进入节点时显示端口graph.value.on('node:mouseenter', ({ node }) => {showPorts(true); // 显示连接桩// 添加删除按钮工具(位于节点右上角)node.addTools({name: 'button-remove',args: { x: '100%', y: 0 },});});graph.value.on('node:added', ({ node }) => {node.attr('label/text', node.label || node.id);});// 鼠标离开节点时隐藏端口graph.value.on('node:mouseleave', ({ node }) => {showPorts(false);node.removeTools();});// 节点点击事件graph.value.on('node:click', ({ node }) => {if (curSelectNode.value) {// 已有选中节点时curSelectNode.value.removeTools(); // 移除前一个节点的工具if (curSelectNode.value !== node) {// 点击新节点时// 添加选中边框(半透明蓝色背景)和删除按钮node.addTools([{name: 'boundary',args: {attrs: {fill: '#16B8AA',stroke: '#2F80EB',strokeWidth: 1,fillOpacity: 0.1,},},},{name: 'button-remove',args: {x: '100%',y: 0,offset: { x: 0, y: 0 },},},]);curSelectNode.value = node;} else {curSelectNode.value = null;}} else {curSelectNode.value = node;node.addTools([{name: 'boundary',args: {attrs: {fill: '#16B8AA',stroke: '#2F80EB',strokeWidth: 1,fillOpacity: 0.1,},},},{name: 'button-remove',args: {x: '100%',y: 0,offset: { x: 0, y: 0 },},},]);}});// 连线鼠标移入事件graph.value.on('cell:mouseenter', ({ cell }) => {if (cell.shape === 'edge') {// 仅处理连线// 添加删除按钮工具cell.addTools([{name: 'button-remove',args: {x: '100%',y: 0,offset: { x: 0, y: 0 },},},]);cell.setAttrs({line: {stroke: '#409EFF', // 连线颜色变为蓝色},});cell.zIndex = 99; // 提升层级防止被遮挡}});// 连线鼠标移出事件graph.value.on('cell:mouseleave', ({ cell }) => {if (cell.shape === 'edge') {cell.removeTools();cell.setAttrs({line: {stroke: 'black',},});cell.zIndex = 1;}});graph.value.on('edge:added', ({ edge }) => {edge.attr({line: {stroke: '#1890ff',strokeWidth: 1.5,targetMarker: {name: 'block',size: 6,},},});});graph.value.on('edge:added', ({ edge }) => {edge.attr('line/stroke', '#ff4d4f');setTimeout(() => {edge.attr('line/stroke', '#1890ff');}, 1000);});};// --------------------------- 画布初始化 ---------------------------/*** 初始化AntV X6画布实例(核心配置)*/const initGraph = () => {const container = document.getElementById('container');try {// 创建X6图形实例graph.value = new Graph({container, // 绑定画布容器width: container.offsetWidth, // 画布宽度(自适应容器)height: container.offsetHeight, // 画布高度(自适应容器)background: false,snapline: true, // 启用对齐线(节点自动吸附到网格)connecting: {snap: true, // 连线端点自动吸附到端口allowBlank: false,allowMulti: true, // 允许多重连接allowLoop: true, // 允许自环highlight: true, // 高亮显示highlighting: {magnetAdsorbed: {name: 'stroke',args: {attrs: {fill: '#5F95FF',stroke: '#5F95FF',},},},},router: {name: 'orth', // 正交路由},connector: {name: 'rounded', // 圆角连接器args: {radius: 8, // 圆角半径},},validateMagnet({ magnet }) {// 允许从所有连接桩创建连接return true;},validateConnection({ sourceMagnet, targetMagnet }) {// 允许任意两个连接桩之间的连接return true;},},// 平移配置panning: {enabled: false,},// 鼠标滚轮缩放配置mousewheel: {enabled: true,zoomAtMousePosition: true,modifiers: 'ctrl',minScale: 0.5,maxScale: 3,},// 网格配置grid: {type: 'dot', // 点状网格size: 20, // 网格大小visible: true,args: {color: '#a0a0a0',thickness: 2,},},});// 添加节点事件nodeAddEvent();} catch (error) {console.error('Graph initialization failed:', error);}};// 组件挂载时初始化流程图onMounted(() => {initGraph();});const getShapeClass = (type) => {switch (type) {case 'process':return 'shape-rectangle';case 'start':return 'shape-ellipse';case 'optProcess':return 'shape-parallelogram';case 'decisionMaking':return 'shape-decisionMaking';case 'nodeData':return 'shape-nodeData ';case 'connect':return 'shape-circle';default:return 'shape-rectangle';}};
</script>
  • css

<style lang="scss" scoped>.dashboard-container {/* 占满视口高度 */height: 100vh;}.antvBox {display: flex; /* 弹性布局实现左右分栏 */width: 100%;height: calc(100vh - 60px);/* 扣除顶部按钮高度 */}.tabsBoxs {width: 25%; /* 左侧菜单占比25% */padding: 10px; /*内边距 */border-right: 1px solid #eee; /*右侧边框*/}.canvas-card {width: 75%; /*右侧画布占比75% */padding: 10px;}#container {width: 100%;height: 100%;/* 虚线边框标识画布区域 */border: 1px dashed #a6a6a6;/* 浅灰色背景 */background: #f8f9fa;}/* --------------------------- 左侧节点样式 --------------------------- */.shape-rectangle {/* 矩形节点:直角矩形 */width: 80px;height: 40px;border-radius: 0;}.shape-ellipse {/* 椭圆节点:通过左右padding和大圆角模拟 - 增加左右内边距 -大圆角实现椭圆效果*/padding: 0 26px;border-radius: 30px;}.shape-decisionMaking {/* 菱形节点:通过clip-path裁剪 */clip-path: polygon(50% 0%, 100% 50%, 50% 100%, 0% 50%);}.shape-circle {/* 圆形节点:正方形+50%圆角 */width: 40px;height: 40px;border-radius: 50%;}.shape-nodeData {/* 倾斜节点:通过transform skew实现 ---  向左倾斜20度 */transform: skew(-20deg);/* 注意:文本需要反向倾斜以保持正立,此处简化处理 */}/* --------------------------- AntV X6 样式覆盖 --------------------------- */:deep(.x6-port-body) {/*    强制显示连接桩(解决默认隐藏问题) */visibility: visible !important;/* 过渡动画 */transition: all 0.3s;&:hover {/*  悬停时端口边框变亮绿色 */stroke: #31d0c6;/* 边框加粗 */stroke-width: 2px;}}
</style>

结束!!!!!

相关文章:

vue项目中使用antvX6(可拖拽,vue3)

参考 先知demons 这位大佬的这篇文章&#xff1a;https://blog.csdn.net/wzy_PROTEIN/article/details/136305034?utm_mediumdistribute.pc_relevant.none-task-blog-2defaultbaidujs_baidulandingword~default-0-136305034-blog-136032209.235v43pc_blog_bottom_relevance_ba…...

文档处理控件Aspose.Words 教程:在 Word 中删除空白页完整指南

Word 文档中的空白页总是令人烦恼。如果您想从 Word 文档中删除空白页&#xff0c;以获得更清晰的演示文稿或整理文件&#xff0c;那么您来对地方了。本指南涵盖了使用 Aspose.Words编码解决方案和在 Microsoft Word 中手动删除方案&#xff0c;让您可以灵活地选择最适合自己的…...

自动驾驶最新算法进展

自动驾驶技术的算法进展迅速&#xff0c;涵盖感知、预测、规划、端到端学习等多个领域。以下是2023年至2024年的关键进展及实例&#xff1a; 1. ‌感知与融合‌ ‌BEVTransformer的进化‌&#xff1a;特斯拉的Occupancy Networks升级至支持动态场景建模&#xff0c;结合NeRF技术…...

OpenCV训练题

一、创建一个 PyQt 应用程序&#xff0c;该应用程序能够&#xff1a; 使用 OpenCV 加载一张图像。在 PyQt 的窗口中显示这张图像。提供四个按钮&#xff08;QPushButton&#xff09;&#xff1a; 一个用于将图像转换为灰度图一个用于将图像恢复为原始彩色图一个用于将图像进行…...

34、Spark实现读取XLS文件

需求背景&#xff1a; 有一些xls大文件数据。使用spark-excel&#xff08;spark-excel&#xff09;来读取时&#xff0c;文件太大会oom&#xff1b;工具提供的流式读取参数&#xff1a;maxRowsInMemory 也只支持xlsx类型文件。搜索了poi流式读取xls的方案&#xff0c;HSSFEvent…...

为什么RPN生成的候选框,要使用rcnn来进行分类和回归操作?

一句大白话总结&#xff1a;RPN是广撒网捕鱼&#xff0c;RCNN是细化鱼的分类和具体尺寸 在目标检测任务中&#xff0c;RPN&#xff08;区域提议网络&#xff09; 生成的候选框需要经过 RCNN&#xff08;如 Fast R-CNN、Faster R-CNN&#xff09; 进行分类和回归&#xff0c;这…...

固定总价合同工程范围变更,如果规避风险

一、合同签订阶段 工程范围的锁定 清单式列举 兜底条款&#xff1a;在合同附件中以 “正负零以上主体结构”“XX 型号设备采够” 等具体描述界定承包范围&#xff0c;同时约定 “超出本附件描述的工程内容视为变更”111。 设计深度要求&#xff1a;对于 EPC 项目&#xff0c;需…...

windows传文件给mac, linux或者其他windows

如果你想把你本地windows的文件传递给mac, linux或者其他windows电脑, 你会使用什么方法呢? 方法一 经常写python的朋友, 肯定会说, 我会用 SimpleHTTPServer python -m SimpleHTTPServer 8000生成一个超链接, 然后在其他机器上直接下载这个链接. 方法二 scp命令 Use SC…...

AI音乐解决方案:1分钟可切换suno、udio、luno、kuka等多种模型,suno风控秒切换 | AI Music API

你有没有觉得&#xff0c;suno风控来了&#xff0c;就要停服了&#xff1f; 你有没有觉得&#xff0c;对接多种音乐模型&#xff0c;让你很疲乏&#xff1f; 你有没有觉得&#xff0c;音乐模型&#xff0c;中文咬字不清楚&#xff0c;让你很苦恼&#xff1f; 别怕&#xff0…...

LeetCode 热题100题解(Java版本)

一、哈希 1、两数之和 https://leetcode.cn/problems/two-sum/?envType=study-plan-v2&envId=top-100-liked使用HashMap,遍历数组,判断当前元素的“补数”是否存在,如果存在直接返回结果,否则在Map中记录当前元素及其下标。 时间复杂度 O(n) 空间复杂度 O(n) class …...

突破传统!SEARCH-R1如何让LLM与搜索引擎协同推理?

大语言模型&#xff08;LLMs&#xff09;虽强大&#xff0c;但在复杂推理和获取最新信息方面存在局限。本文介绍的SEARCH-R1框架&#xff0c;通过强化学习让LLMs能自主与搜索引擎交互&#xff0c;在多个问答数据集上性能大幅提升。想知道它是如何做到的吗&#xff1f;快来一探究…...

C语言复习笔记--字符函数和字符串函数(下)

在上篇我们了解了部分字符函数及字符串函数,下面我们来看剩下的字符串函数. strstr 的使用和模拟实现 老规矩,我们先了解一下strstr这个函数,下面看下这个函数的函数原型. char * strstr ( const char * str1, const char * str2); 如果没找到就返回NULL指针. 下面我们看下它的…...

Git SSH 密钥多个 Git 来源

1. 生成 SSH 密钥 ssh-keygen -t rsa -b 4096 -C "997959066qq.com" 2. 输入指定路径 3. 增加 config文件在.ssh文件下 # GitHub 主账户 Host github.dis.comHostName github.disney.comUser gitIdentityFile ~/.ssh/id_rsa# 其他 Git 服务器 Host github.comHost…...

华为盒式交换机堆叠配置

1. 堆叠线缆连线图(如下图) 2. 配置Swtich1: [HUAWEI] sysname Switch1 [Switch1] interface stack-port 0/1 [Switch1-stack-port0/1] port interface xgigabitethernet 0/0/3 xgigabitethernet 0/0/4 enable Warning: Enabling stack function may cause configuration l…...

map和set的使用

序列式容器和关联式容器 c标准库为我们提供了多种容器类型&#xff0c;可以大体分为两类&#xff1a;序列式容器和关联式容器。 序列式容器按照线性顺序储存数据&#xff0c;元素的位置取决与插入的时间和地点。关联式容器基于键值对存储元素&#xff0c;提供高效的键查找能力…...

VMware制作Windows虚拟机模板注意点

1 、删除右下角的热删除 这个是必须的&#xff01;虚拟机的高级参照中添加devices.hotplug “FALSE”&#xff0c;关闭USB设备热插拔&#xff0c;防止用户把虚拟桌面的网卡删除掉。 2、移除CD/DVD驱动器 这个建议&#xff01;移除CD/DVD驱动器&#xff0c;此项操作…...

公有云攻防5(云函数)

Serverless概述 无服务器(Serverless)不是表示没有服务器,而是表示当您在使用Serverless时,您无需关心底层的资源,也无需登录服务器和优化服务器,只需关注最核心的代码片段,即可跳过复杂的、繁琐的基本工作。 Serverless 拥有近乎无限的扩容能力,空闲时,不运行任何资…...

【计算机网络】第五章 局域网技术

以太网技术 以太网的诞生 网络适配器和MAC地址 单播MAC地址 广播MAC地址 CSMA/CD协议 CSMA/CD协议的基本原理 共享式以太网---广播特性 MA多址接入 CS载波监听---先听后说 CD碰撞检测---边听边说 载波监听检测到总线空闲&#xff0c;但总线并不一定空闲。 使用CSMA/CD协议的…...

傲来云分享,负载均衡:提升网站性能与稳定性

在网站或应用的流量不断增加时&#xff0c;单台服务器往往无法承载过多的并发请求&#xff0c;导致性能下降和响应延迟。负载均衡技术正是为了解决这个问题&#xff0c;它可以将流量分发到多台服务器&#xff0c;从而提高系统的可扩展性、处理能力和可靠性。今天&#xff0c;我…...

Python基础总结(九)之推导式

文章目录 一、列表推导式1.1 列表推导式的格式1.2 列表推导式的注意事项1.3 列表推导式示例 二、 字典推导式2.1 字典推导式格式2.2 字典推导式注意事项2.3 字典推导式示例 三、 元组推导式3.1 元组推导式格式3.3 元组推导式示例 Python中的推导式有列表推导式&#xff0c;字典…...

程序员学商务英文之Terms of Payment Packing

Dia-3: Packing 1 包装-1 1. I’m here to improve my communication skill of English. 我来这里是为了提升我的英文沟通技能。 2. What a co-incidence! Fancy meeting you here. 这么巧&#xff01;真没想到在这见到你。 3. Some birds aren’t meant to be caged…...

学生管理系统项目中的相关问题总结

目录 1. 项目中哪里体现出了封装性及其好处&#xff1f; 2. 项目中的setter/getter模式与封装性 什么是setter/getter&#xff1f;有什么用&#xff1f;怎么自动生成&#xff1f; 3.项目中某些类的toString()方法 4. 项目中几个常用方法解析 5. 项目中的面向对象设计 1…...

AI驱动下的企业学习:人力资源视角下的范式重构与价值觉醒

一、传统企业学习体系的系统性失效 当企业学习市场规模突破3600亿美元&#xff0c;人均年投入达1400美元的历史高位时&#xff0c;这场看似繁荣的图景背后却暗藏结构性危机。正如一份2024年研究报告所指出的&#xff0c;66%的企业仍倾向于通过外部招聘而非内部培养获取AI人才&…...

工厂模式:简单工厂模式

工厂模式 简单工厂模式&#xff1a;子类对象较多&#xff0c;则引入工厂类来创建 简单工厂模式 流程&#xff1a; 实例&#xff1a; 某电视机厂为各个品牌代工生产电视机&#xff0c;可以使用简单工厂的模式来实现 #include <iostream> #include <vector> using…...

Java并发编程-线程池

Java并发编程-线程池 线程池运行原理线程池生命周期线程池的核心参数线程池的阻塞队列线程池的拒绝策略线程池的种类newFixedThreadPoolnewSingleThreadExecutornewCachedThreadPoolnewScheduledThreadPool 创建线程池jdk的Executors(不建议&#xff0c;会导致OOM)jdk的ThreadP…...

ios17 音频加载失败问题

现象&#xff1a; 项目中有2个MP3 格式的音频&#xff0c;在iOS17 下 一个可以播&#xff0c;一个不能播&#xff0c;但是在浏览器可 正常播放 原因&#xff1a; 不能播的mp3是因为其编码格式为mpeg-4 AAC 编码格式 正常的mpeg audio 是可以播放的 解决&am…...

HarmonyOS:Navigation实现导航之页面设置和路由操作

导读 设置标题栏模式设置菜单栏设置工具栏路由操作页面跳转页面返回页面替换页面删除移动页面参数获取路由拦截 子页面页面显示类型页面生命周期页面监听和查询 页面转场关闭转场自定义转场共享元素转场 跨包动态路由系统路由表自定义路由表 示例代码 Navigation组件适用于模块…...

ADB -> pull指令拉取手机文件到电脑上

ADB Pull命令 在Android开发中&#xff0c;ADB的pull命令可以帮助我们将文件从设备拷贝到电脑上。 基本语法 adb pull <设备文件路径> [本地目标路径]<设备文件路径>&#xff1a;必需参数&#xff0c;指定要拉取的设备上的文件或文件夹[本地目标路径]&#xff1…...

Android Studio打开xml布局文件内存会快速增加如何设置

打开xml布局文件内存快速增加的原因是预览设置问题&#xff0c;直接在file-setting-editor-ui tools-editor view mode-resource: 选择code 就好了...

Spring Boot集成Keycloak

前言 本文参考A Quick Guide to Using Keycloak with Spring Boot&#xff0c;整理实战中遇到的问题。 Docker 安装 Keycloak 下载镜像 quay下载镜像 docker pull quay.io/keycloak/keycloak 失败的话&#xff0c;可再次尝试。 启动keycloak docker run -p 6060:8080 -…...

实验七 shell程序设计

实验七 shell程序设计 一、实验目的 理解shell的工作原理&#xff0c;学会编写shell脚本。 二、实验内容 1.编写不同功能的脚本程序。 2.利用chmod修改文件权限。 3.掌握脚本文件执行的方法。 三、主要实验步骤 1.创建一个名为zs_lab7的目录&#xff0c;下边实验步骤都在…...

FlaskRestfulAPI接口的初步认识

FlaskRestfulAPI 介绍 记录学习 Flask Restful API 开发的过程 项目来源&#xff1a;【Flask Restful API教程-01.Restful API介绍】 我的代码仓库&#xff1a;https://gitee.com/giteechaozhi/flask-restful-api.git 后端API接口实现功能&#xff1a;数据库访问控制&#xf…...

自定义错误码的必要性

为什么要使用错误码&#xff0c;直接返回一个错误信息不好么&#xff1f; 下面介绍一下&#xff0c;在程序开发中使用错误码的必要性~ 便于排查问题 想象你开了一家奶茶店&#xff0c;顾客下单后可能出现各种问题&#xff1a; 没珍珠了​​&#xff08;错误码&#xff1a;50…...

快手砍掉本地生活的门槛

一场本地商家的效率革命。 作者|景行 编辑|杨舟 “两斤鸡翅根七块九&#xff0c;两盒蓝莓九块钱&#xff0c;两公斤卫生纸十四块九一提。” 这是朝阳佳惠超市&#xff0c;在快手一则普通的短视频内容。 佳惠超市在辽宁省朝阳市有22家分店&#xff0c;打开佳惠超市的相关快手…...

Python+Word实现周报自动化的完整流程

一、技术方案概述 自动化报表解决方案基于以下技术组件&#xff1a; Python 作为核心编程语言python-docx 库用于处理 Word 文档pandas 库用于数据处理和分析matplotlib 或 plotly 库用于数据可视化Word 模版作为报表的基础格式 这种方案的优势在于&#xff1a;保留了 Word 文…...

PCIE Spec ---Base Address Registers

7.5.1.2.1 Base Address Registers (Offset 10h - 24h) 在 boot 到操作系统之前&#xff0c;系统软件需要生产一个内存映射的 address map &#xff0c;用于告诉系统有多少内存资源&#xff0c;以及相应功能需要的内存空间&#xff0c;所以在设备的 PCI 内存空间中就有了这个 …...

list的学习

list的介绍 list文档的介绍 list是可以在常数范围内在任意位置进行插入和删除的序列式容器&#xff0c;并且该容器可以前后双向迭代。list的底层是双向链表结构&#xff0c;双向链表中每个元素存储在互不相关的独立节点中&#xff0c;在节点中通过指针指向其前一个元素和后一…...

numpy、pandas内存优化操作整理

前言 python作为一款不怎么关注数据类型的语言&#xff0c;不同类型的数据可以往同一个变量中放置 这也就直接导致&#xff0c;作为熟悉C这种一个变量只有一个类型的程序员来说&#xff0c;在解读python程序时&#xff0c;想搞清楚变量中到底存的是什么数据类型的时候时常很头…...

【OSG学习笔记】Day 5: 坐标系与变换节点(Transform)

在前面的学习中,我们对 OpenSceneGraph(OSG)有了基础的认识。 今天,我们将深入探讨 OSG 中的坐标系和变换节点(Transform),并通过平移、旋转、缩放模型以及父子节点层级变换的实战操作,加深对这些概念的理解。 坐标系 1. 基础坐标系 在 OSG 里,使用的是右手坐标系…...

使用 Logstash 迁移 MongoDB 数据到 Easysearch

大家好&#xff01;在前面的文章中&#xff0c;我们已经详细介绍了如何通过 Logstash 和 Canal 工具实现 MySQL 数据向 Easysearch 的迁移。如果您正在使用 MongoDB 作为数据存储&#xff0c;并希望将其数据迁移到 Easysearch 中&#xff0c;这篇指南或许能为您提供一些帮助。 …...

在线查看【免费】vsd, vsdx/wmf, emf /psd, eps/pdf ,ofd, rtf/xmind/bpmn/eml/epub文件格式网

可以免费在线查看 .docx/wps/Office/wmf/ psd/ psd/eml/epub/dwg, dxf/ txt/zip, rar/ jpg/mp3 m.gszh.xyz m.gszh.xyz 免费支持以下格式文件在线查看类型 支持 doc, docx, xls, xlsx, xlsm, ppt, pptx, csv, tsv, dotm, xlt, xltm, dot, dotx, xlam, xla, pages 等 Office 办…...

富诺健康旗下运动营养品牌力爆(LIPOW):以冠军精神定义运动营养新时代

在全民健身热潮持续升温的今天&#xff0c;运动已不仅是竞技场上的较量&#xff0c;更是普通人突破自我的健康生活方式。《全民健身计划&#xff08;2021-2025年&#xff09;》中指出&#xff0c;“十三五”时期&#xff0c;全民健身国家战略深入实施&#xff0c;全民健身公共服…...

自定义多头注意力模型:从代码实现到训练优化

引言 在自然语言处理和序列生成任务中,自注意力机制(Self-Attention)是提升模型性能的关键技术。本文将通过一个自定义的PyTorch模型实现,展示如何构建一个结合多头注意力与前馈网络的序列生成模型(如文本或字符生成)。该模型通过创新的 MaxStateSuper 模块实现动态特征…...

vue部署到nginx服务器 启用gzip

要在使用Vue.js构建的应用程序上启用Nginx的Gzip压缩&#xff0c;你可以通过配置Nginx来实现这一功能&#xff0c;这样可以显著减少传输到客户端的数据量&#xff0c;从而加快页面加载速度。以下是如何配置Nginx以启用Gzip压缩的步骤&#xff1a; 1. 确认你的Vue.js应用已经构…...

Node.js和js到底什么关系

Node.js 和 JavaScript&#xff08;JS&#xff09;是紧密关联但本质不同的技术&#xff0c;它们的关系可以从以下几个关键维度进行解析&#xff1a; 1. 定义与角色 JavaScript&#xff1a; 一种高级、解释型的编程语言&#xff0c;最初设计用于浏览器端&#xff0c;负责网页的…...

如何开发一套TRS交易系统:架构设计、核心功能与风险控制

TRS&#xff08;总收益互换&#xff09;作为场外衍生品的重要工具&#xff0c;近年来在跨境投资、杠杆交易和风险对冲领域备受关注。2021年Archegos资本因TRS交易爆仓导致百亿美元级市场震荡&#xff0c;凸显了TRS系统设计的关键性。本文将从技术实现角度&#xff0c;解析TRS交…...

基于SpringBoot的高校体育馆场地预约管理系统-项目分享

基于SpringBoot的高校体育馆场地预约管理系统-项目分享 项目介绍项目摘要目录总体功能图用户实体图赛事实体图项目预览用户个人中心医生信息管理用户管理场地信息管理登录 最后 项目介绍 使用者&#xff1a;管理员 开发技术&#xff1a;MySQLJavaSpringBootVue 项目摘要 随着…...

MMIO、IOMAP 和 IOMMU 总结

MMIO、IOMAP 和 IOMMU 全面解析 &#x1f4cc; 本文将深入浅出地梳理 Linux 驱动开发中常见的三大术语&#xff1a;MMIO、iomap、IOMMU。它们看似相似&#xff0c;其实职责完全不同&#xff0c;是理解 SoC 系统架构、DMA 安全性和驱动开发的基础。 一、MMIO&#xff08;Memory-…...

Vscode开发STM32标准库

Vscode开发STM32 文章目录 引用一、文档介绍二、实际操作&#xff08;基于标准库&#xff09;总结 使用VScode开发STM32(keil)&#xff0c;基础江科大标准库的串口接收和发送。 引用 VSCodeEIDE开发STM32&#xff0c;支持标准库、HAL库、LL库&#xff0c;可以在VSCode里进行调…...

Lateral 查询详解:概念、适用场景与普通 JOIN 的区别

1. 什么是Lateral查询&#xff1f; Lateral查询&#xff08;也称为横向关联查询&#xff09;是一种特殊的子查询&#xff0c;允许子查询中引用外层查询的列&#xff08;即关联引用&#xff09;&#xff0c;并在执行时逐行对外层查询的每一行数据执行子查询。 语法上通常使用关…...