基于 Vue 和 Node.js 实现图片上传功能:从前端到后端的完整实践

news2025/5/21 4:15:09

在开发一个社交分享平台时,图片上传功能是核心需求之一。本文将基于一个旅拍社交分享系统,详细解析其图片上传功能的实现原理和技术细节,包括前端处理、后端接收和数据库存储等环节。

1. 前端图片上传实现

在这个项目中,图片上传功能主要在resend.vue组件中实现,用户可以通过表单选择图片并提交到服务器。

核心代码片段:

<template>
  <a-form :model="formData" ref="forRef">
    <!-- 其他表单字段 -->
    <a-form-item label="上传图片" name="upload">
      <input type="file" @change="handleAvatarChange" class="file-input" />
    </a-form-item>
    <a-button @click="handelChange">提交</a-button>
  </a-form>
</template>

<script setup>
import { ref } from 'vue'

const formData = ref({})

// 处理图片选择
const handleAvatarChange = (event) => {
  const file = event.target.files[0]
  if (file) {
    formData.value.avatarFile = file  // 保存选中的文件
  }
}

// 提交表单
const handelChange = async () => {
  await forRef.value.validateFields()
  
  // 创建FormData对象
  const formPayload = new FormData()
  
  // 添加文本字段
  formPayload.append('title', formData.value.title)
  formPayload.append('content', formData.value.content)
  // ...其他字段
  
  // 添加文件字段(关键部分)
  if (formData.value.avatarFile) {
    formPayload.append('avatar', formData.value.avatarFile)
  }

  // 发送请求
  const res = await resendDataList(formPayload)
  if (res.code == 200) {
    message.success('发帖成功!')
    router.push('./home')
  } else {
    message.error('发帖失败!')
  }
}
</script>

代码解释:

  1. 文件选择处理

    • 通过<input type="file">触发文件选择对话框
    • handleAvatarChange方法将选中的文件保存到formData.avatarFile
  2. 表单数据处理

    • 使用FormData对象封装表单数据,支持二进制文件上传
    • append方法将文本字段和文件字段添加到FormData
  3. 请求发送

    • 使用axios(封装在resendDataList中)发送请求
    • 设置Content-Typemultipart/form-data,由浏览器自动处理边界

2. 后端图片接收与处理

后端使用 Node.js 和 Express 框架处理图片上传请求,并使用multer中间件处理文件。

核心代码片段(resend.js):

const express = require("express");
const router = express.Router();
const multer = require('multer');
const db = require("../db/index");

// 配置multer存储引擎
const storage = multer.diskStorage({
  destination: function (req, file, cb) {
    cb(null, "E:/test/img");  // 存储路径
  },
  filename: function (req, file, cb) {
    cb(null, Date.now() + '-' + file.originalname);  // 文件名格式
  }
});

const upload = multer({ storage: storage });

// 处理图片上传请求
router.post('/addResendList', upload.single('avatar'), (req, res) => {
  // 调试日志
  console.log('[请求体]', req.body);
  console.log('[上传文件]', req.file);

  // 参数校验
  const requiredFields = ['title', 'content', 'address', 'category', 'longitude', 'latitude'];
  const missingFields = requiredFields.filter(field => !req.body[field]);
  
  if (missingFields.length > 0) {
    return res.status(400).json({
      status: 1,
      message: `缺少必填字段: ${missingFields.join(', ')}`,
      code: 400
    });
  }

  // 处理图片路径
  const imagePath = req.file ? `/static/${req.file.filename}` : '';

  // 插入数据库
  const insertSql = `
    INSERT INTO list_table 
    (account, title, content, address, category, longitude, latitude, images, create_time, visibility) 
    VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
  `;
  
  db.query(insertSql, [
    req.body.account,
    req.body.title,
    req.body.content,
    req.body.address,
    req.body.category,
    req.body.longitude,
    req.body.latitude,
    imagePath,
    new Date(),
    'all'
  ], (err, result) => {
    if (err) {
      console.error('[数据库错误]', err);
      return res.status(500).json({
        status: 1,
        message: '数据库操作失败',
        error: err.message,
        code: 500
      });
    }

    res.status(200).json({
      status: 0,
      data: {
        table_id: result.insertId,
        images: imagePath
      },
      message: "提交成功",
      code: 200
    });
  });
});

代码解释:

  1. Multer 配置

    • diskStorage指定文件存储路径和文件名格式
    • upload.single('avatar')处理单个文件上传,字段名为avatar
  2. 文件处理

    • req.file包含上传的文件信息(文件名、大小、类型等)
    • 图片被保存到指定目录,并生成唯一文件名(时间戳 + 原文件名)
  3. 数据库操作

    • 将图片路径(如/static/1638532765432-avatar.jpg)存入数据库
    • 通过table_id关联图片与打卡点信息

注意⚠️:实际参数可自行调整修改。

3. 图片展示与访问

前端通过拼接完整 URL 访问服务器上的图片:

代码片段(UserPost.vue):

<template>
  <div class="post-card">
    <img :src="'http://localhost:8080' + post.images" alt="User Post" />
  </div>
</template>

后端配置(app.js):

// 设置静态资源目录
app.use('/static', express.static('E://test//img'));

解释:

  • 前端通过http://localhost:8080/static/文件名访问图片
  • 后端使用express.static中间件将/static路径映射到实际存储目录

4. 关键技术点总结

  1. 前端要点

    • 使用FormData处理混合表单数据(文本 + 文件)
    • 文件选择事件绑定与数据缓存
    • 图片预览功能(本文未展示,但可通过FileReader实现)
  2. 后端要点

    • multer中间件处理文件上传
    • 文件存储策略(路径、命名规则)
    • 图片路径与业务数据关联
  3. 安全与优化建议

    • 文件类型验证(限制只允许图片格式)
    • 文件大小限制(防止大文件攻击)
    • 图片压缩处理(减少存储和传输成本)
    • 图片访问权限控制(私有图片需身份验证)

总结

通过这个项目的实现,我们可以看到图片上传功能的完整流程:从前端表单选择,到后端接收存储,再到最终展示。关键在于前后端的协作:前端正确构建FormData对象,后端使用合适的中间件处理文件,并将文件路径与业务数据关联。

这种实现方式不仅适用于旅拍社交系统,也可应用于各种需要图片上传功能的 Web 应用中。通过合理的架构设计和安全措施,可以构建出高效、稳定的图片上传服务。

注意:内容仅供参考,实际项目中可按需要进行修改和排查。

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

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

相关文章

apisix透传客户端真实IP(real-ip插件)

文章目录 apisix透传客户端真实IP需求和背景apisix real-ip插件为什么需要 trusted_addresses&#xff1f;安全架构的最佳实践 示例场景apisix界面配置 apisix透传客户端真实IP 需求和背景 当 APISIX 前端有其他反向代理&#xff08;如 Nginx、HAProxy、云厂商的 LB&#xff…

Oracle 数据库的默认隔离级别

Oracle 数据库的默认隔离级别 默认隔离级别&#xff1a;READ COMMITTED Oracle 默认使用 读已提交(READ COMMITTED) 隔离级别&#xff0c;这是大多数OLTP(在线事务处理)系统的标准选择。 官方文档 https://docs.oracle.com/en/database/oracle/oracle-database/19/cncpt/da…

代码随想录算法训练营第六十四天| 图论9—卡码网47. 参加科学大会,94. 城市间货物运输 I

每日被新算法方式轰炸的一天&#xff0c;今天是dijkstra&#xff08;堆优化版&#xff09;以及Bellman_ford &#xff0c;尝试理解中&#xff0c;属于是只能照着代码大概说一下在干嘛。 47. 参加科学大会 https://kamacoder.com/problempage.php?pid1047 dijkstra&#xff08…

开启健康生活的多元养生之道

健康养生是一门值得终身学习的学问&#xff0c;在追求健康的道路上&#xff0c;除了常见方法&#xff0c;还有许多容易被忽视却同样重要的角度。掌握这些多元养生之道&#xff0c;能让我们的生活更健康、更有品质。​ 室内环境的健康不容忽视。定期清洁空调滤网&#xff0c;避…

【Vite】前端开发服务器的配置

定义一些开发服务器的行为和代理规则 服务器的基本配置 server: {host: true, // 监听所有网络地址port: 8081, // 使用8081端口open: true, // 启动时自动打开浏览器cors: true // 启用CORS跨域支持 } 代理配置 proxy: {/api: {target: https://…

Spring Security与Spring Boot集成原理

Spring Security依赖的是过滤器机制&#xff0c;首先是web容器例如tomcat作为独立的产品&#xff0c;本身有自己的一套过滤器机制用来处理请求&#xff0c;那么如何将tomcat接收到的请求转入到Spring Security的处理逻辑呢&#xff1f;spring充分采用了tomcat的拓展机制提供了t…

VScode各文件转化为PDF的方法

文章目录 代码.py文件.ipynb文本和代码夹杂的文件方法 1:使用 VS Code 插件(推荐)步骤 1:安装必要插件步骤 2:安装 `nbconvert`步骤 3:间接导出(HTML → PDF)本文遇见了系列错误:解决方案:问题原因步骤 1:降级 Jinja2 至兼容版本步骤 2:确保 nbconvert 版本兼容替代…

Vue3学习(组合式API——Watch侦听器、watchEffect()详解)

目录 一、Watch侦听器。 &#xff08;1&#xff09;侦听单个数据。 &#xff08;2&#xff09;侦听多个数据。&#xff08;数组写法&#xff1f;&#xff01;&#xff09; &#xff08;3&#xff09;immediate参数。(立即执行回调) &#xff08;3&#xff09;deep参数。(深层监…

【node.js】安装与配置

个人主页&#xff1a;Guiat 归属专栏&#xff1a;node.js 文章目录 1. Node.js简介1.1 Node.js的特点1.2 Node.js架构 2. Node.js安装2.1 下载和安装方法2.1.1 Windows安装2.1.2 macOS安装2.1.3 Linux安装 2.2 使用NVM安装和管理Node.js版本2.2.1 安装NVM2.2.2 使用NVM管理Node…

《AI大模型应知应会100篇》第62篇:TypeChat——类型安全的大模型编程框架

第62篇&#xff1a;TypeChat——类型安全的大模型编程框架 摘要 在构建 AI 应用时&#xff0c;一个常见的痛点是大语言模型&#xff08;LLM&#xff09;输出的不确定性与格式不一致问题。开发者往往需要手动解析、校验和处理模型返回的内容&#xff0c;这不仅增加了开发成本&a…

EdgeShard:通过协作边缘计算实现高效的 LLM 推理

(2024-05-23) EdgeShard: Efficient LLM Inference via Collaborative Edge Computing (EdgeShard:通过协作边缘计算实现高效的 LLM 推理) 作者: Mingjin Zhang; Jiannong Cao; Xiaoming Shen; Zeyang Cui;期刊: (发表日期: 2024-05-23)期刊分区:本地链接: Zhang 等 - 2024 …

火山 RTC 引擎9 ----集成 appkey

一、集成 appkey 1、网易RTC 初始化过程 1&#xff09;、添加头文件 实现互动直播 - 互动直播 2.0网易云信互动直播产品的基本功能包括音视频通话和连麦直播&#xff0c;当您成功初始化 SDK 之后&#xff0c;您可以简单体验本产品的基本业务流程&#xff0c;例如主播加入房间…

Adminer:一个基于Web的轻量级数据库管理工具

Adminer 是一个由单个 PHP 文件实现的免费数据库管理工具&#xff0c;支持 MySQL、MariaDB、PostgreSQL、CockroachDB、SQLite、SQL Server、Oracle、Elasticsearch、SimpleDB、MongoDB、Firebird、Clickhouse 等数据库。 Adminer 支持的主要功能如下&#xff1a; 连接数据库服…

RK3568下QT实现按钮切换tabWidget

运行效果: 在 Qt 应用程序开发过程中,TabWidget 是一种非常实用的 UI 组件,它能够以选项卡的形式展示多个页面内容,帮助我们有效组织和管理复杂的界面布局。而在实际使用时,常常会有通过按钮点击来切换 TabWidget 页面的需求,本文将通过一个完整的示例,详细介绍如何在 Q…

2025 OceanBase 开发者大会全议程指南

5 月 17 日&#xff0c;第三届 OceanBase 开发者大会将在广州举办。 我们邀请数据库领军者与AI实践先锋&#xff0c;与开发者一起探讨数据库与 AI 协同创新的技术趋势&#xff0c;面对面交流 OceanBase 在 TP、AP、KV 及 AI 能力上的最新进展&#xff0c;深度体验“打破技术栈…

day017-磁盘管理-实战

文章目录 1. 硬盘命名规则2. 添加硬盘2.1 查看硬盘名称 3. 硬盘分区3.1 分区命名规则&#xff1a;mbr分区表格式3.2 创建分区&#xff1a;fdisk3.2.1 fdisk -l&#xff1a;查看硬盘及分区信息3.2.2 fdisk /dev/sdc :为该硬盘分区3.2.3 创建扩展分区和逻辑分区3.2.4 保存设置并退…

【成品设计】STM32和UCOS-II的项目

项目1&#xff1a;《基于STM32和UCOS-II的水质监测系统》 Ps&#xff1a;分为带系统版本和不带系统版本&#xff0c;功能都一样。 功能说明&#xff1a; 1. 单片机主控&#xff1a;STM32F103C8T6单片机作为核心控制。 2. 酸碱度传感器&#xff1a;实时采集当前PH值。 3. 水质…

Ngrok 配置:实现 Uniapp 前后端项目内网穿透

文章目录 一、下载并安装 ngrok二、配置 ngrok Authtoken三、启动本地 uniapp 项目四、使用 ngrok 暴露本地服务五、通过公网 URL 访问项目六、后端API项目的穿透问题排查 (uni-app 后端 API 示例)交互流程图示 七、ngrok Web 界面 (本地监控)八、停止 ngrok总结 ngrok 是一款…

鸿蒙ArkUI体验:Hexo博客客户端开发心得

最近部门也在跟进鸿蒙平台的业务开发&#xff0c;自己主要是做 Android 开发&#xff0c;主要使用 Kotlin/Java 语言。&#xff0c;需要对新的开发平台和开发模式进行学习&#xff0c;在业余时间开了个项目练手&#xff0c;做了个基于 Hexo 博客内容开发的App。鸿蒙主要使用Ark…

鸿蒙NEXT开发动画案例10

1.创建空白项目 2.Page文件夹下面新建Spin.ets文件&#xff0c;代码如下&#xff1a; interface TranslateOffset {x?:numbery?:number } /*** SpinKit动画组件 - SpinTen* author: CSDN-鸿蒙布道师* since: 2025/05/16*/ ComponentV2 export struct SpinTen {Require Para…