文件的上传和下载

news2024/12/24 11:35:38

一、node实现文件上传

1、FormData对象:以对象的方式来表示页面中的表单,又称为表单对象。以key-value的方式来保存数据,XMLHttpRequest对

象可以轻松的表单对象发送的服务器端

​ (1)是一个构造函数:new FormData(),例如:

var  formdata = new FormData(form)  //将页面中的表单form转换成FormData对象(即将表单数据转换成key-value对)

 (2)常用的API:

​ FormData.append(key,value):追加数据。向formdata中追加key-value

​ FormData.get(key):获取key对应的值

​ FormData.delete(key):删除key对应的值

​ FormData.has(key):判断formdata中是否有key

2、node使用formidable模块实现文件上传

​ (1)安装:npm install formidable

​ (2)创建Formidable.IncomingForm对象:本质是一个表单对象

​ let form = new Formidable.IncomingForm()

​ (3)Formidable.IncomingForm对象的属性:

​ form.encoding : 设置字符集

​ form.uploadDir:设置上传文件的保存路径

​ form.keepExtensions:true,表示上传时保留原来的扩展名

​ (4)Formidable.IncomingForm对象的方法:

​ form.parse(request, [callback]):转换请求中的表单数据

​ (5)Formidable.File对象的属性

​ size:上传文件的大小

​ path:上传文件的路径

​ type:上传文件的类型

​ name:上传的文件名

3、案例:

​ 前端代码:

<body>
    <!-- 上传图片:
        (1)将图片文件上传到服务器端的指定目录
        (2)将图片文件名和服务器的地址进行拼接
        (3)将拼接后的图片的路径响应给客户端,在div中显示出来
    -->
    <label>请选择文件:
        <input type="file" id="file">
    </label>
    <br><br>
    <div id="box">div</div>
    <script>
        $(function(){
            //给file控件绑定change事件
            $('#file').change(function(){
                //1. 获取input中得到的文件名
                if($('#file').val().length){ //如果用户选择了文件
                    let fileName = $('#file').val()  //获取文件名
                    console.log(fileName)
                    //获取文件的后缀(扩展名),并把后缀名转换成小写
                    let extenName = fileName.substring(fileName.lastIndexOf('.'),fileName.length).toLowerCase()
                    if(extenName==='.jpg' || extenName==='.png'){ //若文件是图片
                        let  formdata =  new FormData() //创建表单对象
                        console.log($('#file')[0])
                        console.log($('#file')[0].files[0])
                        formdata.append('uploadFile',$('#file')[0].files[0])
                    }else{
                        alert('文件格式错误')
                    }
                }
                // 2.向服务器发起上传请求
                $.ajax({
                    url:'http://localhost:8089/upload',
                    type: 'post',
                    data: formdata,
                    cache: false,//上传时文件不缓存
                    contentType: false, //jQuery不能设置请求头中的contentType ,必须的设置
                    processData: false, //jQuery不能处理上传的数据,必须配置
                    success: function(result){
                        // 2.1 创建一个img标签
                        let img = document.createElement('img')
                        // 2.2 设置img标签的样式
                        img.style.width = 350+'px'
                        img.style.height = 280+'px'
                        // 2.3 设置img标签的src属性
                        img.src = result.path  //path:表示是服务器响应的图片的访问路径
                        // 2.4 显示的设置
                        $('img').remove() //有新图片显示时,将原图片删除
                        // 2.5 将img标签添加到div中
                        $('#box').append(img)
                    },
                    error: function(error){ //请求-响应失败后的处理代码
                        console.log(error)
                    }
                })
            })
        })
    </script>
</body>

后台(上传-下载服务器)

//上传接口:http://localhost:8089/updown/upload
router.post('/upload',(req, res) => {
      // 1.设置上传文件的保存路径:上传文件夹
    let cacheFolder = 'public/images/uploads'
      // 2.判断上传文件夹是否存在,若不存在则创建
    if (!fs.existsSync(cacheFolder)){ //同步判断
        fs.mkdirSync(cacheFolder) //同步创建
    }
      // 3. 创建form对象接收客户端的formdata中的数据:使用formidable模块的IncomingForm
    let form = new formidable.IncomingForm()
    form.encoding = 'utf-8' //设置表单域的字符集
    form.uploadDir = cacheFolder //设置上传目录
    form.keepExtensions = true //保留上传文件的后缀
    form.maxFieldsSize = 2 * 1024 * 1024 // 1K=1024B   1M=1024K
    form.type = true //上传文件的类型为只读
      // 4.接收上传文件并进行处理
    let displayUrl; //上传的文件在服务器端的访问路径
    form.parse(req,function (err,fields,files){ //files是前端的formdata对象
        if (err){ //表示上传错误
            res.send(err)
            return
        }
        let extName='' //存放上传文件的后缀(扩展名)
        switch(files.uploadFile.mimetype){ //uploadFile对应的是前端formdata中的key,mimetype表示上传文件的类型
            case 'image/jpg':
                extName = '.jpg'
                break
            case 'image/jpeg':
                extName = '.jpg'
                break
            case 'image/png':
                extName = '.png'
                break
        }
        //5. 对上传文件的后缀进行处理
        if(extName.length == 0){
            res.send({
                code: 202,
                msg: '只能上传jpg和png格式的文件'
            })
        }else{ //符合上传要求的文件
           // 5.1 将上传文件和上传的文件名进行拼接
            let savePath = form.uploadDir + '/'+ files.uploadFile.originalFilename //originalFilename表示上传文件的原名
            displayUrl = `http://localhost:8089/images/uploads/${files.uploadFile.originalFilename}`
            //5.2 对文件重命名
            //files.uploadFile.filepath : C:\fakepath\2.png
            //savePath:D:\woniu\Web前端11期\12-2\demo\up-down\public\images\uploads\2.png
            fs.renameSync(files.uploadFile.filepath,savePath) //用savePath替换filePath,方法必须是同步方法
            res.json({
                code: 202,
                path: displayUrl
            })
        }
    })
})

4、文件上传需要注意的问题

​ (1)前端FormData对象作用:用于保存上传文件的信息。格式:key-value

​ (2)后台formidable模块的作用:使用parse方法来解析前端的formdata对象

​ (3)上传过程中前后端的对应关系

二、Express实现文件下载

1、使用res对象(响应对象)的download方法即可

三、遍历下载文件夹下的文件,拼接成一个下载的url,传递到前端

后台接口:

/ 遍历下载文件夹的接口:http://localhost:8089/updown/getfiles
router.get('/getfiles',(req, res) => {
    //1.遍历下载文件夹:public/images/uploads
    let filePath = path.join(__dirname,'../../public/images/uploads/') //下载文件夹
    let url_arr = [] //存放下载的url
    fs.readdir(filePath,(err,files)=>{ //files参数中存放的filePath下的子目录名和文件名
        if (err){
            console.log(err)
        }else{
            for(let i=0;i<files.length;i++){
                let fileDir = path.join(filePath,files[i])
                if (fs.statSync(fileDir).isFile()){ //表示当前的files[i]是一个文件
                    let urlObj = {
                        fileName: files[i],  //文件名
                        downloadUrl: `http://localhost:8089/updown/download?fileName=${files[i]}`
                    }
                    url_arr.push(urlObj) //把要下载的文件路径、文件放入数组中
                }
            }
            res.json(url_arr)
        }
    })
})

前端页面:

<body>
    <div id="box"></div>
    <script src="./js/jquery-3.4.1.js"></script>
    <script>
        $(function(){
            $.ajax({
                url: 'http://localhost:8089/updown/getfiles',
                type:'get',
                dataType:'json',
                success: function(result){
                    for(let data of result){
                        $('#box').append(`<a href=${data.downloadUrl}>下载${data.fileName}</a>`)
                                 .append('<br><br>')
                    }
                }
            })
        })
    </script>
</body>

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

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

相关文章

力扣hot100——第4天:19删除链表的倒数第N个节点、20有效的括号、21合并两个有序链表

文章目录1.19删除链表的倒数第N个节点【代码随想录已刷】2.20有效的括号【代码随想录已刷】3.21合并两个有序链表3.1.题目3.2.题解1.19删除链表的倒数第N个节点【代码随想录已刷】 参考&#xff1a;力扣题目链接&#xff1b;自己的博客解答 2.20有效的括号【代码随想录已刷】…

whistle监听方法

视频教程 程序员抓包神器&#xff0c;快速定位线上bug_哔哩哔哩_bilibili whistle官网 关于whistle GitBook 安装whistle后&#xff0c; 复制启动网址 系统代理也设置完毕后&#xff0c;在浏览器打开http://192.168.4.238:8899,即可进行抓包 拦截js&#xff0c;并执行自定…

10年的老测试告诉你八大测试用例设计方法

一&#xff1a;等价类划分法 1:有效等价类: 2:无效等价类: 案例:比如一个登陆输入框,规定只能输入中文,同时长度为6-10。 通过等价类设计测试用例: 测试用例中重要的三步: 输入 操作 预计结果 如果与预期结果不符合就是bug。 有效等价类: 输入:输入长度为6的中文,输入的为…

微信小程序接口请求多文件+参数上传、单文件+参数上传(formData形式) 微信小程序实现formData格式传参(亲测有效)

01.引入所需formData js文件 1.文件链接 链接: https://pan.baidu.com/s/1BDxx0-1KMAnkceXb45L5rg 提取码: 6ibp 2.引入使用 const FormData require(../../../../utils/formData.js)formData.js与mimeMap.js请确保在同一层级 02.formData参数 1.参数设置 let data {a…

Vue3 Composition API(案例)

前言&#xff1a;如果你是从vue2转到vue3的一份子&#xff0c;那么你重点学一下核心内容Composition API 。vue3的更新使代码写起来更加清晰&#xff0c;而且更接近于原生开发&#xff0c;对TS支持友好,现在我们来学习一下API。 如果你也想学一下TS那么请看这里 带你学习语法T…

力扣(LeetCode)1769. 移动所有球到每个盒子所需的最小操作数(C++)

暴力循环 直观模拟&#xff0c;对于某个固定的盒子&#xff0c;可以遍历所有盒子&#xff0c;∑\sum∑ 遍历的盒子里的球数 \times 遍历的盒子到固定的盒子的距离&#xff0c;得移动所有球到固定盒子的最小操作数。依次固定所有盒子&#xff0c;遍历&#xff0c;得到答案。 c…

嵌入式Linux 开发经验:platform_driver_register 的使用方法

前言 嵌入式Linux 设备驱动开发时&#xff0c;经常遇到平台驱动 platform_driver_register 的注册&#xff0c;最近深入了看了驱动开发为何使用平台驱动 开发一个设备驱动时&#xff0c;为了实现 设备的 打开、关闭、控制等操作&#xff0c;可以注册为 Linux misc 设备&#x…

Ra-08透传固件应用

目录1、功能介绍2、硬件接线3、固件烧录4、应用说明指令说明应用示例5、联系我们1、功能介绍 Ra-08透传固件主要功能有&#xff0c;设置发送或者接收模式&#xff0c;配置各个射频参数&#xff0c;设置本地地址与发送的目标地址&#xff0c;设置进入睡眠模式等。 2、硬件接线…

vite+ts-5-Sequelize框架优化

random recording 随心记录 What seems to us as bitter trials are often blessings in disguise. 看起来对我们痛苦的试炼&#xff0c;常常是伪装起来的好运。 此系列是一个前后端整合项目&#xff0c;跟往期关联&#xff0c;链接传送到达&#xff1a; 1.vitets-1-前期准备(尽…

FreeRtos于嵌入式环境的应用

FreeRtos操作系统 首先&#xff0c;应该介绍什么是FreeRtos&#xff0c;他于单片机而言就是一个管理器&#xff0c;作为管理者管理嵌入式芯片中的任务&#xff0c;堆栈&#xff0c;中断&#xff0c;队列等等资源&#xff0c;对于操作系统而言&#xff0c;又分为实时操作系统和…

分布式事物-全面详解(学习总结---从入门到深化)

分布式事物处理_认识本地事物 什么是事物 事务就是针对数据库的一组操作&#xff0c;它可以由一条或多条SQL语句组 成&#xff0c;同一个事务的操作具备同步的特点&#xff0c;事务中的语句要么都执 行&#xff0c;要么都不执行。 举个栗子&#xff1a; 你去小卖铺买东西&#…

kafka中partition数量与消费者对应关系以及Java实践(Spring 版本)

文章目录分区理解一、单播模式&#xff0c;只有一个消费者组1. topic只有1个partition2. topic有多个partition,该组内有多个消费者二、广播模式&#xff0c;多个消费者组2.1. 多个消费者组&#xff0c;1个partition2.2. 多个消费者组&#xff0c;多个partition三、Java实践-pr…

谈谈JS二进制:File、Blob、FileReader、ArrayBuffer、Base64

当互联网进入存量时代&#xff0c;增量正在成为行业的稀缺资源。而本地生活服务恰恰是当前互联网行业为数不多的增量。 前瞻产业研究院数据显示&#xff0c;2021年中国互联网本地生活服务行业市场规模达到2.6万亿元&#xff0c;到2025年&#xff0c;其市场规模有望达到4万亿元&…

吉林优美姿文化:抖音店铺怎么优化页面?

要知道&#xff0c;新手开始做直播间影响人气的三大要素就是在线人数&#xff0c;互动量&#xff0c;以及留存率&#xff0c;那么当这些要素都非常高的时候&#xff0c;抖音系统就会自动把你的直播间推荐给更多的观众&#xff0c;获取更多的流量&#xff0c;那么抖音新手怎么开…

简单了解Vue

1、vue概述 Vue是一套前端框架&#xff0c;可以免除原生JavaScript中的DOM操作&#xff0c;简化书写。 基于MVVM&#xff08;Model-View-View Model&#xff09;思想&#xff0c;实现数据的双向绑定&#xff0c;将编程的关注点放在数据上 vue的官网&#xff1a;https://cn.v…

猿如意---Python3.10版本手把手教学安装和下载.

亲自为大家示范如何使用猿如意以及在猿如意当中下载&#xff0c;安装和使用python3.10版本&#xff0c;让大家喜欢上这款好用的app—猿如意。 文章目录前言一、手把手教你猿如意的安装、下载二、手把手教你Python3.10版本的安装、下载1.找到我需要的工具2.我需要的工具的安装、…

Docker容器

1.什么是Docker&#xff1f; 1.为什么会出现docker&#xff1f; 我们写的代码会接触到好几个环境&#xff1a;开发环境、测试环境以及生产环境等等。多种环境去部署同一份代码&#xff0c;由于环境原因往往会出现软件跨环境迁移的问题&#xff08;也就是“水土”不服&#xf…

Android车载应用开发——DLNA开发浅析

DNLA的建立 DLNA 成立于2003 年6 月24 日, 其前身是DHWG &#xff08;Digital Home Working Group 数字家庭工作组&#xff09;&#xff0c;由Sony、Intel、Microsoft等发起成立、旨在解决个人PC &#xff0c;消费电器&#xff0c;移动设备在内的无线网络和有线网络的互联互通…

基于x86架构的CentOS7虚拟机通过qemu安装ARM架构OpenEuler虚拟机

【原文链接】基于x86架构的CentOS7虚拟机通过qemu安装ARM架构OpenEuler虚拟机 &#xff08;1&#xff09;首先需要有一台CentOS虚拟机&#xff0c;如没有可参考 VMWare安装CentOS7操作系统的虚拟机 安装一台CentOS虚拟机 &#xff08;2&#xff09;安装基础命令 yum install…

17:00面试,17:09就出来了 ,问的实在是太...

从外包出来&#xff0c;没想到算法死在另一家厂子 自从加入这家公司&#xff0c;每天都在加班&#xff0c;钱倒是给的不少&#xff0c;所以也就忍了。没想到8月一纸通知&#xff0c;所有人不许加班&#xff0c;薪资直降30%&#xff0c;顿时有吃不起饭的赶脚。 好在有个兄弟内…