首页 web前端 H5教程 HTML5应用之文件上传的详细介绍

HTML5应用之文件上传的详细介绍

Mar 27, 2017 pm 04:39 PM

这篇文章主要介绍了HTML5应用之文件上传,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧.

长期以来,开发者们一直为此苦恼,大部分为解决这个问题都采用了flash作为解决方案,但flash并非灵丹妙药,因为flash版本,割据造成的问题有时反倒成为了噩梦。有些网站则采用了form标签的enctype=multipart/form-data属性,但这一属性要求服务器作出特殊的设置才能够显示进度,而且本身也比较复杂,复杂就意味着容易出现错误,这可不是我们想要的。

现在我们来看看Html5为什么能够解决这个问题,以及,它到底能做的多好。

用HTML5上传文件

在HTML5标准中,XMLHttpRequest对象被重新定义,被称为“XMLHttpRequest Level 2”,其中包含了以下5个新特性:

1、支持上传、下载字节流,比如文件、blob以及表单数据

2、增加了上传、下载中的进度事件

3、跨域请求的支持

4、允许发送匿名请求(即不发送HTTP的Referer部分)

5、允许设置请求的超时

在这篇教程中,我们主要关注第一和第二项特性,尤其是第二项——它能够提供我们想要的上传进度。和之前的方案不同,这个方案并不要求服务器作出特殊的设置,因此大家边看教程就可以边动手试试了。

HTML5应用之文件上传的详细介绍

HTML5应用之文件上传的详细介绍

上面图示的就是我们能够实现的内容:

1、显示上传的文件信息,比如文件名、类型、尺寸

2、一个能够显示真实进度的进度条

3、上传的速度

4、剩余时间的估算

5、已上传的数据量

6、上传结束后服务器返回的响应

另外,凭借XMLHttpRequest,我们的上传过程整个都是异步的,因此用户在上传文件的时候,依然可以操作网页当中的其它元素,并不需要专门等待上传的完成。而在上传结束后,我们能够获取服务器发回的响应,因此整个上传过程都显得相当顺理成章。

HTML5的进度事件

HTML5当中新增了一个进度事件(Progress Events),这个事件为我们提供了以下信息:

1、total – 文件大小

2、loaded – 已上传的大小

3、lengthComputable – 进度是否可计算

信息并不多,但是在计算文件进度上已经足够了。当然,也还有很多东西它没有直接给出,这非常遗憾。

HTML

与普通的文件上传代码并没有太大差异。不过注意,input标签关联了一个JavaScript函数在onchange上。


 <!DOCTYPE html>
<html>
<head>
    <title>使用XMLHttpRequest上传文件</title>
</head>
<body>
<form id="form1" enctype="multipart/form-data" method="post" action="upload.php">
<p class="row">
      <label for="fileToUpload">Select a File to Upload</label>
<input type="file" name="fileToUpload" id="fileToUpload" onchange="fileSelected();"/>
    </p>
<p id="fileName"></p>
<p id="fileSize"></p>
<p id="fileType"></p>
<p class="row">
<input type="button" onclick="uploadFile()" value="Upload" />
    </p>
<p id="progressNumber"></p>
</form>

</body>
</html>

JavaScript

一旦我们在HTML中使用了input,我们就可以在JS代码中获取到一个FileList对象。这个对象是HTML5中新增加的文件API中的一部分,每一个FileList对象都是一组文件对象的集合,而文件对象则拥有下列的属性:

1、name – 文件名(不包含路径)

2、type – 文件的MIME类型(小写)

3、size – 文件的尺寸(单位为字节)

这正是我们所需要的。当然,HTML5中还有一个FileReader对象,但在这里我们并没有用它。现在,通过上面的三个内容,我们已经能够控制用户上传的文件大小和文件类型,以便减轻服务器再次检测时的压力,并提升安全系数。


 function fileSelected() {
  var file = document.getElementById(&#39;fileToUpload&#39;).files[0];
  if (file) {
    var fileSize = 0;
    if (file.size > 1024 * 1024)
      fileSize = (Math.round(file.size * 100 / (1024 * 1024)) / 100).toString() + &#39;MB&#39;;
    else
      fileSize = (Math.round(file.size * 100 / 1024) / 100).toString() + &#39;KB&#39;;

    document.getElementById(&#39;fileName&#39;).innerHTML = &#39;Name: &#39; + file.name;
    document.getElementById(&#39;fileSize&#39;).innerHTML = &#39;Size: &#39; + fileSize;
    document.getElementById(&#39;fileType&#39;).innerHTML = &#39;Type: &#39; + file.type;
  }
}

那么当用户选择好文件,点击上传之后,又将发生什么呢?


 function uploadFile() {
  var xhr = new XMLHttpRequest();
  var fd = document.getElementById(&#39;form1&#39;).getFormData();

  /* event listners */
  xhr.upload.addEventListener("progress", uploadProgress, false);
  xhr.addEventListener("load", uploadComplete, false);
  xhr.addEventListener("error", uploadFailed, false);
  xhr.addEventListener("abort", uploadCanceled, false);
  /* Be sure to change the url below to the url of your upload server side script */
  xhr.open("POST", "upload.php");
  xhr.send(fd);
}

function uploadProgress(evt) {
  if (evt.lengthComputable) {
    var percentComplete = Math.round(evt.loaded * 100 / evt.total);
    document.getElementById(&#39;progressNumber&#39;).innerHTML = percentComplete.toString() + &#39;%&#39;;
  }
  else {
    document.getElementById(&#39;progressNumber&#39;).innerHTML = &#39;unable to compute&#39;;
  }
}

function uploadComplete(evt) {
  /* This event is raised when the server send back a response */
  alert(evt.target.responseText);
}

function uploadFailed(evt) {
  alert("There was an error attempting to upload the file.");
}

function uploadCanceled(evt) {
  alert("The upload has been canceled by the user or the browser dropped the connection.");
}

在代码的第二行中,我们的JS代码又使用了另一个HTML5推出的新对象——FormData。FormData对象是用户的表单数据的集合,它以键值对的形式存储了表单数据,其值能够包括数字、字符串以及文件。我们通过辗转这个对象,来向服务器提交数据。

当然,这个对象我们也可以在代码中手工构建,比如说像下面这样:


var fd = new FormData();
fd.append("author", "Shiv Kumar");
fd.append("name", "Html 5 File API/FormData");
fd.append("fileToUpload", document.getElementById(&#39;fileToUpload&#39;).files[0]);

回到正题。回顾上一段代码,我们增加了许多有关XMLHttpRequest的事件监听,其目的是为了获取文件上传的真实情况。尤其需要注意的是,我们所挂钩的,并不是XMLHttpRequest本身,而是其属性,比如uploadProgress。

完整代码

最后,来看看完整的代码。


 <!DOCTYPE html>
<html>
<head>
    <title>Upload Files using XMLHttpRequest - Minimal</title>

    <script type="text/javascript">
      function fileSelected() {
        var file = document.getElementById(&#39;fileToUpload&#39;).files[0];
        if (file) {
          var fileSize = 0;
          if (file.size > 1024 * 1024)
            fileSize = (Math.round(file.size * 100 / (1024 * 1024)) / 100).toString() + &#39;MB&#39;;
          else
            fileSize = (Math.round(file.size * 100 / 1024) / 100).toString() + &#39;KB&#39;;

          document.getElementById(&#39;fileName&#39;).innerHTML = &#39;Name: &#39; + file.name;
          document.getElementById(&#39;fileSize&#39;).innerHTML = &#39;Size: &#39; + fileSize;
          document.getElementById(&#39;fileType&#39;).innerHTML = &#39;Type: &#39; + file.type;
        }
      }

      function uploadFile() {
        var fd = new FormData();
        fd.append("fileToUpload", document.getElementById(&#39;fileToUpload&#39;).files[0]);
        var xhr = new XMLHttpRequest();
        xhr.upload.addEventListener("progress", uploadProgress, false);
        xhr.addEventListener("load", uploadComplete, false);
        xhr.addEventListener("error", uploadFailed, false);
        xhr.addEventListener("abort", uploadCanceled, false);
        xhr.open("POST", "UploadMinimal.aspx");
        xhr.send(fd);
      }

      function uploadProgress(evt) {
        if (evt.lengthComputable) {
          var percentComplete = Math.round(evt.loaded * 100 / evt.total);
          document.getElementById(&#39;progressNumber&#39;).innerHTML = percentComplete.toString() + &#39;%&#39;;
        }
        else {
          document.getElementById(&#39;progressNumber&#39;).innerHTML = &#39;unable to compute&#39;;
        }
      }

      function uploadComplete(evt) {
        /* This event is raised when the server send back a response */
        alert(evt.target.responseText);
      }

      function uploadFailed(evt) {
        alert("There was an error attempting to upload the file.");
      }

      function uploadCanceled(evt) {
        alert("The upload has been canceled by the user or the browser dropped the connection.");
      }
    </script>
</head>
<body>
<form id="form1" enctype="multipart/form-data" method="post" action="upload.php">
<p class="row">
      <label for="fileToUpload">Select a File to Upload</label>
<input type="file" name="fileToUpload" id="fileToUpload" onchange="fileSelected();"/>
    </p>
<p id="fileName"></p>
<p id="fileSize"></p>
<p id="fileType"></p>
<p class="row">
<input type="button" onclick="uploadFile()" value="Upload" />
    </p>
<p id="progressNumber"></p>
</form>

</body>
</html>

我们的任务完成了吗?可以说完成了,因为这段代码已经能够完成上传文件的任务,而且也能够显示上传的进度;但是理应说我们没有,因为除了这个骨架HTML之外,我们还有很多没有做的事情,比如CSS的美化等等。不过这就不是我们这篇文章的主题了。

最后,提醒一下,教程的代码应当在支持新特性的浏览器之上运行,如果你不清楚自己的浏览器是否支持,可以在这里查询。

相关文章:

移动端通过HTML5实现文件上传功能

php中文件上传图片

JavaScript进阶(九)JS实现本地文件上传至阿里云服务器

以上是HTML5应用之文件上传的详细介绍的详细内容。更多信息请关注PHP中文网其他相关文章!

本站声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热AI工具

Undress AI Tool

Undress AI Tool

免费脱衣服图片

Undresser.AI Undress

Undresser.AI Undress

人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover

AI Clothes Remover

用于从照片中去除衣服的在线人工智能工具。

Clothoff.io

Clothoff.io

AI脱衣机

Video Face Swap

Video Face Swap

使用我们完全免费的人工智能换脸工具轻松在任何视频中换脸!

热工具

记事本++7.3.1

记事本++7.3.1

好用且免费的代码编辑器

SublimeText3汉化版

SublimeText3汉化版

中文版,非常好用

禅工作室 13.0.1

禅工作室 13.0.1

功能强大的PHP集成开发环境

Dreamweaver CS6

Dreamweaver CS6

视觉化网页开发工具

SublimeText3 Mac版

SublimeText3 Mac版

神级代码编辑软件(SublimeText3)

热门话题

Laravel 教程
1602
29
PHP教程
1504
276
将CSS和JavaScript与HTML5结构有效整合。 将CSS和JavaScript与HTML5结构有效整合。 Jul 12, 2025 am 03:01 AM

HTML5、CSS和JavaScript应通过语义化标签、合理加载顺序与解耦设计高效结合。1.使用HTML5语义化标签如、提升结构清晰度与可维护性,利于SEO和无障碍访问;2.CSS应置于中,使用外部文件并按模块拆分,避免内联样式与延迟加载问题;3.JavaScript推荐放在前引入,使用defer或async异步加载以避免阻塞渲染;4.减少三者间强依赖,通过data-*属性驱动行为、类名控制状态,统一命名规范提升协作效率。这些方法能有效优化页面性能与团队协作。

解释html5`  vs` '元素。 解释html5` vs` '元素。 Jul 12, 2025 am 03:09 AM

是块级元素,适合布局;是内联元素,适合包裹文字内容。1.独占一行,可设置宽高和边距,常用于结构布局;2.不换行,大小由内容决定,适用于局部文本样式或动态操作;3.选择时应根据内容是否需独立空间判断;4.不可嵌套在内,不适合做布局;5.优先使用语义化标签以提升结构清晰度与可访问性。

HTML5表单中有哪些新输入类型? HTML5表单中有哪些新输入类型? Jul 12, 2025 am 03:07 AM

HTML5introducednewinputtypesthatenhanceformfunctionalityanduserexperiencebyimprovingvalidation,UI,andmobilekeyboardlayouts.1.emailvalidatesemailaddressesandsupportsmultipleentries.2.urlchecksforvalidwebaddressesandtriggersURL-optimizedkeyboards.3.num

如何使用HTML5地理位置API访问用户的当前位置? 如何使用HTML5地理位置API访问用户的当前位置? Jul 13, 2025 am 02:23 AM

要获取用户当前位置,可使用HTML5的GeolocationAPI。该API在用户授权后提供经纬度等信息,核心方法是getCurrentPosition(),需处理成功与错误回调;同时要注意HTTPS前提、用户授权机制及错误码处理。①调用getCurrentPosition获取一次位置,失败则触发错误回调;②用户必须授权,否则无法获取,且可能不再提示;③错误处理应区分拒绝、超时、位置不可用等情况;④启用高精度、设置超时时间等可通过第三个参数配置;⑤线上环境必须使用HTTPS,否则可能被浏览器限制

说明HTML5中脚本的'异步”和' defer”属性。 说明HTML5中脚本的'异步”和' defer”属性。 Jul 13, 2025 am 03:06 AM

async和defer的区别在于脚本执行时机。async让脚本并行下载且下载完立即执行,不保证执行顺序;defer则在HTML解析完成后按顺序执行脚本。两者都避免阻塞HTML解析。使用async适用于独立脚本如分析代码;defer适合需访问DOM或依赖其他脚本的场景。

如何在HTML5中使用无线电按钮? 如何在HTML5中使用无线电按钮? Jul 21, 2025 am 01:08 AM

在HTML5中使用单选按钮的关键在于理解其工作原理并正确组织代码结构。1.每个radio按钮的name属性必须相同,以实现互斥选择;2.使用label标签提升可访问性和点击体验;3.推荐将每个选项包裹在div或label中以增强结构清晰度和样式控制;4.通过checked属性设置默认选中项;5.value值应简洁有意义,便于表单提交处理;6.可通过CSS自定义样式,但需确保功能正常。掌握这些要点能有效避免常见问题并提升使用效果。

区分HTML5 LocalStorage和SessionStorage 区分HTML5 LocalStorage和SessionStorage Jul 15, 2025 am 03:12 AM

localStorage与sessionStorage的核心区别在于数据持久性和作用域。1.数据生命周期:localStorage数据长期保存,除非手动清除,而sessionStorage数据在关闭标签页后即清除;2.作用域差异:localStorage在同一网站的所有标签页间共享,sessionStorage则每个标签页独立存储;3.使用场景:localStorage适用于保存用户偏好、登录状态等长期数据,sessionStorage适合临时表单数据或单次会话流程;4.API一致:两者操作方法

HTML5中是否仍在使用标签? HTML5中是否仍在使用标签? Jul 21, 2025 am 02:47 AM

是的,是HTML5的一部分,但其使用已逐渐减少且存在争议。用于将主标题与副标题组合在一起,使文档大纲中仅识别最高级别的标题;例如,主标题和副标题可被包裹在中,以表明仅为辅助标题而非独立章节标题;然而,其不再广泛使用的原因包括:1.浏览器和屏幕阅读器对其支持不一致,2.存在更简单的替代方案如使用CSS控制样式,3.HTML文档大纲算法未被广泛支持;尽管如此,在语义要求较高的网站或文档中仍可考虑使用;而大多数情况下,开发者倾向使用单一、通过CSS管理样式并保持清晰的标题层级。

See all articles