Day29:图片上传 + 存数据库(Multer + MySQL)

这两天把"上传图片"和"存数据库"串起来了。

上传成功后,图片文件存到 uploads/ 文件夹,图片 URL 存到 images 表。

数据流从"前端 → 后端 → 文件 → 数据库 → 前端"完整闭环。


一、核心代码

后端 app.js(上传部分)

javascript

javascript 复制代码
const multer = require('multer');
const path = require('path');
const fs = require('fs');

// 配置文件存储
const storage = multer.diskStorage({
    destination: (req, file, cb) => {
        cb(null, 'uploads/');
    },
    filename: (req, file, cb) => {
        const uniqueName = Date.now() + '-' + file.originalname;
        cb(null, uniqueName);
    }
});

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

// 确保 uploads 文件夹存在
if (!fs.existsSync('uploads')) {
    fs.mkdirSync('uploads');
}

// 静态托管
app.use('/uploads', express.static(path.join(__dirname, 'uploads')));

// 上传接口
app.post('/upload', upload.single('image'), (req, res) => {
    if (!req.file) {
        return res.status(400).json({ error: '没有收到图片文件' });
    }

    const imageUrl = `/uploads/${req.file.filename}`;

    // 存数据库
    pool.query('INSERT INTO images (image_url) VALUES (?)', [imageUrl], (err, result) => {
        if (err) {
            console.error(err);
            return res.status(500).json({ error: '数据库保存失败' });
        }
        res.json({
            url: imageUrl,
            imageId: result.insertId
        });
    });
});

前端 public/index.html(关键部分)

javascript

javascript 复制代码
async function uploadImage() {
    const formData = new FormData();
    formData.append('image', file);

    const response = await fetch('/upload', {
        method: 'POST',
        body: formData
    });
    const data = await response.json();
    console.log('后端返回的图片数据:', data);
    
    if (response.ok) {
        document.getElementById('preview').innerHTML = `<img src="${data.url}" alt="上传的图片">`;
    }
}

二、数据库表结构

sql

sql 复制代码
CREATE TABLE IF NOT EXISTS images (
    id INT PRIMARY KEY AUTO_INCREMENT,
    image_url VARCHAR(255) NOT NULL,
    upload_time TIMESTAMP DEFAULT CURRENT_TIMESTAMP
);

三、数据流图(最终版)

流程简述:

  1. 前端选文件,FormData 包装,fetch POST /upload
  2. 后端 upload.single 中间件接收并自动保存文件到 uploads/
  3. 保存失败 → 返回 400/500 错误
  4. 保存成功 → pool.queryimage_url 插入 images
  5. 插入失败 → 返回 500 错误
  6. 插入成功 → 后端返回 { url: imageUrl, imageId: insertId }
  7. 前端收到数据,用 <img> 显示图片,控制台打印完整数据

四、踩过的坑

原因 解决
数据库关了还能存文件 只关了 Navicat,没关 MySQL 服务 net stop MySQL_3307 真关
后端返回的 JSON 看不到 没在后端加 console.log console.log('返回数据:', { url, imageId })
前端只打印 data.url,没看 data.imageId 只打印了部分属性 打印整个对象 console.log(data)
图片命名"乱码" Date.now() 是时间戳,中文文件名显示问题 正常现象,不影响功能;也可用纯时间戳命名

五、验证清单

  • 上传图片,uploads/ 文件夹能看到文件
  • images 表多一条记录,image_url/uploads/xxx.jpg
  • 前端页面显示刚上传的图片
  • 控制台打印后端返回的 urlimageId
  • 数据库关闭后上传会报错(ECONNREFUSED

六、下一步

  • GET /images 接口,获取所有上传过的图片列表
  • 前端用列表展示"历史图片",按时间倒序
相关推荐
用户34232323763176 分钟前
工业数据采集安全——当 OT 遇见 IT,谁对谁错?
后端
拾贰_C7 分钟前
【python | installation 】python 安装 | Windows | 命令使用
linux·数据库·ubuntu
贺今宵18 分钟前
Vue 3 + Capacitor 使用jeep-sqlite,web端使用本地sqlite数据库
前端·数据库·vue.js·sqlite·web
楼田莉子20 分钟前
C++20新特性:协程
开发语言·c++·后端·学习·c++20
列星随旋23 分钟前
MySQL面经整理
数据库·mysql
AllData公司负责人23 分钟前
大模型赋能AllData数据中台,系列升级|通过联合智谱大模型与Chat2DB开源项目,建设Text2SQL生产场景全新体验的数据源平台!
数据库·人工智能·text2sql·数据中台·数据源·chat2db·智谱大模型
元宝骑士27 分钟前
SpringBoot + Sa-Token 实现 CSRF 令牌校验(进阶篇)
后端·安全
Full Stack Developme30 分钟前
AspectJ 详解
java·后端
武子康38 分钟前
Java-20 深入浅出 MyBatis - 手写ORM框架1 从原始 JDBC 暴露的 6 大问题开始
java·后端
雪隐1 小时前
AI股票小助手06-Backtrader 量化回测
人工智能·后端