vue+Nodejs+Koa搭建前后端系统(九)-- 上传图片

news2025/1/19 18:34:45

web2.0的到来使网页世界正式进入了寒武纪,各式各样的多媒体资源屡见不鲜,上传资源变得刻不容缓!

前言

本文是在该系列的基础上,针对前后端代码的修改。

准备

HTTP上传图片时Content-Type值常见的有2种:application/jsonmultipart/form-data

前端准备

修改axios配置

/** /src/https.ts */
const http: MyAxiosInstance = axios.create({
    baseURL: process.env.NODE_ENV === 'production' ? httpHost : '/nodeApi/',
    /**New Code Start*/
    headers: {
        'Content-Type': 'application/json',//将axiox的Content-Type值默认为application/json
    },
    /**New Code End*/
    timeout: 60000,
});

/**其他代码省略**/

/**请求拦截器 */
http.interceptors.request.use(function (config) {
    const token = window.localStorage.getItem('token');
    const username = userStore.userName;
    if (token) {
        config.headers.authorization = token;
    }
    /**New Code Start*/
    //若请求Content-Type值为application/json,则处理请求数据
    if (config.headers['Content-Type'] === 'application/json' && username && config.data) {
    /**New Code End*/
        try {
            config.data = { username: username, ...config.data }
        } catch (e) {
            console.error(`请求拦截error:${e}`)
        }
    }
    return config;
}, function (error) {
    return Promise.reject(error);
});

上述代码修改了请求拦截,即:若请求类型为application/json时,则处理;否则(主要是针对multipart/form-data),不处理。

后端准备

1.安装koa-body

npm install koa-body

2.安装hexoid

npm install --save hexoid

3.安装dayjs

npm install dayjs

4.新建路由文件 /routes/files.js (用于接收上传图片请求并处理)

在这里插入图片描述

5.新建 /public/uploads/ 目录,用于存放上传的图片

在这里插入图片描述

数据库准备(非必须)

新建一张用于存储上传图片信息的表

CREATE TABLE user_uploads(  
    src VARCHAR(510) NOT NULL PRIMARY KEY COMMENT '文件地址',
    create_time TIMESTAMP NOT NULL COMMENT '上传时间',
    title VARCHAR(255) COMMENT '文件标题',
    description VARCHAR(1000)  COMMENT '文件描述',
    username VARCHAR(20) COMMENT '上传人',
    group_by VARCHAR(20) COMMENT '分组',
    mimetype VARCHAR(255) COMMENT 'MIME类型'
) COMMENT '文件上传维护表';

对之前代码修改

为了更加工整一些,作出如下修改:

1.修改create_user 表

#删除create_user 表的id列
ALTER TABLE create_user DROP COLUMN id;
#设置create_user 表的username列为主键
ALTER TABLE create_user ADD PRIMARY KEY (username);

修改后的表结构

在这里插入图片描述

2.后端代码新增配置页
为了统一后端的一些配置项,在后端项目根目录新建index.config.js文件
在这里插入图片描述
代码为:

/** /index.config.js */
module.exports = {
	//不验证用户名的接口
    no_verify: ["/login/loginIn", "/token/refresh", "/users/register", "/files/upload"],
    //静态文件夹路径
    static_basepath:'./public',
    //上传文件存储的目录
    static_uploadpath:'uploads'
}

然后修改引用的地方

/** /app.js */
/** 其他代码省略 */
const { verifyToken } = require("./middleware/jwt_copy1.js");
const { no_verify, static_basepath } = require("./index.config.js");

app.use(require("koa-static")(path.join(static_basepath)));//__dirname + "/public"
app.use(verifyToken({ no_verify: no_verify }));

前端处理上传图片的几种方式

使用 FormData 对象

可以使用原生form标签,实现FormData 对象上传图片

<!--index.vue-->
<form action="/nodeApi/files/upload" method="POST" enctype="multipart/form-data" target="_blank">
   <input type="file" name="file" multiple />
   <input type="hidden" name="username" value="xiaoyang" />
   <button type="submit">上传</button>
</form>

但该方法有默认行为,比如上传成功后会跳转页面,而且不够灵活。鉴于此,可用js配合,使其更加灵活:

<!--index.vue-->
<template>
	<input type="file" @change="getFiles" />
    <img v-for="i in previewImgs" :key="i" :src="i" />
    <button @click="upload">上传</button>
</template>
<script setup lang="ts">
import { reactive, ref } from "vue";
import http from "@/http";
import { ElMessage } from "element-plus";

//要上传图片的File对象数组,用于提交后台
const files = ref<any>([]);
//要上传图片的DataUrl数组,用于预览图片(这里也可以用BlobUrl)
const previewImgs = ref<(string | ArrayBuffer)[]>([]);

const getFiles = (e: any) => {
  files.value = files.value.concat([...e.target.files]);
  files.value.forEach((file: File, index: number) => {
    const reader = new FileReader();
    reader.readAsDataURL(file);
    reader.addEventListener("load", () => {
      previewImgs.value[index] = reader.result;
    });
  });
};
const upload = () => {
  //创建formData对象
  const formData = new FormData();
  //给formData对象添加files[]成员,多图片
  for (let i in files.value) {
    formData.append("files[]", files.value[i]);
  }
  formData.append("username", 'xyz');
  http.post("files/upload", formData, {
      headers: {
        "Content-Type": "multipart/form-data",
      },
  }).then((data: any) => {
      ElMessage({
        message: "上传成功",
        type: "success",
      });
    }).catch((err: any) => {
      ElMessage({
        message: err.message,
        type: "error",
      });
    });
};
</script>

使用 Base64 编码

这种方法将文件转换成 Base64 编码的字符串,然后通过普通的 JSON 格式发送给服务器。这种方式适用于较小的文件,因为 Base64 编码会增加数据大小。

<!--index.vue-->
<template>
	<input type="file" @change="getFiles" multiple />
    <button @click="upload">上传</button>
</template>
<script setup lang="ts">
import { reactive, ref } from "vue";
import http from "@/http";
import { ElMessage } from "element-plus";

//要上传图片的File对象数组,用于提交后台
const files = ref<any>([]);

const getFiles = (e: any) => {
  files.value = files.value.concat([...e.target.files]);
};
/**
*** 将文件转换为base64
*** files 文件数组,File[]
*** callback 解析完成的回调方法,该回调方法第一个参数是解析的base64数组,第二个参数是对应的文件名称数组
*/
const readFiles = (files: File[] = [], callback: (f: string[],n:string[]) => void) => {
  const len = files.length;
  let readNum = 0;
  const results = [];
  for (let i = 0; i < len; i++) {
    const reader = new FileReader();
    reader.onload = (e: any) => {
      readNum++;
      results.push(e.target.result);
      if (readNum === len) {
        callback(results,files.map((item: any) => item.name));
      }
    };
    reader.onerror = () => {
      readNum++;
      if (readNum === len) {
        callback(results,files.map((item: any) => item.name));
      }
    };
    reader.readAsDataURL(files[i]);
  }
};
const upload = () => {
 readFiles(files.value, (base64List: string[], filenameList: string[]) => {
    http.post("files/upload", {
      files: base64List,
      filename: filenameList,
    }).then((data: any) => {
      ElMessage({
         message: "上传成功",
         type: "success",
       });
    }).catch((err: any) => {
       ElMessage({
         message: err.message,
         type: "error",
       });
    });
  });
};
</script>

后端处理上传图片

1.编写路由文件

/** /routes/files.js */
const path = require("path");
const { uploadFiles, removeFiles } = require("../module/files")
const { koaBody } = require('koa-body');
const router = require('koa-router')();
const { jsonable } = require('../middleware/files')
const fs = require("fs");
const { static_basepath, static_uploadpath } = require("../index.config");
const dayjs = require('dayjs');
/** 上传图片 */
router.post('/upload', async (ctx, next) => {
    const content_type = ctx.request.header['content-type']
    if (content_type === 'application/json') {
        return jsonable()(ctx, next);
    } else if (content_type.includes('multipart/form-data')) {
        return koaBody({
            // 支持多文件格式
            multipart: true,
            formidable: {
                // 上传目录
                uploadDir: path.join(static_basepath, static_uploadpath),
                // 保留文件扩展名
                keepExtensions: true,
                //图片上传前的事件句柄
                onFileBegin(formName, file) {
                    ctx.request.formName = formName;
                    //上传图片的根目录
                    const dirpath = path.join(static_basepath, static_uploadpath, dirname);
                    const D = new Date();
                    //上传图片的二级目录(以‘年月日’格式为目录名)
                    const dirname = dayjs(D).format("YYYYMMDD");
                    //检查上传图片的目录是否存在,若不存在,则创建
                    if (!fs.existsSync(dirpath)) {
                        fs.mkdirSync(dirpath);
                    }
                    //改写文件存储的路径
                    file.filepath = path.join(static_basepath, static_uploadpath, dirname, file.newFilename)
                }
            }
        })(ctx, next)
    }
}, uploadFiles);
module.exports = router;

此处上传图片采用koa-router中间件router.post(path,middleware,callback)格式,middleware是中间件,该中间件代码逻辑为:

判断请求的content-type值,若为application/json,利用自定义的jsonable中间件处理上传,若为multipart/form-data,利用koa-body中间件处理上传。上传图片完成后执行uploadFiles回调函出,该函数用来处理数据库。

上传的图片如图:

在这里插入图片描述

2.编写自定义的jsonable中间件处理application/json的请求

新建 /middleware/files.js 文件,代码如下:

/** /middleware/files.js */
const path = require("path");
const fs = require("fs");
const { static_basepath, static_uploadpath } = require("../index.config");
const dayjs = require('dayjs');
const hexoid = require('hexoid');
/** 生成上传图片的目录(若没有该目录则创建) */
function fileDir() {
    const D = new Date();
    const dirname = dayjs(D).format("YYYYMMDD");
    const dirpath = path.join(static_basepath, static_uploadpath, dirname)
    if (!fs.existsSync(dirpath)) {
        fs.mkdirSync(dirpath);
    }
    return dirpath;
}
function jsonable() {
    return async (ctx, next) => {
    	//请求参数
        const params = ctx.request.body;
        //生成上传图片的目录
        const dirpath = fileDir();
        //获取文件数组
        const files = Array.isArray(params.files) ? params.files : [params.files];
        //获取文件名称数组
        const filenames = params.filename || [];
        //当前被处理的文件
        let file = null;
        //存储图片信息的数组
        const filepathArr = [];
        //循环处理图片(该图片为base64数据)
        while (file = files.shift()) {
        	//当前循环的文件名
        	const filename = filenames.shift() || '.png';
         	//生成上传图片的名称--为25为的随机名称
        	const uuid = hexoid(25)();
        	//图片base64数据主体
            const base64Data = file.split('base64,')
            //图片的MIMEType,即Content-Type
            const mimetype = base64Data[0].replace(/^data:(.+?);/, '$1');
            //图片扩展文件名
            const ext = '.' + filename.split(".").reverse()[0];
            //图片要存储的路径
            const filepath = path.join(dirpath, uuid + ext);
            //将图片上传到指定路径
            fs.writeFileSync(filepath, Buffer.from(base64Data[1], 'base64'));
            //获取图片信息对象
            const f = fs.statSync(filepath)
            filepathArr.push({ filepath: filepath, mimetype, lastModifiedDate: f.birthtime })
        }
        /** 改写请求参数以备后续使用(这里是模拟koa-body) Start*/
        ctx.request.formName = 'file[]';
        ctx.request.files = { [ctx.request.formName]: filepathArr };
        /** 改写请求参数以备后续使用(这里是模拟koa-body) End*/
        return next();
    }
}
module.exports = {
    jsonable,
}

该中间件用于处理Content-Type为application/json的上传图片请求,图片格式须为base64,可以是多图上传。

3.编写上传图片完成后的回调方法

新建 /module/files.js 文件,其代码如下:

/** /module/files.js */
const fs = require('fs')
const path = require("path")
const os = require("os");
const { URL } = require("url");
const { static_basepath } = require("../index.config");
const dayjs = require('dayjs');
/** 获取本机IPv4地址 */
function getIPAdress() {
    var interfaces = os.networkInterfaces();
    for (var devName in interfaces) {
        var iface = interfaces[devName];
        for (var i = 0; i < iface.length; i++) {
            var alias = iface[i];
            if (
                alias.family === "IPv4" &&
                alias.address !== "127.0.0.1" &&
                !alias.internal
            ) {
                return alias.address;
            }
        }
    }
}
/** 对于上传的图片写入数据库 */
async function uploadFiles(ctx, next) {
    const files = ctx.request.files;
    const formName = ctx.request.formName;
    const f = files[formName];
    const params = ctx.request.body
    const host = `http://${getIPAdress()}:${process.env.PORT}/`

	//用于存储sql表信息的数组
    const v = [];
    if (Array.isArray(f)) {//多文件
        f.forEach((item) => {
            const url = `${host}${path.relative(static_basepath, item.filepath)}`;
            const myURL = new URL(url);
            v.push({
                username: params?.username || "",//上传图片的用户
                create_time: dayjs(item.lastModifiedDate).format("YYYY-MM-DD HH:mm:ss"),//上传时间
                src: myURL.pathname,//上传路径
                mimetype: item.mimetype,//MimeType值 
                group_by: params?.group || "",//组
            })
        })
    } else {//单文件
        const url = `${host}${path.relative(static_basepath, f.filepath)}`;
        const myURL = new URL(url);
        v.push({
            username: params?.username || "",
            create_time: dayjs(f.lastModifiedDate).format("YYYY-MM-DD HH:mm:ss"),
            src: myURL.pathname,
            mimetype: f.mimetype,
            group_by: params?.group || "",
        })
    }
    //整合sql语句
    const keys = Object.keys(v[0]);
    const columns = keys.join(",");
    const values = v.map(item => `(${keys.map(key => typeof item[key] === "string" ? `'${item[key]}'` : item[key]).join(",")})`)
    const sql = `INSERT INTO user_uploads (${columns}) VALUES ${values.join(",")}`;
	//执行sql语句并响应请求
    try {
        const r = await ctx.db.query(sql);
        ctx.response.status = 200;
        ctx.body = { message: "上传成功", code: 0, data: { path: v.map(item => item.src).join(",") } };
    } catch (e) {
        ctx.response.status = 500;
        ctx.body = { message: e, code: 99 };
    }
}
module.exports = {
    uploadFiles
};

uploadFiles回调方法主要作用就是将上传的图片写入数据库,数据库表如下:

在这里插入图片描述
这里对于上传图片信息写入数据库是非必须的,这里我只是为了后续做图片管理打基础。

总结下我这里后端处理上传图片的逻辑:
1.在koa-router的中间件参数中处理图片上传,在其回调函数参数中处理上传图片的数据库
2.在处理图片上传中,根据请求数据的Content-Type给以不同的处理方案

后端处理删除图片

1.编写路由文件

/routes/files.js 文件中添加如下代码:

/** /routes/files.js */
/**  其他代码省略... */
router.post('/remove', removeFiles);
module.exports = router;

2.编写删除图片回调方法

/module/files.js 文件中添加如下代码:

/** /module/files.js */
/** 
*** 查询该用户是否有删除该图片的权限--目前规则是只有上传者才有删除权限 
*** paths 要删除的图片路径,String类型,多个时用逗号分隔
*** username 用户名
*/
function filePermissions(ctx, next, paths, username) {
    const pathArr = paths.split(',');
    //将传来的paths去掉域名
    const pathSql = pathArr.map(item => `'${item.replace(/^(http|https):\/\/[^\/]+/, "").replace(/^\s+|\s+$/gim, "")}'`).join(",")
    //sql:筛选paths的用户名和路径集合
    const sql = `SELECT src,username FROM user_uploads WHERE src in (${pathSql})`;
    
    return new Promise(async (resolve, reject) => {
        try {
            const r = await ctx.db.query(sql);
            if (r) {//若路径存在,则继续
                const v = r.filter(item => item.username !== username);
                //只要有一个权限不对,则拒绝,并将拒绝的sql行返回,否则,返回图片路径
                if (v.length) {
                    reject(v)
                } else {
                    resolve(r.map(item => item.src))
                }
            } else {//若路径都不存在
                resolve('未查到该文件权限')
            }
        } catch (e) {
            reject(e)
        }
    })
}
/**
*** 删除服务器中的图片(物理删除),并将
*** fileArr 要删除图片路径,数组
*/
function delFile(fileArr = []) {
    return new Promise((resolve, reject) => {
        let t = "";
        const d = [];
        const errInfo = {
            atleastOneSucces: false,//true 至少一个删除成功
            notExit: false,//至少一个文件不存在
            otherExit: false,//删除文件时至少一个文件发生其他错误
        }
        /** 循环删除文件 */
        while (t = fileArr.pop()) {
            const delPath = path.join(static_basepath, t);
            try {
                /**
                 * @des 判断文件或文件夹是否存在
                 */
                if (fs.existsSync(delPath)) {
                    fs.unlinkSync(delPath);
                    d.push({ message: "删除成功", path: t })
                    errInfo.atleastOneSucces = true;
                } else {
                    d.push({ errMsg: "文件不存在", path: t });
                    errInfo.notExit = true;
                }
            } catch (error) {
                d.push({ errMsg: error, path: t });
                errInfo.otherExit = true;
            }
        }
        if (errInfo.atleastOneSucces) {
            if (errInfo.notExit|| errInfo.otherExit) {
                return resolve({ message: "部分删除成功", code: 2, data: d });
            } else {
                return resolve({ message: "删除成功", code: 0, data: d });
            }
        } else if (!errInfo.atleastOneSucces) {
            return reject({ message: "删除失败", code: 3, data: d });
        }
    })
}
/** 
***在数据库删除图片信息
***pathArr 要删除图片路径,数组
*/
function delUserUploadsTableRow(ctx, next, pathArr) {
    const pathSql = pathArr.map(item => `'${item.replace(/^(http|https):\/\/[^\/]+/, "").replace(/^\s+|\s+$/gim, "")}'`).join(",");
    const sql = `DELETE FROM user_uploads WHERE src in (${pathSql})`;
    return new Promise(async (resolve, reject) => {
        try {
            await ctx.db.query(sql);
            resolve(true);
        } catch (e) {
            reject(e)
        }
    })
}
/** 删除图片总方法 */
async function removeFiles(ctx, next) {
    const params = ctx.request.body;
    //要删除的图片路径
    const paths = params.path;
    //当前操作用户
    const username = params.username;
    let filepathArr = [];
    /** 查看当前用户是否有权限 START*/
    try {
        filepathArr = await filePermissions(ctx, next, paths, username)
    } catch (e) {
        if (Array.isArray(e)) {
            ctx.response.status = 403;
            ctx.body = { message: '没有权限', code: 1, data: e };
        } else {
            ctx.response.status = 500;
            ctx.body = { message: e, code: 99 };
        }
        return;
    }
    /** 查看当前用户是否有权限 END*/
    
	/** 物理删除图片 START*/
	//存储删除信息
    let fileDeleteInfo;
    try {
        fileDeleteInfo = await delFile(filepathArr);
    } catch (e) {
        fileDeleteInfo = e;
        ctx.response.status = 500;
        ctx.body = { message: e, code: 99 };
        return;
    }
    /** 物理删除图片 END*/
    /** 在数据库删除已物理删除图片的信息 START*/
    try {
        const successDelArr = fileDeleteInfo.data.filter(item => item.message).map(item => item.path);
        await delUserUploadsTableRow(ctx, next, successDelArr);
    } catch (e) {
        fileDeleteInfo = { message: e, code: 4 };
    }
    /** 在数据库删除已物理删除图片的信息 END*/
    ctx.body = fileDeleteInfo;
}

番外

formidable.js

上面后端代码处理formData数据使用的是koa-body中间件,而koa-body是基于formidable.js,那么就可以使用formidable.js替代koa-body:

安装formidable.js

npm install formidable@v3

代码如下:

/** /routes/files.js */
const path = require("path");
const { uploadFiles, removeFiles } = require("../module/files")
const router = require('koa-router')();
const { IncomingForm } = require('formidable');//引入formidable
const { jsonable } = require('../middleware/files')
const fs = require("fs");
const { static_basepath, static_uploadpath } = require("../index.config");
const dayjs = require('dayjs');


router.post('/upload', async (ctx, next) => {
  const content_type = ctx.request.header['content-type']
   if (content_type === 'application/json') {
       return jsonable()(ctx, next);
   } else if (content_type.includes('multipart/form-data')) {
   		/** formidable插件使用 START */
   		//配置项
       const p = {
        //支持多文件上传
        multipart: true,
        // 上传目录
        uploadDir: path.join(__dirname, '../public/uploads'),
        // 保留文件扩展名
        keepExtensions: true,
       }
       const form = new IncomingForm(p);
       //上传前的事件
       form.on('fileBegin', (formName, file) => {
           ctx.request.formName = formName;
           const dirname = dayjs().format("YYYYMMDD");
           const dirpath = path.join(static_basepath, static_uploadpath, dirname)
           if (!fs.existsSync(dirpath)) {
               fs.mkdirSync(dirpath);
           }
           file.filepath = path.join(static_basepath, static_uploadpath, dirname, file.newFilename)
       })
       await new Promise((resolve, reject) => {
       		//解析Node.js Request对象
           form.parse(ctx.req, (err, fields, files) => {
               if (err) {
                   reject(err);
                   return;
               }
               ctx.request.files = files;
               ctx.request.body = {};
               for (let i in fields) {
                   ctx.request.body[i] = fields[i][0]
               }
               resolve({ fields, files });
           });
       });
       return next();
       /** formidable插件使用 START */
   }
}, uploadFiles);
router.post('/remove', removeFiles)
module.exports = router;

本文实现的后端方法不仅能上传图片,也可以上传其他文件,且可以多文件上传。

结语

新纪元的到来让世界勃勃生机,而角角落落依然遍布着草履虫们,因为它们仍旧适应这个世界。

参考资料

Axios 中怎么上传文件(Upload File)?上传方法有哪几种?
思否:在Koa.js中实现文件上传的接口
简书:NodeJs koa2实现文件上传
CSDN:koa-body koa2 使用 koa-body 代替 koa-bodyparser 和 koa-multer
CSDN:koa-body4接收formData数据
CSDN:koa-body 文件上传自定义文件夹及文件名称
稀土掘金:原生nodejs 处理文件上传
简书:node原生处理前端传送的数据
稀土掘金:上传文件时获取上传进度
W3cways:koa + formidable实现文件上传
npm:koa-body
npm:hexoid
npm:node-mime-types
CitCode–GitHub镜像

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

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

相关文章

spring boot3token拦截器链的设计与实现

⛰️个人主页: 蒾酒 &#x1f525;系列专栏&#xff1a;《spring boot实战》 &#x1f30a;山高路远&#xff0c;行路漫漫&#xff0c;终有归途。 目录 写在前面 流程分析 需要清楚的 实现步骤 1.定义拦截器 2.创建拦截器链配置类 3.配置拦截器链顺序 4.配置拦截…

OpenAI反击Elon Musk

每周跟踪AI热点新闻动向和震撼发展 想要探索生成式人工智能的前沿进展吗&#xff1f;订阅我们的简报&#xff0c;深入解析最新的技术突破、实际应用案例和未来的趋势。与全球数同行一同&#xff0c;从行业内部的深度分析和实用指南中受益。不要错过这个机会&#xff0c;成为AI领…

Ribbon实现Cloud负载均衡

安装Zookeeper要先安装JDK环境 解压 tar -zxvf /usr/local/develop/jdk-8u191-linux-x64.tar.gz -C /usr/local/develop 配置JAVA_HOME vim /etc/profile export JAVA_HOME/usr/local/develop/jdk1.8.0_191 export PATH$JAVA_HOME/bin:$PATH export CLASSPATH.:$JAVA_HOM…

力扣515. 在每个树行中找最大值(BFS,DFS)

Problem: 515. 在每个树行中找最大值 文章目录 题目描述思路复杂度Code 题目描述 思路 思路1&#xff1a;BFS 套用BFS模板&#xff0c;直接在遍历树的某一层时将当前层的最大值存入数组中 思路2&#xff1a;DFS 回溯思想&#xff0c;在递归时不断更新可选列表&#xff08;根据…

【word】引用文献如何标注右上角

一、在Word文档中引用文献并标注在右上角的具体步骤如下 1、将光标移动到需要添加文献标注的位置&#xff1a; 2、在文档上方的工具栏中选择“引用”选项&#xff1a; 3、点击“插入脚注”或“插入尾注”&#xff1a; ①如果选择的是脚注&#xff0c;则脚注区域会出现在本页的…

git:git revert 和git reset 回退版本的使用方式

目录 git revert还原某些现有提交git reset删除提交参考 git revert还原某些现有提交 中文文档&#xff1a;https://git-scm.com/docs/git-revert/zh_HANS-CN 版本会递增&#xff0c;不影响之前提交的内容 例如&#xff1a;撤销记录为 abc123 的提交 git revert abc123git r…

企微hook源码

企微hook源码已经在QQ群内开源。速度进群下载&#xff0c;避免和谐。 QQ群&#xff1a;649480745

【sgPhotoPlayer】自定义组件:图片预览,支持点击放大、缩小、旋转图片

特性&#xff1a; 支持设置初始索引值支持显示标题、日期、大小、当前图片位置支持无限循环切换轮播支持鼠标滑轮滚动、左右键、上下键、PageUp、PageDown、Home、End操作切换图片支持Esc关闭窗口 sgPhotoPlayer源码 <template><div :class"$options.name"…

【JS】关于this的使用

this 前言一、this是什么&#xff1f;二、做什么&#xff1f;1.全局环境2.函数环境3.new实例对象4.apply、bind、call绑定4.1 apply()4.2 call()4.3 bind() 三、为什么用this&#xff1f;四、如何改变this&#xff1f;五、应用场景&#xff1f;总结 前言 痛点 经常写Vue项目&a…

springboot集成logback打印彩色日志

一、logback介绍 Logback是由log4j创始人设计的另一个开源日志组件,官方网站&#xff1a; logback.qos.ch。它当前分为以下三个模块&#xff1a; logback-core&#xff1a;其它两个模块的基础模块。logback-classic&#xff1a;它是log4j的一个改良版本&#xff0c;同时它完整实…

qt练习案例

记录一下qt练习案例&#xff0c;方便学习qt知识点 基本部件 案例1 需求&#xff0c;做一个标签&#xff0c;显示"你好"知识点&#xff0c;QLabel画面 4. 参考&#xff0c;Qt 之 QLabel 案例2 需求&#xff0c;做一个标签&#xff0c;显示图片 知识点&#xff0c;…

【linux进程信号(二)】信号的保存,处理以及捕捉

&#x1f493;博主CSDN主页:杭电码农-NEO&#x1f493;   ⏩专栏分类:Linux从入门到精通⏪   &#x1f69a;代码仓库:NEO的学习日记&#x1f69a;   &#x1f339;关注我&#x1faf5;带你学更多操作系统知识   &#x1f51d;&#x1f51d; 进程信号 1. 前言2. 信号阻塞…

仅通过头部便可控制机器人实现生活自理!四肢瘫患者福音真的来了?

运动功能障碍对个体执行基本日常生活活动&#xff08;如沐浴、更衣、用餐&#xff09;以及进行工具性日常生活活动&#xff08;包括娱乐和社交互动&#xff09;造成了显著影响&#xff0c;不仅限制了他们的活动范围&#xff0c;而且削弱了他们维持独立生活的基础。受此类障碍困…

Netty架构

Netty逻辑架构 Netty 的逻辑处理架构为典型网络分层架构设计&#xff0c;网络通信层、事件调度层、服务编排层。 一、 网络通信层 网络通信层的职责是执行网络 I/O 的操作。它支持多种网络协议和 I/O 模型的连接操作。当网络数据读取到内核缓冲区后&#xff0c;会触发网络事件…

还在用Jenkins?试试这款阿里出品的IDEA插件,部署项目贼香

CloudToolkit简介 CloudToolkit是阿里出品的一款IDEA插件,通过它我们可以更方便地实现自动化部署,其内置的终端工具和文件上传功能,即使用来管理服务器也非常方便!这款IDEA插件不仅功能强大,而且完全免费! 安装 CloudToolkit的安装是非常简单的,直接在IDEA的插件市场…

如何在Vue中实现拖拽功能?

Vue.js是一款流行的JavaScript框架&#xff0c;用于构建用户界面。其中一个常见的需求是在Vue中实现拖拽功能&#xff0c;让用户可以通过拖拽元素来进行交互。今天&#xff0c;我们就来学习如何在Vue中实现这一功能。 首先&#xff0c;我们需要明白拖拽功能的基本原理&#xf…

yduibuilder,拖拽式开发轻松高效自动生成前端代码

给大家分享一个#开源项目# &#xff1a;#yduibuilder# &#xff0c;他可以通过拖拽式的开发方式自动生成前端代码&#xff0c;这种#低代码开发工具# 已经很多了&#xff0c;没什么新鲜的&#xff1b; 但yduibuilder式通过编译的方式#生成终端代码# &#xff0c;没有预设各种功能…

2.2 评估方法 机器学习

我们若有一个包含m个样例的数据集&#xff0c;若我们既需要训练&#xff0c;也需要测试&#xff0c;我们该如何处理呢&#xff1f;下面是几种方法&#xff1a; 2.2.1 留出法 “留出法”直接将数据集D划分为两个互斥的集合&#xff0c;其中一个作为训练集S&#xff0c;另一个作…

[数据结构初阶】栈

各位读者老爷好&#xff0c;鼠鼠我好久没写博客了&#xff08;太摆烂了&#xff09;&#xff0c;今天就基于C语言浅介绍一下数据结构里面的栈&#xff0c;希望对你有所帮助吧。 目录 1.栈的概念及结构 2.栈的实现 2.1定义栈 2.2.初始化栈 2.3.入栈 2.4.出栈 2.5.获取栈…

小白跟做江科大51单片机之LCD1602滚动显示效果

1.查看原理图 图1 LCD1602接口 图2 LCD1602与STC的接口 2.编写代码 图3 时序结构 根据时序结构编写命令和写入数据代码 #include <REGX52.H> #include "Delay.h" sbit LCD1602_ENP2^7; sbit LCD1602_RSP2^6; sbit LCD1602_WRP2^5; #define LCD1602_lCD0 …