最近做了一个JS(Jquery)上传本地图片的需求,做完之后才觉得如果不依靠一些前端框架来做这个事情,过程中真的会遇到很多的坑。

思路

image

遇到的坑
  • 浏览器本地安全模式,不能读取本地文件的问题
  • 本地图片没有onload事件,complete属性为flase,并且取不到宽和高的问题
  • 图片对象转base64的问题
代码(可直接运行)
<html>
    <title></title>
    <head>
        <style>
            .pro_file{
                width: 500;
            }
        </style>
    </head>
    <body>
        <input type="file" class="pro_file"/><br/>
        <input type="button" value="上传" id = "btn">
    </body>
    <script src="http://cdn.bootcss.com/jquery/3.2.1/jquery.js"></script>
    <script>
        $(function(){
            $(".pro_file").change(function () {
                var $file = $(this);
                var fileObj = $file[0];
                var windowURL = window.URL || window.webkitURL;
                var dataURL;
                if (fileObj && fileObj.files && fileObj.files[0]) {
                    dataURL = windowURL.createObjectURL(fileObj.files[0]);
                }
                if (fileObj.value != "" && fileObj.value != undefined) {
                    getBase64WithImageUrl(dataURL).then(function (base64) {
                        console.log(base64);
                        $("body").append("<img  src='" + base64 + "'>");
                        }, function (err) {
                            console.log(err);
                        });
                }
            });
        });
        //根据图片路径获取图片的base64字符串
        function getBase64WithImageUrl(imgurl) {
            var image = new Image();
            image.src = imgurl;
            var deferred = $.Deferred();
            if (imgurl) {
                image.onload = function () {
                    deferred.resolve(getBase64WithImage(image));//将base64传给done上传处理
                }
                return deferred.promise();//问题要让onload完成后再return
            }
        }
        //根据一个图片对象获取base64字符串
        function getBase64WithImage(img, width, height) {
                var canvas = document.createElement("canvas");
                canvas.width = width ? width : img.width;
                canvas.height = height ? height : img.height;
                var ctx = canvas.getContext("2d");
                ctx.drawImage(img, 0, 0, canvas.width, canvas.height);
                var dataURL = canvas.toDataURL();
                return dataURL;
         }
     </script>
</html>
注意

js将图片转为base64字符串的格式是:

...

真正的base64字符串其实是:data:image/png;base64,后面的部分

iVBORw0KGgoAAAANSUhEUgAAABQA...

所以在一些服务端如果要将这个base64字符串转图片的话,应该先将后面的base64字符串截取出来,前面的那段只是一个文件的内容描述。这个坑我刚刚已经踩过了(痛哭三秒/(ㄒoㄒ)/~~,刚刚跟后台的妹子捣腾了半天才发现这个是这个原因),并且到网上一搜,发现其实很多人遇到了很跟我一样的问题,并且都没有什么好的解决办法...他们大多是这样问的,《为什么我的base64用网上的一些工具可以转图片,在java/C#/...里面却不行》,大概是这些老铁跟我一样一开始并没有搞清楚真正的base64的格式,才会一直纠结在这个问题上吧,当然base64转不了图片也可能是在传输的过程中被修改了,或者里面有一些特殊字符的出现也不一定。

Base64
Base64只是一种编码格式,并不是什么加密...

刚刚去找解决方案的时候,发现很多人都说Base64加密,吓得我立马去找了一下百度百科:

Base64
Base64是网络上最常见的用于传输8Bit字节代码的编码方式之一,大家可以查看RFC2045~RFC2049,上面有MIME的详细规范。Base64编码可用于在HTTP环境下传递较长的标识信息。例如,在Java Persistence系统Hibernate中,就采用了Base64来将一个较长的唯一标识符(一般为128-bit的UUID)编码为一个字符串,用作HTTP表单和HTTP GET URL中的参数。在其他应用程序中,也常常需要把二进制数据编码为适合放在URL(包括隐藏表单域)中的形式。此时,采用Base64编码具有不可读性,即所编码的数据不会被人用肉眼所直接看到。

反思

以前自己以为自己会使用javascript和jquery,并且觉得这些东西很简单,现在接触的越多越发现自己只会一些简单的操作,真正遇到比较棘手的问题的时候,就显得力不从心,虽然现在的自己主攻的不是web相关的东西,full stock developer在路上。

END