+ 큰 파일 자바 업로드

 위조 전에 uploadify 파일 업로드 HTML5 플러그인의 버전을 작성, 친구가 보지 못한 여기를 클릭 ~ 집에서 많은 친구가, 나 또한 프로젝트에 익숙해 보면 사용자 아바타 업로드, 또는 각 여부 업로드 미디어 파일의 종류뿐만 아니라, 개인 비즈니스의 다양한 요구 충족 될 수있다. 작은 행복.

  그러나 상관없이, 모든 요구 사항을 충족하기 어려운 방법 유연한 플러그인 다시 예를 들어, 당신은 2G에 파일을 업로드 할 없습니다. 이제 우리의 네트워크 속도에, 나는 곧 30 분을 통과해야 두렵다. 실수로 브라우저 또는 손으로 가볍게 눌러에게 F5, 완성 된 닫히지 않습니다 시간의 90 %를 업로드하는 경우 문제는, 모든이 모든 것을 다시 시작해야합니다. 이 사용자 경험은 단순히 너무 나쁘다. 그래서, 그것은 매우 필요한 재개. 내가 설명하지 않습니다 이력서, 이렇게 많은 년간 QQ 패스와 파일이 무엇인지, 우리는 그것을 보았다.

  여기에 기술 포인트를 HTTP이있는 것입니다. 제출 또는 HTML5 FormData의 전통적인 형태의 사용은 "블록 EN"파일 파일 서버에 제출하고 전송받을, 다른 작업을 이름을 변경하고, 따라서, 실시간 부분에 업로드 된 파일을 저장할 수 있습니다. 그리고 http 프로토콜에서, 우리는 긴 연결을 브라우저를 유지할 수 있고 서버가 제출 파일 스트림의 형태가 될 수 없습니다. 그래서 질문은 구체적으로 다음과 같은 점을 해결해야합니다 :

  1. 파일 분할을 업로드하려면 시간이 작은 조각을 업로드합니다. 서비스 원래 부품에 추가 서류를 접수 한 후 종료, 그리고 마지막으로 전체 파일에 병합합니다.

  2. 파일을 업로드하는 각각의 조각 파일 크기를 업로드하기 전에, 이것의 위치를 ​​절단해야 결정

  3. 각 업로드가되면 완전한 업데이트 기록은 파일 크기를 업로드되었습니다

  4. 내용이 파일 파일 B에 추가되지 않도록 클라이언트와 서버 파일을 확인

  의 Zhangxin 쑤 형제를 참조에서 이 글 이 기술은 내 플러그인 Huploadify, HTTP 기능의 성공적인 추가로 배울 수 있습니다. 다른 사람에게 기술과 플러그인합니다.

공사 / 기술 포인트

  중요한 것부터 먼저, 우리가 10M 파일이있는 경우, 각 절단 1M를 업로드, 분명히, 우리는 완료하는 데 10 개 요청을 보내야합니다. http 프로토콜에서, 당신은 단지 그렇게 할 수 있습니다. 중단 점 업로드는 세 단계를 완료합니다 :

  1. 파일을 선택한 후에, 서버에 사용자 정의 기능 파일 사이즈를 취득 또는 로컬 저장소에 의해 획득 될 수있다.

  2. 받는 제출 된 요청에 따라 제 n 파일 서버 칩을 유지하기 위해 전송 업로드 된 파일의 크기를 절단, 서버는 파일의 내용을 추가하는 것을 계속

  3. 업로드하면 파일 크기가 끝을 업로드, 총 파일 크기에 도달  

  우선, 파일 분할되고, HTML5는 BLOB 데이터 유형을 첨가하고, 데이터의 방법은 분할 될 수 제공 슬라이스 () 및 그것의 사용 문자열 배열 슬라이스 이진 파일의 일부를 차단할 수 있도록 () 메소드.

  먼저 때마다 파일 file_put_contents를 추가, 특정 문구 나중에 참조하거나 내 파일이 좋은 패키지 다운로드 할 수 file_get_contents와 바이너리 형식 파일을 구 PHP로 작성 내 배경의 추가 조각으로 파일을 저장 하였다.

  다음으로 우리는 또한 다음 컷 업로드하기 전에 해결하기 위해, 업로드 된 파일의 크기를 저장하는 실시간이 필요합니다. HTML5 로컬 스토리지의 사용은 다음 읽기 지역을 시작 업로드하기 전에 로컬로 저장의 크기에 업로드 된 방법이다. 그러나,이 방법은 파일에 사용자 페이지가 페이지 B에서 다른 위치에 파일을 업로드하려고 다음 50 % 업로드하고, 경우에 사용자는 제외하고, 말을하지 않는 로컬 데이터 청지기의 다양한 통해 삭제 설정할 수 있습니다, 매우 제한되어 로컬 파일의 결과는 업로드, 분명히 실수를 시작하는 위치에서 직접 50 %, 51 %의 독서를 업로드했습니다. 문제는 로컬 전용 파일 API, 서버의 올바른 파일을하지 않을 권리 일치를 통해 파일의 원래 이름을 얻기 위해, 너무 많은 정보가 저장되지 않습니다. 그래서 프로젝트와 실제는, 데이터를 저장하는 서버에 의존해야했다.

  서버 측 데이터가있을 것입니다 방법에 대한, 내가 다음에 대해 이야기합니다 프런트 엔드 데이터를 촬영하는 방법이었다.

  많은 위에 기술 점은, 사실, 내 플러그인을 사용하는 방법을보기 위해 많은 기술적 인 내용 하 ~이 없다.

사용 재시작 기능

  파일의 도입은, 하나의 플러그인에 대한 소개를 참조 할 수 있습니다에 대해 이야기하지 않습니다. 핵심은 몇 가지 구성, 첫번째보기를 추가하는 것입니다 :


breakPoints:false,//是否开启断点续传 fileSplitSize:1024*1024,//断点续传的文件块大小,单位Byte,默认1M getUploadedSize:null,//类型:function,自定义获取已上传文件的大小函数,用于开启断点续传模式,可传入一个参数file,即当前上传的文件对象,需返回number类型 saveUploadedSize:null,//类型:function,自定义保存已上传文件的大小函数,用于开启断点续传模式,可传入两个参数:file:当前上传的文件对象,value:已上传文件的大小,单位Byte saveInfoLocal:false,//用于开启断点续传模式,是否使用localStorage存储已上传文件大小

  这是插件中的默认配置值。一个续传功能竟然要配置五个项,真要命!不要着急听我慢慢道来,这五个并不是要同时出现的,是为了满足可能出现的复杂业务而准备的。

  breakPoints是开启断点续传的开关,要使用的话设为true,默认是不开启的。

  fileSplitSize是每次切割的文件片的大小,默认是1M,可根据实际情况来定。如果你的系统上传的文件普遍都在1G以上,可以配置的大一点。

  getUploadedSize是用来自定义获取已上传的文件大小的函数,还记得上面说过的localStorage的局限吧,所以我这里直接把获取文件大小的函数交给你来定义,你可以从session、cookie,从文件、数据库或者任何地方取,可以发送一个ajax请求到你想要的地址,传递你需要的参数。注意你定义的函数将来会被插件调用,所以一定要返回一个Number类型的结果。

  saveUploadedSize与getUploadedSize对应,你自己定义如何保存已上传文件的大小,只要你存的数据你自己能取到就OK。当然前提是你要注意到上面说过的localStorage的局限,确保你的逻辑正确能够操作到正确的文件。

  saveInfoLocal是当你使用localStorage保存数据时需要开启的一个开关。插件默认提供使用localStorage方式的支持。只要开启此选项就可以了。当然,这种情况下你的业务逻辑必须足够简单,比如只是做一个上传的demo,或者这系统的用户只有你一个人,你明白如何避开那些局限的地方。

  掌握了这五个配置的作用,你就可以实现一个足够灵活的断点上传功能了!在我打包好的文件里,提供了使用localStorage方式的demo,抱歉我无法将数据库表都发给你,所以只能用本地存储来演示。

在服务端保存数据

  用户在使用上传的时候可能有各种你意想不到的操作,这里我发挥想象描述一下用户可能的行为:

  • 同一台机器使用不同帐号登录,上传同一个文件

  • 文件上传了一部分,然后修改了文件内容,再次上传

  • 文件上传完成100%,再次上传该文件

  • 同一个页面有多个上传按钮,上传同一个文件,或在不同页面上传同一个文件

  仅仅上面四条,是不是情况就够复杂了?再加上你系统还有自己的业务逻辑,所以在服务端保存已上传文件数据是非常有必要的。而且保存数据和获取数据的函数都交给你来定义,抱着插件有足够的灵活性。

  因为涉及到了服务端的技术,无法演示,我将我项目中的真实使用场景在此讲解一下,来展示一下如何自已定义方法来实现服务端保存数据的可靠上传。我定义的getUploadedSize函数如下:

getUploadedSize:function(file){
            var data = {
                data : {
                    fileName : file.name,
                    lastModifiedDate : file.lastModifiedDate.getTime()
                }
            };
            var url = 'http://localhost/uploadfile/';
            var uploadedSize = 0;
            $.ajax({
                url : url,
                data : data,
                async : false,
                type : 'POST',
                success : function(returnData){
                    returnData = JSON.parse(returnData);
                    uploadedSize = returnData.uploadedSize;
                }
            });
            return uploadedSize;
        }

 

  我向后台的某个地址发送一个请求,传递文件名和文件的最后修改时间为参数,后台根据这两个参数来找到与前台所选择的文件对应的服务器上的文件,将服务器返回的文件大小return出去,来被插件使用。为什么要传递这两个参数呢?我们在前台无法知道服务器上的这个文件的名称,所以使用原始文件名作为一个辅助标识。为了防止用户在两次上传间隔修改了文件,我们把文件的最后修改时间也传给服务端,让服务端进行比较,若时间不对应则返回已上传大小为0,重新上传此文件。

  再来看后台都要做哪些工作。数据库中需要有一张表来记录每个已文件的情况,包含的字段大致有:

字段 描述
client_filename 文件在客户端的原始名称
server_filename 文件在服务器上重命名后的名称
last_modified_date 文件的最后修改时间,时间戳
status 文件的状态,已完成、未完成
uploaded_size 已上传文件的大小

  根据client_filename和last_modified_date,再加上系统中的其他关联信息,可以定位到本次上传的文件在服务端的大小,然后返回给客户端。当然这是我自己的用法,你也可以根据自己的需求灵活设计。总之最终的目的就是要找到前台选择的文件在服务器上真正对应的文件,并将已上传大小正确返回。

  另外需注意的一点,就是在续传的第二步,不断提交文件片的过程中,也需要服务端准确定位到相应的文件,不能把A的数据追加到B上。采用的方式也是提交fileName和lastModifyDate两个参数(已写在插件内部,可服务端直接获取),服务端找到对应的文件进行追加。

  另外再啰嗦一句,后台获取文件的时候需要取成二进制的,而我们提交是使用FormData来提交的,所以PHP代码需要这么写:

file_put_contents('uploads/'.$filename,file_get_contents($_FILES["file"]["tmp_name"]),FILE_APPEND);

  如果上面的说明还是不够清楚,就需要你自己来探索一下了,毕竟考虑到插件可能应用在复杂的系统中,很多工作还是需要你来做的。或者你也可以给我留言,我很乐意为你解答疑惑。

该版本的其他改动

  从1.0到2.0,Huploadify又新加了很多东西,不过只是新加,使用方式跟之前的没有变化。例如上面的断点续传功能,你如果不想使用,只需设置breakPoints为false即可,插件仍按照以前的方式工作。除了断点续传这个大头,插件还做了如下改动:

  1. 增加了onSelect回调函数,在选择了文件之后触发,用法与uploadify官网的一致

  2. 删除掉正在上传的文件,中断发送请求

  3. 完善了input file组件的accept属性支持,浏览时只显示运行的文件格式,就是这个东东:

  4. 对外开放了方法调用接口,upload、stop、cancel、disable、ennable。我在demo中有演示。使用方法如下:var up = $('#upload').Huploadify({

    auto:false,     fileTypeExts:'*.jpg;*.png;*.exe;*.mp3;*.mp4;*.zip;*.doc;*.docx;*.ppt;*.pptx;*.xls;*.xlsx;*.pdf',     multi:true });  up.upload(1);//开始上传文件,接收一个参数,表示上传第几个文件,可传入*上传队列中的所有文件 up.stop();//暂停上传队列中的所有文件,不接收参数。用于开启了断点需传 up.cancel(1);//删除队列中的某个文件,接收一个参数,表示删除第几个文件,可传入*删除队列中的所有文件 up.disable();//使选择文件按钮失效,不接收参数 up.ennable();//使选择文件按钮生效,不接收参数  5. 修改其他已知bug

结束

  插件刚刚完成,与我们的后端程序员调试完成了断点续传功能暂未发现问题,欢迎大家在使用的时候给我提任何问题。老实来讲这个功能使用起来还是挺费解的,为了最大程度的保证灵活做成这样,大家可以与我多多交流~

  我在demo中使用了本地存储来做已上传文件大小的保存,下载压缩包后可看一下效果。上传一个比较大的视频文件,上传到中间关闭浏览器,再次打开浏览器上传同一个文件,会看到从上次断掉的地方继续上传。
详细内容可以参考我写的这篇文章:http://blog.ncmem.com/wordpress/2019/08/09/%e5%a4%a7%e6%96%87%e4%bb%b6%e4%b8%8a%e4%bc%a0%e8%a7%a3%e5%86%b3%e6%96%b9%e6%a1%88/

추천

출처www.cnblogs.com/songsu/p/11350471.html