图片上传和预览踩坑

功能需求

  • 1.完成图片的上传和上传之后图片的预览功能

遇到的问题

  • 1.使用html的src标签,可以直接发送请求。
    • 通过html标签直接发送的请求没有办法添加header(a、img、form)
    • 因此如果服务器需要check你的Headers中的Authorization来确定是否给当前请求提供服务,那么通过标签发送的请求必然不能被授予服务
  • 2.图片的上传需要给服务器发送post请求,那么post请求的Headers以及body要求是什么呢?

完成功能

使用expressjs 作为服务器处理图片上传和预览。使用antd上传组件。

  • Q1:使用了antd upload组件本身带的上传功能,那么如何确定请求的路径headers以及body呢?

  • A1: Upload组件本身带了很多属性

    • action:确定上传的路径
    • headers:确定上传请求的headers
    • name:确定了后端抓取文件需要访问的字段
    • body:如果你没有改写上传的方法 body是确定的
      { 
        fieldname: 'file',
        originalname: 'WechatIMG378.jpeg',
        encoding: '7bit',
        mimetype: 'image/jpeg'
      }
      
  • Q2:如何在文件上传之前check文件的类型和大小?

  • A2:antd的Upload组件包含属性beforeUpload function这个函数会在上传前被调用并且传入上传文件的对象file作为参数

    • file.size: 以bytes为单位。
      • 例子:要求文件大小小于3M
        • file.size / 1024 / 1024 < 3
    • file.type: 指的是文件发送的类型
      • 例子:'image/jpeg', 'image/jpg', 'image/png', 'image/bmp'
  • Q3: 对于图片预览部分我们不能直接使用img src,那么我们该怎么办?

  • A3:

    • 首先src的值可以是什么呢?
      • 1.是一个被webpack打包好的图片
      • 2.是一个可以请求到图片的路径
      • 3.是图片的base64编码
      • 但是前两种方式都是通过发请求获取的
    • 这两种方式的区别
      • url:
        • 1.需要发送一次请求
        • 2.如果图片文件的所有信息都不变俺么将会被缓存在浏览器中
      • base64:
        • 1.不需要发送请求直接内嵌在html中
        • 2.不能被浏览器缓存
        • 3.当图片尺寸小的时候我们会选择使用这种方式,不占用一次http的会话
    • 因此我们可以采用发送ajax请求获取图片的编码
    • 将base64赋给img src即可
  • Q4: 对于请求一个图片(文件)的情况,这些图片和文件是以什么样的模式在传输呢?为什么浏览器可以直接展示出图片?

  • A4: 图片和文件都是以二进制流传输的(或者可以说是文件对象的原始数据),浏览器可以直接将从服务器接受的二进制流自动转换成对应的文件和图片


  • Q5:客户端接受服务器传来的多类型数据又是如何被浏览器识别并处理的呢?(浏览器怎么知道该如何处理二进制数据流或者对象呢)
  • A5: 首先需要知道MIME类型:
    • 是一种通知客户端其接收文件的多样性的机制
    • 服务器会在response的Content-Type设置当前response body的MIME type
    • 当浏览器接收到请求的时候会根据MIME type使用相应的方式进行对数据的处理
    • 举个栗子
      • 在img 的src中设置一段请求静态图片的路径
      • express.js做服务器 res.sendFile
      • 浏览器查看请求信息
      • 再次查看请求的信息发现response headers的content-type是image/jpeg
      • response body是一个二进制数据流
      • 因此浏览器通过识别出content-type是image/jpeg然后采用对图片的处理机制去处理图片并生成
    • 和问题相关
      • 因此当你使用ajax发送请求获取图片,可以获取图片的二进制流
      • 但是没有方法处理这个二进制流
      • 所以没有办法展示图片

  • Q5: 客户端该如何处理返回的二进制流并将其转换成base64编码呢?
  • A5:首先得让服务器返回一个你知道如何转换的二进制流(让返回的数据类型已知可控)
    • 对于不同的发送请求的方式,我们都有一个方法可以设置返回的数据类型
      • 举个栗子:使用axios发送请求
        axios.get('/****',{
            responseType: 'blob'
        })
        
    • 响应的类型(responseType的类型):
      • document: 静态页面
      • blob:适用于读取二进制数据,比如图片文件
      • arrayBuffer:也是用来处理处理二进制数据但是是按照数组的方式处理二进制数据
      • json: 数据,虽然和text差不多但是最大的优点可以使用JSON.parse将string变成对象
      • text: 字符串
        • 默认值是json
    • 这时候知道了返回的数据一定是一个blob类型的数据就可以想办法将blob数据读成base64即可

  • Q7:responseType的blob和arraybuffer之间的区别是什么呢?
  • A7:都是二进制数据的容器,那么对于这种二进制流的数据我们该选择转换成哪一种容器呢?
    • blob:
      • 代表原始的二进制数据(js都不一定认得)
      • blob更偏重于整体操作(整个文件的二进制流)
    • arrayBuffer:
      • js可以识别的二进制数据
      • 可以做流的切割,所有注重于对二进制流中的字节进行处理
    • 因此在这里,我们更加偏重于对整个图片二进制流的处理因此选择将response转换成一个blob

  • Q8:如何将blob转换成base64给src赋值呢?
  • A8:FileReader可以将blob或者file类型的数据装换成其他类型读出
    • FileReader
    • 创建一个FileReader对象new FileReader()
    • 然后调用fileReader.readAdDataUrl(blob)
    • 将后端传回的url直接转成src的value
    • 但是fileReader的方法都是异步的
    • 因此需要定义一个回调函数
    • fileReader.onLoad((e)=>{
      //onload会在读取成功的时候执行
      document.createElement('img').src = e.target.result;
      })
发现js中文件的处理部分还是很不熟悉js文件和二进制流的处理
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,470评论 6 501
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,393评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,577评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,176评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,189评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,155评论 1 299
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,041评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,903评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,319评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,539评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,703评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,417评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,013评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,664评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,818评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,711评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,601评论 2 353