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 接口,获取所有上传过的图片列表
  • 前端用列表展示"历史图片",按时间倒序
相关推荐
码路高手3 小时前
Hermes Agent 整体了解
后端·架构
日月云棠4 小时前
JAVA数据结构与算法 - 基础:链表
java·后端
日月云棠4 小时前
JAVA数据结构与算法 - 基础:栈 (Stack) 深度解析
java·后端
xiguolangzi4 小时前
java使用Map映射遍历方法
java·后端
日月云棠4 小时前
JAVA数据结构与算法 - 基础:队列 (Queue) 全方位解析
java·后端
lolo大魔王4 小时前
MongoDB 索引机制详解:单字段索引、复合索引、唯一索引与性能优化
数据库·mongodb
newnazi4 小时前
RedHat10 安装MS SQL Server2025
linux·服务器·数据库
KaMeidebaby4 小时前
卡梅德生物技术快报|单 B 细胞抗体制备:流程优化、表达系统适配与性能数据
前端·数据库·其他·百度·新浪微博
IT策士4 小时前
Django 从 0 到 1 打造完整电商平台:为什么用 Django 做电商?
后端·python·django