input标签选择文件并回显,以及深入ajax文件的上传

Html结构

<input ref="fileInput" type="file" style="display: none" @change="handleFileUpdate" :accept="fileAccept">
<button @click="handleUploadBtnClick">点击上传<button/>

需要说明的是,这是在vue中写的。其中input标签上的accept属性表示选择文件时的文件类型,change事件主要是上传了文件后,input的值改变触发该事件,实现选择文件的获取。

fileAccept定义如下

fileAccept = '.png, .jpg, .jpeg, .svg'

Vue中实现点击按钮代码触发input点击事件。

通常来说,页面上我们不会直接放一个input来作为用户点击上传文件的按钮,一般会设置一个比较好看的按钮来代替。通过点击这个按钮来触发input的点击事件,显示文件选择对话框。所以这里 handleUploadBtnClick 方法的作用就是去触发input的点击。

handleUploadBtnClick() {
    this.$refs.fileInput.dispatchEvent(new MouseEvent('click'))
}

this.$refs.fileInput就是去获取input标签,以前在jquery中一般是下面这样写的

    $('input).trigger('click')

当在对话框中选择了文件过后,会触发input标签的change事件,也就是进入handleFileUpdate方法的逻辑。handleFileUpdate 方法在这里实现两个事情,第一是获取选择的文件,第二是将选择的文件回显到页面上。第二个逻辑在实际开发的需求中一般是将文件上传给后台接口。

handleFileUpdate(e) {
  var fileList = e.target.files;    // 文件默认是列表
  var file = fileList[0]
  var fileReader = new FileReader() // 文件读取
  fileReader.readAsDataURL(file)
  fileReader.onload = () => {
    console.log(fileReader.result);
    this.userIcon = fileReader.result
  }
}

e是js原生事件的默认参数。

e.target.files就是刚刚选择的图片,这里图片默认是一个列表,因为如果在input标签上设置multiple属性为"multiple"时,我们就可以选择多个文件。这里如果需要上传文件,一般处理的方式通过循环遍历这个列表,一个一个文件的上传。

FIleReader 是Html5中读取文件的API,FileReader的实例一共包含4个方法,其中readAsDataURL的作用是将文件读取为DataURL,也就是Base64编码的文件;读取的结果包含在其result属性中。其实例包含一套完整的事件模型,其中onload表示文件读取成功后触发的事件。

实现文件上传

FormData

要实现文件上传,首先得知道FormData的含义。MND上是这样解释FormData的,FormData 接口提供了一种表示表单数据的键值对 key/value 的构造方式,并且可以轻松的将数据通过XMLHttpRequest.send() 方法发送出去,本接口和此方法都相当简单直接。如果送出时的编码类型被设为 "multipart/form-data",它会使用和表单一样的格式

一共是两句话,

  • 从第一句话我们得到的信息是FormData实例是key/value的数据结构,并且可以通过XMLHttpRequest实例的send方法发送。

  • 第二句话表明如果请求头设置的编码类型为"multipart/form-data",那么它会和表单一样直接发送而不会进行额外的编码。这里还需要了解的是,为什么需要设置multipart/form-data 这种请求类型

HTTP Content-Type

Content-Type是什么?
Content-Type(内容类型),用于定义网络文件的类型和网页的编码,决定浏览器将以什么形式、什么编码读取这个文件。

常见的Content-Type类型一般为 application/x-www-form-urlencoded,它表示在发送前编码所有字符。还有一种常见的是text/plain,它也会在发送前进行字符编码编码,但只会编码空格,特殊字符不会进行编码。
multipart/form-data 的含义就是不对字符进行编码。最开始,因为application/x-www-form-urlencoded不适合用于传输大型二进制数据或者非ASCII字符的数据,所以重新添加了multipart/form-data类型,此类型专门用于有效的传输的文件。

实现文件上传

原生js实现

let formData = new FormData()
formData.append('file', file)   // file为input选择的文件
formData.append('name': 'fileName') // 可以继续添加其他值

let xhr = new XMLHttpRequest()
xhr.open('post', url, true)
xhr.send(formData)

xhr.open(method, url, async) 接收三个参数分别是请求方法、请求地址、是否为异步

另外formData.append(key, value)添加键值对时,可以重复添加相同key的键值对,不会覆盖之前的。

以为这样就可以了?上面的代码是不是忘记什么东西了啊?
Content-Type的设置去哪儿了啊???

所以,这里又出现了一个小的疑问,为什么我们并没有设置请求头的类型Content-Type为multipart/form-data还是可以成功的上传文件?是的,我们真的上传成功,我们可以在控制台看看请求信息:


Content-Type1.png

是的,就是你现在想的,浏览器自动帮我们把请求头加上了multipart/form-data类型的设置。那么,如果我们手动设置一下会有影响吗?

我们在发送请求之前线设置一下请求头

let xhr = new XMLHttpRequest()
xhr.open('post', url, true)
xhr.setRequestHeader('Content-Type', 'multipart/form-data')
xhr.send(formData)

同样看看控制台的请求有没有不一样


Content-Type2.png

仔细看(也不用仔细看了,就是我红框标出来的地方),Content-Type确实也设置成了multipart/form-data。但是后面怎么少了一串???

通过比较两次请求可以看到少的是boundary=xxx这么一串,再仔细一看这个boundary=xxx后面的xxx是不是在下面的参数列表中也有啊。那么,这个boundary是什么呢?

boundary是分割符,分割多个参数;其中xxx是即时生成的字符串,避免在实际参数内容中出现。后台可以通过分隔符获取请求体里的数据

好了,搞了这么多东西,明白了我们不用在代码中特地设置Content-Type了。好像是练太极,我:师傅我全忘了。师傅:好了,你已经练成了。

jquery实现

$.ajax({
    url,
    type: 'post',
    data: formData,
    processData: false, // 不用将数据转换为对象
    contentType: false, // 不用设置数据类型
})

通过上面的说明,我们也知道为什么会设置contentType为false这句看似矛盾的代码了,为的就是避免jQuery再设置一次Content-Type,导致没有分隔符。

axios 实现

axios({
    method: 'post',
    url: 'http://192.168.2.164:3000/file_upload',
    data: formData,
    withCredentials: false,
    headers: {
      // 'Content-Type': 'multipart/form-data'
    }
})

经过实测,"axios": "^0.20.0" 即使设置了'Content-Type': 'multipart/form-data'也不会没有分隔符。个人建议还是加上这个代码设置。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 218,941评论 6 508
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,397评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 165,345评论 0 356
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,851评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,868评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,688评论 1 305
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,414评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,319评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,775评论 1 315
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,945评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,096评论 1 350
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,789评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,437评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,993评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,107评论 1 271
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,308评论 3 372
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,037评论 2 355