React Flow 节点事件处理实战:鼠标 / 键盘事件全解析(含节点交互代码示例)
本文为《React Agent:从零开始构建 AI 智能体》专栏系列文章。 专栏地址:https://blog.csdn.net/suiyingy/category_12933485.html。项目地址:https://gitee.com/fgai/react-agent(含完整代码示例与实战源)。完整介绍:https://blog.csdn.net/suiyingy/article/details/146983582。
1 鼠标事件
React Flow 为节点提供了丰富的鼠标事件,如onClick、onDoubleClick、onMouseEnter、onMouseLeave等。这些事件可以用于实现节点的各种交互功能。onClick事件可以用于打开节点的详细信息面板,onMouseEnter事件可以用于显示节点的提示信息。
const InfoNode = ({ data }) => {return (<divclassName="react-flow__node-default"onClick={() => alert('This is the info of the node')}onMouseEnter={() => console.log('Mouse entered the node')}onMouseLeave={() => console.log('Mouse left the node')}>{data.label}</div>);
};
alert和console.log都是 JavaScript 中常用的工具,在调试程序时发挥着不同作用。alert是浏览器提供的一个全局函数,它会弹出一个包含指定消息的对话框,此对话框会暂停当前程序的执行,直到用户点击“确定”关闭它。在调试中,它可用于快速确认变量的值或某些代码块是否被执行。不过,由于它会中断程序,若频繁使用会影响用户体验,并且只能展示简单的文本信息。
console.log则是向浏览器的控制台输出信息。控制台是开发者调试代码的重要工具,它可以显示各种类型的数据,如字符串、数字、对象、数组等。使用console.log不会中断程序执行,所以能在程序运行过程中持续输出信息,方便开发者观察变量的变化和程序的执行流程。
若要在浏览器中查看console.log输出的信息,不同浏览器的操作方式略有不同,但基本步骤一致。首先,在浏览器中打开包含该代码的网页;然后,打开开发者工具,常见的方法是在网页上右键点击,选择 “检查” 或者 “审查元素”,也可以使用快捷键(通常为 F12);打开开发者工具后,切换到 “控制台”(Console)面板,在这里就能看到console.log输出的信息。运行上面程序后会有如下输出。
图1 鼠标事件
我们也可对连接点 Handle 设置事件,下面程序可双击修改 Handle 名称。
import React, { useCallback, useContext, useState } from 'react';
import { ReactFlow, Handle, useNodesState, useEdgesState, addEdge } from 'reactflow';
import 'reactflow/dist/style.css';
import { FiDatabase, FiCloud } from 'react-icons/fi';
import { toast, Toaster } from 'react-hot-toast';// 创建上下文用于节点更新
const NodeUpdateContext = React.createContext();const CustomNode = ({ id, data, selected }) => {const setNodes = useContext(NodeUpdateContext);const [editingHandle, setEditingHandle] = useState(null);const [tempName, setTempName] = useState('');const handleNameChange = (handleId, newName) => {setNodes(nds => nds.map(node => node.id === id ? {...node,data: {...node.data,handleNames: {...node.data.handleNames,[handleId]: newName}}} : node));};const startEdit = (handleId) => {setEditingHandle(handleId);setTempName(data.handleNames?.[handleId] || '');};const confirmEdit = () => {if (editingHandle) {handleNameChange(editingHandle, tempName);setEditingHandle(null);}};const renderHandle = (localId, position, type) => {const fullId = `${id}-${localId}`;const isEditing = editingHandle === fullId;const displayName = data.handleNames?.[fullId] || '';return (<div className={`handle-group handle-${position}`}>{isEditing && (<inputtype="text"value={tempName}onChange={(e) => setTempName(e.target.value)}onBlur={confirmEdit}onKeyPress={(e) => e.key === 'Enter' && confirmEdit()}autoFocusclassName="handle-input"/>)}<Handleid={fullId}type={type}position={position}className={`!bg-${type === 'target' ? 'teal' : 'purple'}-500`}onDoubleClick={() => startEdit(fullId)}title={displayName}/></div>);};return (<div className={`custom-node ${selected ? 'selected' : ''}`}>{renderHandle('target-top', 'top', 'target')}<div className="node-header"><FiCloud className="node-icon" /><h3 className="node-title">{data.label}</h3></div><div className="node-body"><FiDatabase className="node-icon" /><span className="node-info">{data.content}</span></div>{renderHandle('source-bottom', 'bottom', 'source')}{renderHandle('source-right', 'right', 'source')}</div>);
};// 初始节点配置(保持其他内容不变,增加handleNames字段)
const initialNodes = [{ id: '1', position: { x: 0, y: 0 }, data: { label: '开始节点',content: '输入数据源',handleNames: {'1-target-top': '输入','1-source-bottom': '主输出','1-source-right': '备选输出'}},type: 'custom',},{ id: '2', position: { x: 200, y: 150 }, data: { label: '处理节点',content: '数据处理流程',handleNames: {'2-target-top': '输入','2-source-bottom': '结果输出','2-source-right': '日志输出'}},type: 'custom',},
];const initialEdges = [{ id: 'e1-2', source: '1', target: '2',animated: true,style: { stroke: '#94a3b8' },
}];const nodeTypes = {custom: CustomNode,
};// 样式增加连接点标签相关样式
const nodeStyle = `.custom-node {position: relative;background: linear-gradient(145deg, #ffffff, #f1f5f9);border-radius: 8px;border: 2px solid #cbd5e1;box-shadow: 0 4px 6px -1px rgba(0, 0, 0, 0.1);padding: 16px;min-width: 200px;transition: all 0.2s ease;}.custom-node.selected {border-color: #6366f1;box-shadow: 0 4px 15px rgba(99, 102, 241, 0.2);}.custom-node:hover {transform: translateY(-2px);}.node-header {display: flex;align-items: center;margin-bottom: 12px;border-bottom: 1px solid #e2e8f0;padding-bottom: 8px;}.node-title {margin: 0;font-size: 1.1rem;color: #1e293b;margin-left: 8px;}.node-body {display: flex;align-items: center;color: #64748b;}.node-icon {font-size: 1.2rem;margin-right: 8px;color: #6366f1;}.node-info {font-size: 0.9rem;}.react-flow__handle {width: 24px;height: 14px;border-radius: 3px;border: none;cursor: pointer;transition: background-color 0.2s;}.react-flow__handle:hover {filter: brightness(1.2);}/* 按位置调整具体坐标 */
.react-flow__handle[data-position="top"] {top: -7px !important;left: 50% !important;transform: translateX(-50%) !important;
}.react-flow__handle[data-position="bottom"] {bottom: -7px !important;left: 50% !important;transform: translateX(-50%) !important;
}.react-flow__handle[data-position="right"] {right: -12px !important;top: 50% !important;transform: translateY(-50%) !important;
}.handle-group {z-index: 10;}.handle-input {position: absolute;width: 80px;padding: 2px 4px;font-size: 0.8rem;border: 1px solid #6366f1;border-radius: 4px;background: white;z-index: 100;}.handle-top .handle-input {bottom: calc(100% + 8px);left: 50%;transform: translateX(-50%);}.handle-bottom .handle-input {top: calc(100% + 8px);left: 50%;transform: translateX(-50%);}.handle-right .handle-input {left: calc(100% + 8px);top: 50%;transform: translateY(-50%);}
`;export default function App() {const [nodes, setNodes, onNodesChange] = useNodesState(initialNodes);const [edges, setEdges, onEdgesChange] = useEdgesState(initialEdges);return (<NodeUpdateContext.Provider value={setNodes}><div style={{ height: '100vh', background: '#f8fafc' }}><style>{nodeStyle}</style><Toaster position="top-right" /><ReactFlownodes={nodes}edges={edges}onNodesChange={onNodesChange}onEdgesChange={onEdgesChange}onConnect={useCallback((conn) => setEdges(eds => addEdge({...conn,animated: true,style: conn.sourceHandle?.endsWith('-right') ? { stroke: '#ec4899' } : { stroke: '#94a3b8' }}, eds)), [setEdges])}nodeTypes={nodeTypes}isValidConnection={useCallback((conn) => conn.source !== conn.target && !edges.some(e => e.target === conn.target), [edges])}fitViewdefaultEdgeOptions={{ type: 'smoothstep', animated: true,style: { strokeWidth: 2 }}}/></div></NodeUpdateContext.Provider>);
}
2 键盘事件
在某些场景下,需要通过键盘来操作节点,React Flow 支持键盘事件。可以监听onKeyDown、onKeyUp等事件,实现节点的键盘导航、快捷键操作等功能。例如,通过按下特定的快捷键来删除节点:
const InfoNode = ({ data }) => {const handleKeyDown = (event) => {if (event.key === 'Delete') {alert('节点将被删除');}};return (<divclassName="react-flow__node-default"tabIndex={0}onKeyDown={handleKeyDown}>{data.label}</div>);
};
3 拖动与连接事件
与节点的拖动和连接相关的事件也非常重要。onDragStart、onDrag、onDragEnd事件可以用于处理节点拖动过程中的逻辑,如更新节点的位置、限制拖动范围等。onConnect事件则在节点之间建立连接时触发,可以用于验证连接的合法性、更新数据等。需要注意,直接在自定义节点组件上使用 draggable 和原生的 onDragStart 等事件会与 React Flow 的拖拽行为冲突。如果开发者希望为节点添加额外的拖动逻辑,可以利用 React Flow 自身提供的事件(例如 onNodeDragStart、onNodeDrag 和 onNodeDragStop)来实现。
import React, { useCallback } from 'react';
import {ReactFlow,Handle,useNodesState,useEdgesState,addEdge,
} from 'reactflow';
import 'reactflow/dist/style.css';// 自定义节点组件
const InfoNode = ({ data }) => {return (<div className="react-flow__node-default">{data.label}</div>);
};const nodeTypes = {infoNode: InfoNode,
};// 节点配置
const infoNode = {id: 'info-node-1',type: 'infoNode',data: { label: 'Info Node' },position: { x: 250, y: 100 },
};const initialNodes = [infoNode];
const initialEdges = [];export default function App() {const [nodes, setNodes, onNodesChange] = useNodesState(initialNodes);const [edges, setEdges, onEdgesChange] = useEdgesState(initialEdges);const onConnect = useCallback((params) => setEdges((eds) => addEdge(params, eds)),[setEdges]);const handleNodeDragStart = (event, node) => {console.log('Node drag started:', node);};const handleNodeDrag = (event, node) => {console.log('Node dragging:', node);};const handleNodeDragStop = (event, node) => {console.log('Node drag stopped:', node);};return (<div style={{ width: '100vw', height: '100vh' }}><ReactFlownodes={nodes}edges={edges}nodeTypes={nodeTypes}onNodesChange={onNodesChange}onEdgesChange={onEdgesChange}onConnect={onConnect}onNodeDragStart={handleNodeDragStart}onNodeDrag={handleNodeDrag}onNodeDragStop={handleNodeDragStop}fitView/></div>);
}
4 创建自定义事件
除了使用内置事件,开发者还可以根据项目需求创建自定义事件。在 React 里,自定义事件属于开发者自行定义的事件,它可以让组件之间更好地进行通信和交互。它能够在特定条件达成时触发特定的行为,从而增强组件的灵活性与可复用性。而 useEffect 是 React 提供的一个 Hook,其主要作用是处理副作用操作,像数据获取、订阅、DOM 操作等。它会在组件渲染之后执行,并且可以依据依赖项数组的变化来决定是否重新执行。通过结合自定义事件和 useEffect,我们能够在组件的生命周期内灵活地触发和处理自定义事件,从而实现更为复杂的交互逻辑。
const InfoNode = ({ data }) => {const nodeRef = useRef(null);useEffect(() => {const nodeEl = nodeRef.current;if (!nodeEl) return;const handleClick = () => alert('This is the info of the node');const handleMouseEnter = () => console.log('Mouse entered the node');const handleMouseLeave = () => console.log('Mouse left the node');nodeEl.addEventListener('click', handleClick);nodeEl.addEventListener('mouseenter', handleMouseEnter);nodeEl.addEventListener('mouseleave', handleMouseLeave);// 清理return () => {nodeEl.removeEventListener('click', handleClick);nodeEl.removeEventListener('mouseenter', handleMouseEnter);nodeEl.removeEventListener('mouseleave', handleMouseLeave);};}, []); // 空依赖数组,确保只在挂载和卸载时执行return (<divref={nodeRef}className="react-flow__node-default">{data.label}</div>);
};
5 事件冒泡与捕获
开发者在处理多个节点的事件时需要了解事件冒泡和捕获机制。事件冒泡是指事件从最内层的元素开始触发,然后逐级向上传播到外层元素;事件捕获则相反,从最外层元素开始,逐级向内层元素传播。合理利用事件冒泡和捕获能够实现更高效的事件处理。例如,在一个包含多个节点的区域中,我们可以在父元素上监听事件,通过事件属性来判断具体是哪个节点触发了事件,从而减少事件处理函数的重复定义。
import React, { useCallback } from 'react';
import {ReactFlow,Handle, useNodesState,useEdgesState,addEdge,
} from 'reactflow';
import 'reactflow/dist/style.css';// 自定义节点组件
const InfoNode = ({ data }) => {const { onNodeClick, onNodeMouseEnter, onNodeMouseLeave, label } = data;return (<divclassName="react-flow__node-default"onClick={() => onNodeClick(label)}onMouseEnter={() => onNodeMouseEnter(label)}onMouseLeave={() => onNodeMouseLeave(label)}>{label}</div>);
};const nodeTypes = {infoNode: InfoNode,
};// 节点配置
const infoNode1 = {id: 'info-node-1',type: 'infoNode',data: { label: 'Info Node 1' },position: { x: 250, y: 100 },
};const infoNode2 = {id: 'info-node-2',type: 'infoNode',data: { label: 'Info Node 2' },position: { x: 500, y: 100 },
};const initialNodes = [infoNode1, infoNode2];
const initialEdges = [];export default function App() {const [nodes, setNodes, onNodesChange] = useNodesState(initialNodes);const [edges, setEdges, onEdgesChange] = useEdgesState(initialEdges);const onNodeClick = (label) => {alert(`This is the info of ${label}`);};const onNodeMouseEnter = (label) => {console.log(`Mouse entered the ${label}`);};const onNodeMouseLeave = (label) => {console.log(`Mouse left the ${label}`);};const newNodes = nodes.map((node) => ({...node,data: {...node.data,onNodeClick,onNodeMouseEnter,onNodeMouseLeave,},}));const onConnect = useCallback((params) => setEdges((eds) => addEdge(params, eds)),[setEdges]);return (<div style={{ width: '100vw', height: '100vh' }}><ReactFlownodes={newNodes}edges={edges}nodeTypes={nodeTypes}onNodesChange={onNodesChange}onEdgesChange={onEdgesChange}onConnect={onConnect}fitView/></div>);
}
立即关注获取最新动态
点击订阅《React Agent 开发专栏》,每周获取智能体开发深度教程。项目代码持续更新至React Agent 开源仓库,欢迎 Star 获取实时更新通知!FGAI 人工智能平台:FGAI 人工智能平台
相关文章:
React Flow 节点事件处理实战:鼠标 / 键盘事件全解析(含节点交互代码示例)
本文为《React Agent:从零开始构建 AI 智能体》专栏系列文章。 专栏地址:https://blog.csdn.net/suiyingy/category_12933485.html。项目地址:https://gitee.com/fgai/react-agent(含完整代码示例与实战源)。完整介绍…...
AIGC在电商行业的应用:革新零售体验
AIGC在电商行业的应用:革新零售体验 引言 人工智能生成内容(AIGC)正在深刻改变电商行业的格局。从个性化推荐到智能客服,从产品描述生成到虚拟试衣,AIGC技术正在为电商平台带来前所未有的创新和效率提升。本文将深入探…...
【数据结构】线性表--队列
【数据结构】线性表--队列 一.什么是队列二.队列的实现1.队列结构定义:2.队列初始化函数:3.队列销毁函数:4.入队列函数(尾插):5.出队列函数(头删):6.取队头元素ÿ…...
CSS- 4.1 浮动(Float)
本系列可作为前端学习系列的笔记,代码的运行环境是在HBuilder中,小编会将代码复制下来,大家复制下来就可以练习了,方便大家学习。 HTML系列文章 已经收录在前端专栏,有需要的宝宝们可以点击前端专栏查看! 点…...
Node.js 源码架构详解
Node.js 的源码是一个庞大且复杂的项目,它主要由 C 和 JavaScript 构成。要完全理解每一部分需要大量的时间和精力。我会给你一个高层次的概述,并指出一些关键的目录和组件,帮助你开始探索。 Node.js 的核心架构 Node.js 的核心可以概括为以…...
OpenCV级联分类器
概念 OpenCV 级联分类器是一种基于 Haar 特征、AdaBoost 算法和级联结构的目标检测方法,通过多阶段筛选快速排除非目标区域,实现高效实时检测(如人脸、行人等)。 加载级联分类器 // 加载级联分类器CascadeClassifier cascade;// …...
远程主机状态监控-GPU服务器状态监控-深度学习服务器状态监控
远程主机状态监控-GPU服务器状态监控-深度学习服务器状态监控 ⭐️ 推荐文章: DockerPyCharm远程调试&环境隔离解决方案 1. 整体架构 在本监控系统中,我们采用了Prometheus作为核心监控解决方案,并结合Node Exporter和Grafana构建了一个完整的监控体…...
谈谈未来iOS越狱或巨魔是否会消失
2024年10月的预测,先说结论: 巨魔iOS17.1消失概率为99%。 因为巨魔强依赖的漏洞就是一个签名漏洞,攻击面有限又经过2轮修复,第3次出现漏洞的概率极低。而越狱的话由于系统组件和服务较多,所以出现漏洞概率高攻击面多&…...
【OpenGL学习】(二)OpenGL渲染简单图形
文章目录 【OpenGL学习】(二)OpenGL渲染简单图形OpenGL渲染图形流程顶点,图元和片元VAO,VBO ,EBO着色器示例:使用OpenGL渲染三角形 【OpenGL学习】(二)OpenGL渲染简单图形 OpenGL渲…...
学习深度学习是否要先学习机器学习?
有小伙伴问我,最近做毕设要做一个神经网络的课题,想请教一下需不需要把机器学习也都学习一遍? 永远正确的回答是:建议先学机器学习,再学深度学习。 上面那句你从哪都挑不出毛病,毕竟机器学习是深度学习的基…...
六、绘制图片
文章目录 1.创建一个红色图片2.加载bmp图片3.加载png、jpg图片 前面的几个示例,我们已经展示过如果在Linux系统下使用xlib接口向窗口中绘制文本、线、矩形;并设置文本、线条的颜色。并利用xlib提供的接口结合事件处理机制完成了一个自绘按钮控件功能。有…...
【OpenCV】基本数据类型及常见图像模式
是什么?能做什么?解决什么问题?为什么用它? OpenCV:是一个基于开源发行的跨平台计算机视觉库,实现 一、应用场景: 目标识别:人脸、车辆、车牌...自动驾驶医学影像分析视频内容理解与分析&…...
C# WPF .NET Core和.NET5之后引用System.Windows.Forms的解决方案
双击项目名称打开工程文件(.csporj)添加“Microsoft.WindowsDesktop.App.WindowsForms”引用; <Project Sdk"Microsoft.NET.Sdk"><PropertyGroup><OutputType>WinExe</OutputType><TargetFramework&g…...
Mysql 8.0.32 union all 创建视图后中文模糊查询失效
记录问题,最近在使用union all聚合了三张表的数据,创建视图作为查询主表,发现字段值为中文的筛选无法生效.......... sql示例: CREATE OR REPLACE VIEW test_view AS SELECTid,name,location_address AS address,type,"1" AS data_type,COALESCE ( update_time, cr…...
PYTHON训练营DAY28
类 (一)题目1:定义圆(Circle)类 要求: 包含属性:半径 radius。包含方法: calculate_area():计算圆的面积(公式:πr)。calculate_circ…...
pytorch小记(二十一):PyTorch 中的 torch.randn 全面指南
pytorch小记(二十一):PyTorch 中的 torch.randn 全面指南 PyTorch 中的 torch.randn 全面指南一、接口定义二、参数详解三、常见使用场景四、位置参数 vs. Tuple 传参 —— 数值示例五、必须用关键字传入小结 PyTorch 中的 torch.randn 全面指…...
LeetCode 第 45 题“跳跃游戏 II”
好的,我来帮你解释一下 LeetCode 第 45 题“跳跃游戏 II”,这是一道经典的贪心算法题目。 题目描述: 给你一个非负整数数组 nums,你最初位于数组的第一个位置。数组中的每个元素代表你在该位置可以跳跃的最大长度。你的目标是使用…...
【leetcode】逐层探索:BFS求解最短路的原理与实践
前言 🌟🌟本期讲解关于力扣的几篇题解的详细介绍~~~ 🌈感兴趣的小伙伴看一看小编主页:GGBondlctrl-CSDN博客 🔥 你的点赞就是小编不断更新的最大动力 🎆那么废话不…...
副业小程序YUERGS,从开发到变现
文章目录 我为什么写这个小程序网站转小程序有什么坑有什么推广渠道个人开发者如何变现简单介绍YUERGS小程序给独立开发者一点小建议 我为什么写这个小程序 关注我的粉丝应该知道,我在硕士阶段就已经掌握了小程序开发技能,并写了一个名为“约球online”…...
Vue-键盘事件
键盘事件 回车事件 回车输出Input控件输入的内容 代码 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><title>键盘事件</title><!-- 引入Vue --><script type"text/javascript&quo…...
区块链可投会议CCF C--IPCCC 2025 截止6.7 附录用率
Conference:44th IEEE -- International Performance Computing and Communications Conference CCF level:CCF C Categories:计算机网络 Year:2025 Conference time:Nov 21 – 23, 2025 Austin, Texas, USA 录用率…...
Linux `mkdir` 命令深度解析与高阶应用指南
Linux `mkdir` 命令深度解析与高阶应用指南 一、核心功能解析1. 基本作用2. 与类似工具对比二、选项系统详解1. 常用基础选项2. 高级选项组合三、高阶应用场景1. 自动化部署系统2. 安全审计合规3. 容器环境初始化4. 多用户协作体系四、特殊文件处理1. 符号链接处理2. 挂载点管理…...
JVM 调优实战入门:从 GC 日志分析到参数调优
手把手教你理解 GC 日志、识别性能瓶颈并合理配置 JVM 参数! 你是否曾遇到线上系统莫名卡顿、内存暴涨甚至频繁 Full GC? 本篇文章将带你从实际 GC 日志出发,深入剖析 JVM 性能问题,并学会如何通过参数调优提升系统稳定性和吞吐能…...
论文解读:ICLR2025 | D-FINE
[2410.13842] D-FINE: Redefine Regression Task in DETRs as Fine-grained Distribution Refinement D-FINE 是一款功能强大的实时物体检测器,它将 DETRs 中的边界框回归任务重新定义为细粒度分布细化(FDR),并引入了全局最优定位…...
Kafka 生产者工作流程详解
以下是 Kafka 生产者工作流程的清晰分步解释,结合关键机制与用户数据: 1. 生产者初始化与数据发送 主线程创建生产者对象,调用 send(ProducerRecord) 发送消息。 拦截器(可选):可添加自定义逻辑(…...
leetcode 239. 滑动窗口最大值
暴力解法是一种简单直接的方法,虽然效率较低,但可以帮助你更好地理解问题的逻辑。以下是使用暴力解法解决“滑动窗口最大值”问题的 C 实现。 暴力解法的思路 遍历每个滑动窗口: 使用一个外层循环,从数组的起始位置开始ÿ…...
第3章 自动化测试:从单元测试到硬件在环(HIL)
在前两章中,我们已完成从环境搭建到流水线编译的自动化配置。为了真正保障软件质量、降低回归风险,本章将聚焦测试自动化,涵盖从最基础的单元测试,到集成测试,再到硬件在环(Hardware-in-the-Loop, HIL)测试的全流程。通过脚本驱动、测试报告可视化和与 CI 平台深度集成,…...
flutter 配置 安卓、Ios启动图
android 配置启动图 launch_background.xml <?xml version"1.0" encoding"utf-8"?> <!-- Modify this file to customize your launch splash screen --> <layer-list xmlns:android"http://schemas.android.com/apk/res/android&…...
实验八 基于Python的数字图像问题处理
一、实验目的 培养利用图像处理技术解决实际问题的能力。 培养利用图像处理技术综合设计实现的能力。 掌握在Python环境下解决实际问题的能力。 熟练掌握使用cv2库对图像进行处理 熟练掌握使用区域生长法提取图片中感兴趣的区域 二、实验内容 本次实验内容为…...
Dockerfile学习指南
目录 一、Dockerfile 本质与价值 二、基础语法结构 1. 指令格式 2. 核心指令详解 三、构建流程解析 1. 典型构建过程 2. 分层构建原理 四、高级特性 1. 多阶段构建 2. 环境变量管理 3. 健康检查 五、最佳实践指南 1. 优化建议 2. 安全实践 六、典型应用场景 1. …...
数据库原理及其应用 第六次作业
题目 参考答案 题目1. 教材P148第1题 问题:什么是数据库的安全性? 答案:数据库的安全性是指保护数据库以防止不合法的使用所造成的数据泄露、更改或破坏 。它通过用户身份鉴别、存取控制(包括自主存取控制和强制存取控制&#x…...
c/c++的opencv的轮廓匹配初识
OpenCV 轮廓匹配:形状识别与比较 📐✨ 轮廓匹配是计算机视觉中一个重要的技术,它允许我们比较两个形状的相似度。OpenCV 提供了强大的函数来实现这一功能,核心是 cv::matchShapes()。本文将引导你了解轮廓匹配的基本原理、OpenCV…...
Oracle APEX IR报表下载CSV文件的方法
目录 0. 准备工作 1. 下载--自定义SQL 2. 下载--检索结果 0. 准备工作 -- 建表 CREATE TABLE T_DL_EMP(EMPNO NUMBER(4) NOT NULL -- 雇员编号,由四个数字组成。, ENAME VARCHAR2(10) -- 雇员姓名,由10个字符组成。, JOB …...
JVM 双亲委派机制
一、从 JDK 到 JVM:Java 运行环境的基石 在 Java 开发领域,JDK(Java Development Kit)是开发者的核心工具包。它不仅包含了编译 Java 代码的工具(如 javac),还内置了 JRE(Java Run…...
shell脚本之条件判断,循环控制,exit详解
if条件语句的语法及案例 一、基本语法结构 1. 单条件判断 if [ 条件 ]; then命令1命令2... fi2. 双分支(if-else) if [ 条件 ]; then条件为真时执行的命令 else条件为假时执行的命令 fi3. 多分支(if-elif-else) if [ 条件1 ]…...
什么是私有IP地址?如何判断是不是私有ip地址
在互联网的世界中,IP地址是设备之间通信的基础标识。无论是浏览网页、发送邮件还是在线游戏,IP地址都扮演着至关重要的角色。然而,并非所有的IP地址都是公开的,有些IP地址被保留用于内部网络,这就是我们所说的私有IP地…...
BGP路由策略 基础实验
要求: 1.使用Preva1策略,确保R4通过R2到达192.168.10.0/24 2.用AS_Path策略,确保R4通过R3到达192.168.11.0/24 3.配置MED策略,确保R4通过R3到达192.168.12.0/24 4.使用Local Preference策略,确保R1通过R2到达192.168.1.0/24 …...
Java 原生网络编程(BIO | NIO | Reactor 模式)
1、基本常识 Socket 是应用层与 TCP/IP 协议族通信的中间软件抽象层,是一组接口,使用了门面模式对应用层隐藏了传输层以下的实现细节。TCP 用主机的 IP 地址加上主机端口号作为 TCP 连接的端点,该端点叫做套接字 Socket。 比如三次握手&…...
大语言模型 11 - 从0开始训练GPT 0.25B参数量 MiniMind2 准备数据与训练模型 DPO直接偏好优化
写在前面 GPT(Generative Pre-trained Transformer)是目前最广泛应用的大语言模型架构之一,其强大的自然语言理解与生成能力背后,是一个庞大而精细的训练流程。本文将从宏观到微观,系统讲解GPT的训练过程,…...
【Java ee初阶】HTTP(4)
构造HTTP请求 1)开发中,前后端交互。浏览器运行的网页中,构造出HTTP请求 2)调试阶段,通过构造HTTP请求测试服务器 朴素的方案: 通过tcp socket 的方式构造HTTP请求 按照HTTP请求格式,往TCP…...
永久免费!专为 Apache Doris 打造的可视化数据管理工具 SelectDB Studio V1.1.0 重磅发布!
作为全球领先的开源实时数据仓库, Apache Doris Github Stars 已超过 13.6k,并在 5000 余家中大型企业生产环境得到广泛应用,支撑业务核心场景,成为众多企业数据分析基础设施不可或缺的重要基座。过去,Apache Doris 用…...
React中useDeferredValue与useTransition终极对比。
文章目录 前言一、核心差异对比二、代码示例对比1. useDeferredValue:延迟搜索结果更新2. useTransition:延迟路由切换 三、应用场景总结四、注意事项五、原理剖析1. 核心机制对比2. 关键差异3. 代码实现原理 总结 前言 在React的并发模式下,…...
Git 项目切换到新的远程仓库地址
已存在的 Git 项目切换到新的远程仓库地址,比如你要换成: gitgithub.com:baoing/test-app.git步骤如下: 查看当前的远程仓库地址(可选) git remote -v你会看到类似: origin gitgithub.com:old-user/old…...
科技晚报 AI 速递:今日科技热点一览 丨 2025 年 5 月 17 日
科技晚报AI速递:今日科技热点一览 丨2025年5月17日 我们为您汇总今日的科技领域最新动向,带您快速了解前沿技术、突破性研究及行业趋势。 黄仁勋劝特朗普:AI 芯片出口规则得改,中国紧追其后:英伟达 CEO 黄仁勋在华盛顿 “山与谷论…...
基于OpenCV的SIFT特征和FLANN匹配器的指纹认证
文章目录 引言一、概述二、代码解析1. 图像显示函数2. 核心认证函数3. 匹配点筛选4. 认证判断 三、主程序四、技术要点五、总结 引言 在计算机视觉领域,图像特征匹配是一个非常重要的技术,广泛应用于物体识别、图像拼接、运动跟踪等场景。今天将介绍一个…...
【Linux】共享内存
🌻个人主页:路飞雪吖~ 🌠专栏:Linux 目录 ☃️共享内存 🪄 shmget函数 用来创建共享内存 ✨共享内存的管理指令: 🌠 shmid VS key ✨共享内存函数 🍔 shmget() 创建共享内存 &a…...
Cookie、Session、Token
Cookie 1. 什么是cookie? Cookie 是一种由服务器发送到客户端浏览器的小数据片段,用于存储用户的状态信息。例如,用户登录状态或用户偏好设置可以通过Cookie进行管理。计算机cookie更正式地称为 HTTP cookie、网络 cookie、互联网 cookie 或浏览器 coo…...
设计模式Java
UML类图 概述 类图(Class diagram)是显示了模型的静态结构,特别是模型中存在的类、类的内部结构以及它们与其他类的关系等。类图不显示暂时性的信息。类图是面向对象建模的主要组成部分。 类图的作用 在软件工程中,类图是一种静态的结构图,…...
Word文档图片排版与批量处理工具推荐
先放下载链接:夸克网盘下载 前几天给大家推荐了 Excel 图片调整软件,当时好多小伙伴问有没有 Word 相关的软件。我在网上找了一圈都没找到合适的,最后在我好久之前记录的一个文档里发现了,这不,马上就来给大家推荐,有…...
[案例五] 实体——赋值质量
最近翻阅了实验室其他人编写的一个“质量赋值”功能,能够直接为实体或组件设定质量。出于好奇,我对其进行了分析。由于自己平时没有用到该功能,所以也借此机会学习一下。 在分析过程中,我发现NX 官方其实并没有提供直接修改质量的功能。一般来说,质量是通过“密度 体积”…...