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

React从基础入门到高级实战:React 核心技术 - 表单处理与验证深度指南

React 表单处理与验证深度指南

在现代 Web 应用中,表单是用户与应用交互的核心方式之一。无论是注册、登录、结账还是数据提交,表单都扮演着至关重要的角色。React 作为一款流行的前端框架,提供了多种处理表单的工具和方法,帮助开发者构建高效、用户友好的表单体验。

本文专为需要处理用户输入的开发者设计,旨在帮助你全面掌握 React 中的表单处理与验证技术。我们将从基础概念讲起,逐步深入到高级实践,包括受控组件与非受控组件、表单验证、动态表单、错误处理与用户反馈等内容。通过一个用户注册表单案例和一个多步骤结账流程表单练习,你将学会如何高效地管理表单状态和验证用户输入。此外,我们将特别推荐 React Hook Form,并深入探讨其性能优势。

文章内容通俗易懂,同时保持深度和丰富性,包含大量代码示例和实践案例,适合希望系统学习 React 表单开发的开发者。


1. 引言

表单是 Web 应用中不可或缺的组成部分,承担着用户数据收集、验证和提交的重要任务。在传统的 HTML 中,表单处理依赖 DOM 的原生行为,而在 React 中,表单管理与组件状态紧密结合,提供了更高的灵活性和控制力。

React 中的表单处理主要分为受控组件非受控组件两种方式,每种方式都有其独特的优势和适用场景。此外,随着应用的复杂性增加,表单验证、动态字段管理和错误处理成为开发者必须掌握的关键技能。本文将通过理论讲解、代码示例和实践案例,带你逐步掌握这些技术。

我们还将重点介绍 React Hook Form,一个轻量、高效的表单管理库,它通过非受控组件的方式减少渲染开销,成为现代 React 开发的首选工具。无论你是初学者还是有经验的开发者,本文都将为你提供实用的指导和深入的洞察。


2. 受控组件与非受控组件

在 React 中,表单元素(如 <input><textarea><select>)的管理方式与传统 HTML 有所不同。React 提供了两种主要方法:受控组件非受控组件。理解它们的区别和应用场景是掌握 React 表单处理的第一步。

2.1 受控组件

受控组件是指表单元素的值由 React 的状态(state)控制。开发者通过 state 设置表单的值,并通过事件处理函数更新 state,实现数据的单向流动。

特点
  • 表单元素的值与 React 的 state 保持同步。
  • 每次用户输入都会触发状态更新和组件重新渲染。
  • 适合需要实时验证或动态交互的场景。
代码示例

以下是一个简单的受控输入框示例:

import { useState } from 'react';function ControlledInput() {const [value, setValue] = useState('');const handleChange = (event) => {setValue(event.target.value);};return (<div><label>受控输入</label><inputtype="text"value={value}onChange={handleChange}placeholder="请输入内容"/><p>当前值: {value}</p></div>);
}
  • value={value}:将输入框的值绑定到 state。
  • onChange={handleChange}:监听输入变化并更新 state。
优势
  • 实时控制:可以立即获取和验证用户输入。
  • 动态行为:易于实现条件渲染、表单联动等功能。
  • 一致性:与 React 的数据驱动理念高度契合。
注意事项
  • 频繁的状态更新可能导致性能问题,尤其是在大型表单中。
  • 需要为每个表单字段定义 state 和事件处理函数。

2.2 非受控组件

非受控组件是指表单元素的值由 DOM 自身管理,React 不直接控制其状态。开发者通过 ref 获取 DOM 元素的值。

特点
  • 表单数据存储在 DOM 中,与传统 HTML 表单行为类似。
  • 不需要为每个字段维护 state,减少渲染开销。
  • 适合简单表单或性能敏感的场景。
代码示例

以下是一个非受控输入框示例:

import { useRef } from 'react';function UncontrolledInput() {const inputRef = useRef(null);const handleSubmit = () => {alert(`输入值: ${inputRef.current.value}`);};return (<div><label>非受控输入</label><inputtype="text"ref={inputRef}placeholder="请输入内容"/><button onClick={handleSubmit}>提交</button></div>);
}
  • ref={inputRef}:通过 ref 引用 DOM 节点。
  • inputRef.current.value:在需要时获取输入值。
优势
  • 性能更高:无需频繁更新 state 和重新渲染。
  • 简单直接:代码量少,接近原生 HTML 表单。
注意事项
  • 不适合需要实时验证的场景。
  • 数据获取依赖手动操作,灵活性较低。

2.3 对比与选择

特性受控组件非受控组件
状态管理由 React state 管理由 DOM 管理
数据同步实时同步按需获取
性能可能频繁渲染渲染开销小
验证易于实时验证通常提交时验证
适用场景复杂表单、动态交互简单表单、性能敏感
选择建议
  • 受控组件:推荐用于大多数场景,特别是需要实时验证、动态字段或复杂交互的表单。
  • 非受控组件:适用于简单的静态表单或性能要求极高的应用。

3. 表单验证

表单验证是确保用户输入数据有效性、完整性和安全性的重要环节。在 React 中,开发者可以手动编写验证逻辑,但这往往繁琐且难以维护。借助现代工具如 React Hook FormYup,我们可以更高效地实现表单验证。

3.1 React Hook Form 简介

React Hook Form 是一个轻量、高性能的表单管理库,专为 React 函数组件设计。它通过 Hook API 提供表单状态管理、验证和提交功能,极大简化了开发流程。

核心优势
  • 性能优越:基于非受控组件,减少不必要的渲染。
  • 简洁 API:易于学习和集成。
  • 灵活验证:支持多种验证库(如 Yup、Zod)。
  • 动态支持:轻松处理动态字段。
安装
npm install react-hook-form

3.2 基本使用

以下是一个简单的表单示例,使用 React Hook Form 实现基础验证:

import { useForm } from 'react-hook-form';function SimpleForm() {const { register, handleSubmit, formState: { errors } } = useForm();const onSubmit = (data) => {console.log('表单数据:', data);};return (<div><label>用户名</label><input {...register('username', { required: '用户名必填' })} />{errors.username && <p>{errors.username.message}</p>}<label>密码</label><input type="password" {...register('password', { required: '密码必填' })} />{errors.password && <p>{errors.password.message}</p>}<button onClick={handleSubmit(onSubmit)}>提交</button></div>);
}
  • useForm():初始化表单管理。
  • register('fieldName'):注册表单字段并定义验证规则。
  • handleSubmit(onSubmit):处理表单提交。
  • errors:访问验证错误信息。

3.3 结合 Yup 实现复杂验证

Yup 是一个强大的模式验证库,通过声明式规则定义复杂的验证逻辑,与 React Hook Form 无缝集成。

安装
npm install yup @hookform/resolvers
示例:带验证的注册表单
import { useForm } from 'react-hook-form';
import { yupResolver } from '@hookform/resolvers/yup';
import * as yup from 'yup';const schema = yup.object().shape({username: yup.string().required('用户名必填').min(3, '至少 3 个字符'),email: yup.string().email('无效的邮箱').required('邮箱必填'),password: yup.string().required('密码必填').min(6, '至少 6 个字符'),
});function ValidatedForm() {const { register, handleSubmit, formState: { errors } } = useForm({resolver: yupResolver(schema),});const onSubmit = (data) => {console.log('表单数据:', data);};return (<div><label>用户名</label><input {...register('username')} />{errors.username && <p>{errors.username.message}</p>}<label>邮箱</label><input {...register('email')} />{errors.email && <p>{errors.email.message}</p>}<label>密码</label><input type="password" {...register('password')} />{errors.password && <p>{errors.password.message}</p>}<button onClick={handleSubmit(onSubmit)}>提交</button></div>);
}
  • yup.object().shape():定义验证规则。
  • yupResolver(schema):将 Yup 集成到 React Hook Form。
  • 错误信息通过 errors.fieldName.message 显示。
高级验证

Yup 支持条件验证、自定义规则等。例如,验证密码和确认密码是否一致:

const schema = yup.object().shape({password: yup.string().required('密码必填').min(6, '至少 6 个字符'),confirmPassword: yup.string().oneOf([yup.ref('password'), null], '密码不一致').required('确认密码必填'),
});

4. 动态表单

动态表单允许用户根据需要添加或删除表单字段,例如输入多个地址或教育经历。这种功能在复杂表单中非常常见。

4.1 使用 useFieldArray

React Hook Form 提供了 useFieldArray Hook,专门用于管理动态字段数组。

示例:动态地址字段
import { useForm, useFieldArray } from 'react-hook-form';function DynamicForm() {const { control, register, handleSubmit } = useForm({defaultValues: { addresses: [{ address: '' }] },});const { fields, append, remove } = useFieldArray({control,name: 'addresses',});const onSubmit = (data) => {console.log('表单数据:', data);};return (<div><h2>地址</h2>{fields.map((field, index) => (<div key={field.id}><input{...register(`addresses.${index}.address`, { required: '地址必填' })}placeholder="地址"/><button type="button" onClick={() => remove(index)}>删除</button></div>))}<button type="button" onClick={() => append({ address: '' })}>添加地址</button><button onClick={handleSubmit(onSubmit)}>提交</button></div>);
}
  • useFieldArray:管理动态字段。
  • append:添加新字段。
  • remove:删除指定字段。
  • register(addresses.${index}.address):注册动态字段。
验证动态字段

结合 Yup 验证动态字段:

const schema = yup.object().shape({addresses: yup.array().of(yup.object().shape({address: yup.string().required('地址必填'),})),
});

5. 错误处理与用户反馈

良好的错误处理和用户反馈是提升表单体验的关键。React Hook Form 提供了灵活的错误管理和反馈机制。

5.1 显示错误信息

通过 formState.errors 访问错误:

<input {...register('username', { required: '用户名必填' })} />
{errors.username && <p>{errors.username.message}</p>}

5.2 自定义反馈

  • 实时反馈:通过 mode: 'onChange' 启用实时验证。
const { register, formState: { errors } } = useForm({ mode: 'onChange' });
  • 表单级错误:提交时显示所有错误。
  • 焦点管理:使用 setFocus 将焦点移到错误字段。

5.3 用户友好建议

  • 使用颜色(如红色)高亮错误。
  • 提供清晰的错误消息。
  • 支持无障碍(ARIA 属性)。

6. 案例:用户注册表单

以下是一个完整的用户注册表单,支持验证和动态地址字段。

import { useForm, useFieldArray } from 'react-hook-form';
import { yupResolver } from '@hookform/resolvers/yup';
import * as yup from 'yup';const schema = yup.object().shape({username: yup.string().required('用户名必填').min(3, '至少 3 个字符'),email: yup.string().email('无效的邮箱').required('邮箱必填'),password: yup.string().required('密码必填').min(6, '至少 6 个字符'),addresses: yup.array().of(yup.object().shape({address: yup.string().required('地址必填'),})),
});function RegisterForm() {const { register, control, handleSubmit, formState: { errors } } = useForm({resolver: yupResolver(schema),defaultValues: { addresses: [{ address: '' }] },});const { fields, append, remove } = useFieldArray({control,name: 'addresses',});const onSubmit = (data) => {console.log('注册成功:', data);};return (<div><label>用户名</label><input {...register('username')} />{errors.username && <p>{errors.username.message}</p>}<label>邮箱</label><input {...register('email')} />{errors.email && <p>{errors.email.message}</p>}<label>密码</label><input type="password" {...register('password')} />{errors.password && <p>{errors.password.message}</p>}<h3>地址</h3>{fields.map((field, index) => (<div key={field.id}><input{...register(`addresses.${index}.address`)}placeholder="地址"/><button type="button" onClick={() => remove(index)}>删除</button>{errors.addresses?.[index]?.address && (<p>{errors.addresses[index].address.message}</p>)}</div>))}<button type="button" onClick={() => append({ address: '' })}>添加地址</button><button onClick={handleSubmit(onSubmit)}>注册</button></div>);
}
关键点
  • 使用 Yup 定义多层验证。
  • useFieldArray 管理动态字段。
  • 实时错误反馈提升体验。

7. 练习:多步骤表单

实现一个多步骤结账流程表单,包含以下步骤:

  1. 填写地址
  2. 选择支付方式
  3. 确认订单

实现思路

  • 使用 state 管理当前步骤。
  • 每个步骤为独立组件。
  • React Hook Form 管理整个表单状态。

参考代码

import { useForm, FormProvider } from 'react-hook-form';
import { useState } from 'react';function Step1({ nextStep }) {const { register, formState: { errors } } = useForm();return (<div><h3>步骤 1: 填写地址</h3><input {...register('address', { required: '地址必填' })} />{errors.address && <p>{errors.address.message}</p>}<button onClick={nextStep}>下一步</button></div>);
}function Step2({ nextStep, prevStep }) {const { register, formState: { errors } } = useForm();return (<div><h3>步骤 2: 选择支付方式</h3><select {...register('payment', { required: '请选择支付方式' })}><option value="">请选择</option><option value="credit">信用卡</option><option value="paypal">PayPal</option></select>{errors.payment && <p>{errors.payment.message}</p>}<button onClick={prevStep}>上一步</button><button onClick={nextStep}>下一步</button></div>);
}function Step3({ prevStep, onSubmit }) {return (<div><h3>步骤 3: 确认订单</h3><p>请确认您的订单信息。</p><button onClick={prevStep}>上一步</button><button onClick={onSubmit}>提交订单</button></div>);
}function MultiStepForm() {const methods = useForm();const [step, setStep] = useState(1);const nextStep = () => setStep(step + 1);const prevStep = () => setStep(step - 1);const onSubmit = (data) => {console.log('订单数据:', data);alert('订单已提交');};return (<FormProvider {...methods}>{step === 1 && <Step1 nextStep={nextStep} />}{step === 2 && <Step2 nextStep={nextStep} prevStep={prevStep} />}{step === 3 && <Step3 prevStep={prevStep} onSubmit={methods.handleSubmit(onSubmit)} />}</FormProvider>);
}

8. React Hook Form 的性能优势

React Hook Form 的核心优势在于其基于非受控组件的实现方式,避免了传统受控组件的频繁渲染问题。

8.1 为什么性能更好?

  • 减少渲染:仅在必要时更新 DOM。
  • 无状态管理开销:无需为每个字段维护 state。
  • 高效事件处理:通过 ref 管理表单数据。

8.2 与其他库的对比

与 Formik 等库相比,React Hook Form 在大型表单中渲染次数显著减少。例如:

  • Formik:每个输入变化触发整个表单重新渲染。
  • React Hook Form:仅更新相关字段。

8.3 数据支持

根据官方性能测试,React Hook Form 在复杂表单中的渲染次数可减少 50%-70%(具体数据见官方文档)。


9. 总结与进阶建议

本文系统介绍了 React 中的表单处理与验证技术,从受控与非受控组件的基础知识,到 React Hook Form 和 Yup 的高级应用,再到动态表单和多步骤表单的实践案例,内容全面且实用。

总结

  • 受控组件适合复杂交互,非受控组件适合简单场景。
  • React Hook Form 是高效表单管理的首选工具。
  • 结合 Yup 可以轻松实现复杂验证。
  • 动态表单和多步骤表单提升了用户体验。

进阶建议

  • 探索 React Hook Form 的 useWatchuseFormContext
  • 学习 Yup 的条件验证和自定义规则。
  • 在实际项目中实践多步骤表单设计。

掌握本文内容后,你将能够自信地处理任何表单需求,并构建高效、用户友好的 React 应用。


附代码:

<!DOCTYPE html>
<html lang="zh-CN">
<head><meta charset="UTF-8"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>React 表单处理与验证深度指南</title><script src="https://cdn.jsdelivr.net/npm/react@18/umd/react.development.js"></script><script src="https://cdn.jsdelivr.net/npm/react-dom@18/umd/react-dom.development.js"></script><script src="https://cdn.jsdelivr.net/npm/@babel/standalone/babel.min.js"></script><script src="https://cdn.jsdelivr.net/npm/react-hook-form@7/dist/index.umd.min.js"></script><script src="https://cdn.jsdelivr.net/npm/yup@1/dist/yup.umd.min.js"></script><script src="https://cdn.jsdelivr.net/npm/@hookform/resolvers@3/dist/umd/index.min.js"></script><link href="https://cdn.jsdelivr.net/npm/tailwindcss@2/dist/tailwind.min.css" rel="stylesheet">
</head>
<body class="bg-gray-100 p-6"><div id="root" class="max-w-4xl mx-auto bg-white p-6 rounded-lg shadow-md"></div><script type="text/babel">const { useState, useForm, useFieldArray, FormProvider } = ReactHookForm;const { yupResolver } = HookFormResolvers;const yup = window.yup;// 受控组件示例function ControlledInput() {const [value, setValue] = useState('');const handleChange = (e) => setValue(e.target.value);return (<div className="mb-4"><label className="block text-gray-700">受控输入</label><inputtype="text"value={value}onChange={handleChange}placeholder="请输入内容"className="mt-1 p-2 border rounded w-full"/><p className="mt-2 text-gray-600">当前值: {value}</p></div>);}// 非受控组件示例function UncontrolledInput() {const inputRef = React.useRef(null);const handleSubmit = () => alert(`输入值: ${inputRef.current.value}`);return (<div className="mb-4"><label className="block text-gray-700">非受控输入</label><inputtype="text"ref={inputRef}placeholder="请输入内容"className="mt-1 p-2 border rounded w-full"/><buttononClick={handleSubmit}className="mt-2 bg-blue-500 text-white p-2 rounded">提交</button></div>);}// 带验证的简单表单function SimpleForm() {const { register, handleSubmit, formState: { errors } } = useForm();const onSubmit = (data) => console.log('表单数据:', data);return (<div className="mb-4"><label className="block text-gray-700">用户名</label><input{...register('username', { required: '用户名必填' })}className="mt-1 p-2 border rounded w-full"/>{errors.username && <p className="text-red-500">{errors.username.message}</p>}<label className="block text-gray-700 mt-2">密码</label><inputtype="password"{...register('password', { required: '密码必填' })}className="mt-1 p-2 border rounded w-full"/>{errors.password && <p className="text-red-500">{errors.password.message}</p>}<buttononClick={handleSubmit(onSubmit)}className="mt-2 bg-blue-500 text-white p-2 rounded">提交</button></div>);}// 用户注册表单const schema = yup.object().shape({username: yup.string().required('用户名必填').min(3, '至少 3 个字符'),email: yup.string().email('无效的邮箱').required('邮箱必填'),password: yup.string().required('密码必填').min(6, '至少 6 个字符'),addresses: yup.array().of(yup.object().shape({address: yup.string().required('地址必填'),})),});function RegisterForm() {const { register, control, handleSubmit, formState: { errors } } = useForm({resolver: yupResolver(schema),defaultValues: { addresses: [{ address: '' }] },});const { fields, append, remove } = useFieldArray({control,name: 'addresses',});const onSubmit = (data) => console.log('注册成功:', data);return (<div className="mb-4"><label className="block text-gray-700">用户名</label><input{...register('username')}className="mt-1 p-2 border rounded w-full"/>{errors.username && <p className="text-red-500">{errors.username.message}</p>}<label className="block text-gray-700 mt-2">邮箱</label><input{...register('email')}className="mt-1 p-2 border rounded w-full"/>{errors.email && <p className="text-red-500">{errors.email.message}</p>}<label className="block text-gray-700 mt-2">密码</label><inputtype="password"{...register('password')}className="mt-1 p-2 border rounded w-full"/>{errors.password && <p className="text-red-500">{errors.password.message}</p>}<h3 className="mt-4 text-lg font-semibold">地址</h3>{fields.map((field, index) => (<div key={field.id} className="flex items-center mt-2"><input{...register(`addresses.${index}.address`)}placeholder="地址"className="p-2 border rounded w-full"/><buttontype="button"onClick={() => remove(index)}className="ml-2 bg-red-500 text-white p-2 rounded">删除</button>{errors.addresses?.[index]?.address && (<p className="text-red-500">{errors.addresses[index].address.message}</p>)}</div>))}<buttontype="button"onClick={() => append({ address: '' })}className="mt-2 bg-green-500 text-white p-2 rounded">添加地址</button><buttononClick={handleSubmit(onSubmit)}className="mt-4 bg-blue-500 text-white p-2 rounded">注册</button></div>);}// 多步骤表单function Step1({ nextStep }) {const { register, formState: { errors } } = useForm();return (<div><h3 className="text-lg font-semibold">步骤 1: 填写地址</h3><input{...register('address', { required: '地址必填' })}className="mt-1 p-2 border rounded w-full"/>{errors.address && <p className="text-red-500">{errors.address.message}</p>}<buttononClick={nextStep}className="mt-2 bg-blue-500 text-white p-2 rounded">下一步</button></div>);}function Step2({ nextStep, prevStep }) {const { register, formState: { errors } } = useForm();return (<div><h3 className="text-lg font-semibold">步骤 2: 选择支付方式</h3><select{...register('payment', { required: '请选择支付方式' })}className="mt-1 p-2 border rounded w-full"><option value="">请选择</option><option value="credit">信用卡</option><option value="paypal">PayPal</option></select>{errors.payment && <p className="text-red-500">{errors.payment.message}</p>}<buttononClick={prevStep}className="mt-2 bg-gray-500 text-white p-2 rounded mr-2">上一步</button><buttononClick={nextStep}className="mt-2 bg-blue-500 text-white p-2 rounded">下一步</button></div>);}function Step3({ prevStep, onSubmit }) {return (<div><h3 className="text-lg font-semibold">步骤 3: 确认订单</h3><p>请确认您的订单信息。</p><buttononClick={prevStep}className="mt-2 bg-gray-500 text-white p-2 rounded mr-2">上一步</button><buttononClick={onSubmit}className="mt-2 bg-blue-500 text-white p-2 rounded">提交订单</button></div>);}function MultiStepForm() {const methods = useForm();const [step, setStep] = useState(1);const nextStep = () => setStep(step + 1);const prevStep = () => setStep(step - 1);const onSubmit = (data) => {console.log('订单数据:', data);alert('订单已提交');};return (<FormProvider {...methods}>{step === 1 && <Step1 nextStep={nextStep} />}{step === 2 && <Step2 nextStep={nextStep} prevStep={prevStep} />}{step === 3 && <Step3 prevStep={prevStep} onSubmit={methods.handleSubmit(onSubmit)} />}</FormProvider>);}// 主应用function App() {return (<div><h1 className="text-3xl font-bold mb-6">React 表单处理与验证深度指南</h1><h2 className="text-2xl font-semibold mb-4">受控组件示例</h2><ControlledInput /><h2 className="text-2xl font-semibold mb-4">非受控组件示例</h2><UncontrolledInput /><h2 className="text-2xl font-semibold mb-4">简单表单示例</h2><SimpleForm /><h2 className="text-2xl font-semibold mb-4">用户注册表单</h2><RegisterForm /><h2 className="text-2xl font-semibold mb-4">多步骤表单</h2><MultiStepForm /></div>);}ReactDOM.render(<App />, document.getElementById('root'));</script>
</body>
</html>

这篇指南通过理论与实践相结合,帮助你从基础到高级掌握 React 表单处理技术。希望对你有所帮助!如果有任何问题或改进建议,请随时告诉我。

相关文章:

React从基础入门到高级实战:React 核心技术 - 表单处理与验证深度指南

React 表单处理与验证深度指南 在现代 Web 应用中&#xff0c;表单是用户与应用交互的核心方式之一。无论是注册、登录、结账还是数据提交&#xff0c;表单都扮演着至关重要的角色。React 作为一款流行的前端框架&#xff0c;提供了多种处理表单的工具和方法&#xff0c;帮助开…...

【C++】stack,queue和priority_queue(优先级队列)

文章目录 前言一、栈&#xff08;stack&#xff09;和队列&#xff08;queue&#xff09;的相关接口1.栈的相关接口2.队列的相关接口 二、栈&#xff08;stack&#xff09;和队列&#xff08;queue&#xff09;的模拟实现1.stack的模拟实现2.queue的模拟实现 三、priority_queu…...

ubuntu中上传项目至GitHub仓库教程

一、到github官网注册用户 1.注册用户 地址&#xff1a;https://github.com/ 2.安装Git 打开终端&#xff0c;输入指令git,检查是否已安装Git 如果没有安装就输入指令 sudo apt-get install git 二、上传项目到github 1.创建项目仓库 进入github主页&#xff0c;点击号…...

[Java实战]Spring Boot整合达梦数据库连接池配置(三十四)

[Java实战]Spring Boot整合达梦数据库连接池配置&#xff08;三十四&#xff09; 一、HikariCP连接池配置&#xff08;默认&#xff09; 1. 基础配置&#xff08;application.yml&#xff09; spring:datasource:driver-class-name: dm.jdbc.driver.DmDriverurl: jdbc:dm://…...

windows 下用yolov5 训练模型 给到opencv 使用

windows 使用yolov5训练模型&#xff0c;之后opencv加载模型进行推理。 一&#xff0c;搭建环境 安装 Anaconda 二&#xff0c;创建虚拟环境并安装yolov5 conda create -n yolov5 python3.9 -y conda activate yolov5 git clone https://github.com/ultralytics/yolov5 cd …...

Spark集群架构解析:核心组件与Standalone、YARN模式深度对比(AM,Container,Driver,Executor)

一、核心组件定义与关系拆解 1. ApplicationMaster&#xff08;AM&#xff09; 定义&#xff1a;YARN 框架中的应用管理器&#xff0c;每个应用程序&#xff08;如 Spark 作业&#xff09;对应一个 AM。职责&#xff1a; 向 YARN 的 ResourceManager 申请资源&#xff08;Con…...

Linux Kernel调试:强大的printk(二)

前言 如果你对printk的基本用法还不熟悉&#xff0c;请先阅读&#xff1a; Linux Kernel调试&#xff1a;强大的printk&#xff08;一&#xff09; 上一篇Linux Kernel调试&#xff1a;强大的printk&#xff08;一&#xff09;我们介绍了printk的基础知识和基本用法&#xf…...

Kafka Kraft模式集群 + ssl

文章目录 启用集群资源规划准备证书创建相关文件夹配置文件启动各Kafka节点 故障转移测试spring boot集成 启用集群 配置集群时关键就是提前梳理好需要的网络资源&#xff0c;完成对应server.properties文件的配置。在执行前先把这些梳理好&#xff0c;可以方便后面的配置&…...

[crxjs]自己创建一个浏览器插件

参考官方 https://crxjs.dev/vite-plugin/getting-started/vue/create-project 按照流程操作会失败的原因 是因为跨域的问题, 在此处添加 server: {host: "localhost",port: 5173,cors: true,headers: {"Access-Control-Allow-Origin": "*",}…...

类的设计模式——单例、工厂以及建造者模式

1.单例模式 1.1 饿汉模式 单例模式&#xff1a;一个类只能创建一个对象&#xff0c;这个设计模式可以保证系统中该类只有一个实例&#xff0c;并提供一个访问它的全局访问点&#xff0c;该实例被所有程序模块共享。 饿汉模式指在程序初始化时就创建一个唯一的实例对象。适用…...

STM32之看门狗(IWDG)

一、看门狗外设的原理与应用 背景说明 随着单片机的发展&#xff0c;单片机在家用电器、工业自动化、生产过程控制、智能仪器仪表等领域的应用越来越广泛。然而处于同一电力系统中的各种电气设备通过电或磁的联系彼此紧密相连&#xff0c;相互影响&#xff0c;由于运行方式的…...

PyTorch实现MLP信用评分模型全流程

知识点回顾&#xff1a; 过拟合的判断&#xff1a;测试集和训练集同步打印指标模型的保存和加载 仅保存权重保存权重和模型保存全部信息checkpoint&#xff0c;还包含训练状态 早停策略 浙大疏锦行 import torch import torch.nn as nn import torch.optim as optim from skle…...

语音识别——文本转语音

python自带的pytts说话人的声音比较机械&#xff0c;edge-tts提供了更自然的语音合成效果&#xff0c;支持多种语音选择。 项目地址&#xff1a;GitHub - rany2/edge-tts: Use Microsoft Edges online text-to-speech service from Python WITHOUT needing Microsoft Edge or …...

跟着华为去变革 ——读《常变与长青》有感

《常变与长青》&#xff0c;是华为郭平总2024年上市的著作。走进这本书&#xff0c;我们能够清晰看到华为30多年的成长过程和伴随期间的变革历程&#xff1a;从一家设备代理商开始&#xff0c;起步蹒跚&#xff0c;砥砺前行&#xff0c;在闯过一个又一个磨难之后&#xff0c;成…...

图像分割技术的实现与比较分析

引言 图像分割是计算机视觉领域中的一项基础技术&#xff0c;其目标是将数字图像划分为多个图像子区域&#xff08;像素的集合&#xff09;&#xff0c;以简化图像表示&#xff0c;便于后续分析和理解。在医学影像、遥感图像分析、自动驾驶、工业检测等众多领域&#xff0c;图…...

node.js配置变量

一、下载安装包 1、官网下载 大家可以在官网下载&#xff0c;适合自己电脑以及项目的需要的版本。 二、node.js安装 1、安装 双击下载的安装包文件&#xff0c;通常为 .exe 或 .msi 格式&#xff08;Windows&#xff09;或 .dmg 格式&#xff08;Mac&#xff09;。系统会…...

Ubuntu+Docker+内网穿透:保姆级教程实现安卓开发环境远程部署

文章目录 前言1. 虚拟化环境检查2. Android 模拟器部署3. Ubuntu安装Cpolar4. 配置公网地址5. 远程访问小结 6. 固定Cpolar公网地址7. 固定地址访问 前言 本文将详细介绍一种创新性的云开发架构&#xff1a;基于Ubuntu系统构建Android仿真容器环境&#xff0c;并集成安全隧道技…...

为什么需要清除浮动?清除浮动的方式有哪些?

导语: 在前端面试中,“清除浮动”几乎是每位面试官都会问到的基础题。虽然浮动已经不如 Flex 和 Grid 那么常用了,但它在许多老项目中仍然占有一席之地。理解浮动的机制、掌握清除浮动的方式,是面试中体现你前端基础扎实度的关键点。 一、面试主题概述 浮动(float)最初是…...

计算机网络学习20250526

SMTP——简单邮件传输协议 TCP 端口号&#xff1a;25 Alice给Bob发送邮件过程&#xff1a; Alice使用邮件代理程序写邮件给Bob用户代理把报文发给邮件服务器&#xff0c;放入报文队列中邮件服务器上SMTP客户端建立与Bob服务器上SMTP服务器的TCP连接经过初始的握手后&#xff…...

ArkUI:鸿蒙应用响应式与组件化开发指南(一)

文章目录 引言1.ArkUI核心能力概览1.1状态驱动视图1.2组件化&#xff1a;构建可复用UI 2.状态管理&#xff1a;从单一组件到全局共享2.1 状态装饰器2.2 状态传递模式对比 引言 鸿蒙生态正催生应用开发的新范式。作为面向全场景的分布式操作系统&#xff0c;鸿蒙的北向应用开发…...

YOLOv11改进 | Neck篇 | 双向特征金字塔网络BiFPN助力YOLOv11有效涨点

YOLOv11改进 | Neck篇 | 双向特征金字塔网络BiFPN助力YOLOv11有效涨点 引言 目标检测领域的最新进展表明,特征金字塔网络(FPN)的设计对模型性能具有决定性影响。本文详细介绍如何将**双向特征金字塔网络(BiFPN)**集成到YOLOv11的Neck部分,通过改进的多尺度特征融合机制…...

C/C++的OpenCV 进行轮廓提取

使用 C/C的OpenCV 进行轮廓提取 轮廓可以简单地描述为连接所有具有相同颜色或强度的连续点&#xff08;沿着边界&#xff09;的曲线。轮廓是形状分析以及对象检测和识别的有用工具。OpenCV 提供了非常方便的函数来查找和绘制轮廓。 本文将指导您完成使用 C 和 OpenCV 库从图像…...

计算机网络总结(物理层,链路层)

目录 第一章 概述 1.基本概念 2.- C/S模式&#xff0c;B/S模式&#xff0c;P2P模式 3.- LAN,WAN,MAN,PAN的划分 4.电路交换与分组交换&#xff0c;数据报交换和虚电路交换 第二章 物理层 1.信号编码&#xff1a;不归零编码&#xff0c;曼切斯特编码 2.几种复用技术的特…...

TIGER - 一个轻量高效的语音分离模型,支持人声伴奏分离、音频说话人分离等 支持50系显卡 本地一键整合包下载

TIGER 是一种轻量级语音分离模型&#xff0c;通过频段分割、多尺度及全频帧建模有效提取关键声学特征。该项目由来自清华大学主导研发&#xff0c;通过频率带分割、多尺度以及全频率帧建模的方式&#xff0c;有效地提取关键声学特征&#xff0c;从而实现高效的语音分离。 TIGER…...

yolov8,c++案例汇总

文章目录 引言多目标追踪案例人体姿态估计算法手势姿态估计算法目标分割算法 引言 以下案例,基于c,ncnn,yolov8既可以在windows10/11上部署, 也可以在安卓端部署, 也可以在嵌入式端部署, 服务器端可支持部署封装为DLL,支持c/c#/java端调用 多目标追踪案例 基于yolov8, ncnn,…...

无人机降落伞设计要点难点及原理!

一、设计要点 1. 伞体结构与折叠方式 伞体需采用轻量化且高强度的材料&#xff08;如抗撕裂尼龙或芳纶纤维&#xff09;&#xff0c;并通过多重折叠设计&#xff08;如三重折叠缝合&#xff09;减少展开时的阻力&#xff0c;同时增强局部承力区域的强度。 伞衣的几何参数&am…...

20250526给荣品PRO-RK3566的Android13单独编译boot.img

./build.sh init ./build.sh -K ./build.sh kernel 20250526给荣品PRO-RK3566的Android13单独编译boot.img 2025/5/26 15:25 缘起&#xff1a;需要给荣品PRO-RK3566的Android13单独编译内核&#xff0c;但是不想编译整个系统。于是&#xff1a; 如果特调试某些特别的改动/文件…...

vue3项目动态路由的相关配置踩坑记录

1.路由文件中引入store的报错解决 import { useUserStore } from /stores/user // 错误&#xff1a;此时 Pinia 未初始化const store useUserStore() // 报错 解决方案&#xff1a; import pinia,{ useUserStore } from /stores/user 或者在路由前置守卫中调用useUserSto…...

git子模块--命令--列表版

Git子模块指令查询手册 一、基本操作指令 添加子模块 git submodule add <仓库地址> [路径] 添加子模块并生成.gitmodules。 克隆含子模块项目 git clone --recursive <主仓库地址> 克隆主仓库及所有子模块。 初始化子模块 git submodule init 将.gitmodules…...

C++(4)

四、模板与容器 1. 模板 1.1 函数模板 #include <iostream> using namespace std;// 函数模板声明 template<typename T> // 也可使用 class T add(T a, T b) {return a b; }int main() {string a "hello";string b "world";cout <&…...