当前位置:   article > 正文

asp.net文件分块上传_asp网站实现分块传输

asp网站实现分块传输

一般10M以下的文件上传通过设置Web.Config,再用VS自带的FileUpload控件就可以了,但是如果要上传100M甚至1G的文件就不能这样上传了。我这里分享一下我自己开发的一套大文件上传控件供大家参考

控件功能:

1. 文件批量上传

此文件管理器支持文件批量上传。您可以上传30G及以上的大型文件,在上传过程中您不需要担心刷新网页造成的进度丢失问题。也不需要担心浏览器重启或崩溃,电脑重启等极端应用场景造成的进度丢失问题。文件管理器能够自动定时保存文件上传进度。以便为您提供更好的用户体验。

 

2. 文件夹批量上传

此文件管理器提供文件夹的批量上传功能。您可以同时上传一个或者多个文件夹,文件管理器会将这些文件夹以及他们的结构信息同时保存在服务器中。您不需要担心浏览器重启或崩溃造成的进度丢失的问题。

 

3. 文件批量下载

此文件管理器提供了文件批量下载功能。您现在可以同时下载多个文件,并将他们保存在本地指定的目录中。这一功能为图库应用场景,多资料共享应用场景提供了使用便利。

 

4. 文件夹批量下载

此文件管理器提供了文件夹的批量下载功能。您可以同时下载多个文件夹,这些文件夹下载完毕后,他们的层级信息也将会同时在本地保留。

 

5. 新建目录

此文件管理器提供了多层级目录管理功能。您现在可以根据需求新建目录。

 

6. 文件目录重命名。

7. 树型目录导航

8. 路径导航

9. 开源

此文件管理器是开款开源产品,无论您是个人还是企业都可以获取他的源代码来进行二次开发。我们为ASP.NET,JAVA,PHP等语言提供了示例,您可以同时选择这3种开发语言来进行项目开发。同时我们将提供长期的更新和维护服务,帮助企业节省后期的产品维护成本。

 

这是前端部分脚本代码:

  1. //文件上传对象
  2. function FileUploader(fileLoc, mgr)
  3. {
  4.     var _this = this;
  5.     this.id = fileLoc.id;
  6.     this.ui = { msg: null, process: null, percent: null, btn: { del: null, cancel: null,post:null,stop:null }, div: null};
  7.     this.isFolder = false; //不是文件夹
  8.     this.app = mgr.app;
  9.     this.Manager = mgr; //上传管理器指针
  10.     this.event = mgr.event;
  11.     this.FileListMgr = mgr.FileListMgr;//文件列表管理器
  12.     this.Config = mgr.Config;
  13.     this.fields = jQuery.extend({}, mgr.Config.Fields, fileLoc.fields);//每一个对象自带一个fields幅本
  14.     this.State = this.Config.state.None;
  15.     this.uid = this.fields.uid;
  16.     this.fileSvr = {
  17.           pid: ""
  18.         , id: ""
  19.         , pidRoot: ""
  20.         , f_fdTask: false
  21.         , f_fdChild: false
  22.         , uid: 0
  23.         , nameLoc: ""
  24.         , nameSvr: ""
  25.         , pathLoc: ""
  26.         , pathSvr: ""
  27.         , pathRel: ""
  28.         , md5: ""
  29.         , lenLoc: "0"
  30.         , sizeLoc: ""
  31.         , FilePos: "0"
  32.         , lenSvr: "0"
  33.         , perSvr: "0%"
  34.         , complete: false
  35.         , deleted: false
  36.     };//json obj,服务器文件信息
  37.     this.fileSvr = jQuery.extend(this.fileSvr, fileLoc);
  38.     //准备
  39.     this.Ready = function ()
  40.     {
  41.         this.ui.msg.text("正在上传队列中等待...");
  42.         this.State = this.Config.state.Ready;
  43.     };
  44.     this.svr_error = function ()
  45.     {
  46.         alert("服务器返回信息为空,请检查服务器配置");
  47.         this.ui.msg.text("向服务器发送MD5信息错误");
  48.         this.ui.btn.cancel.text("续传");
  49.     };
  50.     this.svr_create = function (sv)
  51.     {
  52.         if (sv.value == null)
  53.         {
  54.             this.svr_error(); return;
  55.         }
  56.         var str = decodeURIComponent(sv.value);//
  57.         this.fileSvr = JSON.parse(str);//
  58.         //服务器已存在相同文件,且已上传完成
  59.         if (this.fileSvr.complete)
  60.         {
  61.             this.post_complete_quick();
  62.         } //服务器文件没有上传完成
  63.         else
  64.         {
  65.             this.ui.process.css("width", this.fileSvr.perSvr);
  66.             this.ui.percent.text(this.fileSvr.perSvr);
  67.             this.post_file();
  68.         }
  69.     };
  70.     this.svr_update = function () {
  71.         if (this.fileSvr.lenSvr == 0) return;
  72.         var param = { uid: this.fields["uid"], offset: this.fileSvr.lenSvr, lenSvr: this.fileSvr.lenSvr, perSvr: this.fileSvr.perSvr, id: this.id, time: new Date().getTime() };
  73.         $.ajax({
  74.             type: "GET"
  75.             , dataType: 'jsonp'
  76.             , jsonp: "callback" //自定义的jsonp回调函数名称,默认为jQuery自动生成的随机函数名
  77.             , url: this.Config["UrlProcess"]
  78.             , data: param
  79.             , success: function (msg) {}
  80.             , error: function (req, txt, err) { alert("更新文件进度错误!" + req.responseText); }
  81.             , complete: function (req, sta) { req = null; }
  82.         });
  83.     };
  84.     this.post_process = function (json)
  85.     {
  86.         this.fileSvr.lenSvr = json.lenSvr;//保存上传进度
  87.         this.fileSvr.perSvr = json.percent;
  88.         this.ui.percent.text("("+json.percent+")");
  89.         this.ui.process.css("width", json.percent);
  90.         var str = json.lenPost + " " + json.speed + " " + json.time;
  91.         this.ui.msg.text(str);
  92.     };
  93.     this.post_complete = function (json)
  94.     {
  95.         this.fileSvr.perSvr = "100%";
  96.         this.fileSvr.complete = true;
  97.         $.each(this.ui.btn, function (i, n)
  98.         {
  99.             n.hide();
  100.         });
  101.         this.ui.process.css("width", "100%");
  102.         this.ui.percent.text("(100%)");
  103.         this.ui.msg.text("上传完成");
  104.         this.Manager.arrFilesComplete.push(this);
  105.         this.State = this.Config.state.Complete;
  106.         //从上传列表中删除
  107.         this.Manager.RemoveQueuePost(this.fileSvr.id);
  108.         //从未上传列表中删除
  109.         this.Manager.RemoveQueueWait(this.fileSvr.id);
  110.         var param = { md5: this.fileSvr.md5, uid: this.uid, id: this.fileSvr.id, time: new Date().getTime() };
  111.         $.ajax({
  112.             type: "GET"
  113.              , dataType: 'jsonp'
  114.              , jsonp: "callback" //自定义的jsonp回调函数名称,默认为jQuery自动生成的随机函数名
  115.              , url: _this.Config["UrlComplete"]
  116.              , data: param
  117.              , success: function (msg)
  118.              {
  119.                  _this.event.fileComplete(_this);//触发事件
  120.                  _this.FileListMgr.UploadComplete(_this.fileSvr);//添加到服务器文件列表
  121.                  _this.post_next();
  122.              }
  123.              , error: function (req, txt, err) { alert("文件-向服务器发送Complete信息错误!" + req.responseText); }
  124.              , complete: function (req, sta) { req = null; }
  125.         });
  126.     };
  127.     this.post_complete_quick = function ()
  128.     {
  129.         this.fileSvr.perSvr = "100%";
  130.         this.fileSvr.complete = true;
  131.         this.ui.btn.stop.hide();
  132.         this.ui.process.css("width", "100%");
  133.         this.ui.percent.text("(100%)");
  134.         this.ui.msg.text("服务器存在相同文件,快速上传成功。");
  135.         this.Manager.arrFilesComplete.push(this);
  136.         this.State = this.Config.state.Complete;
  137.         //从上传列表中删除
  138.         this.Manager.RemoveQueuePost(this.fileSvr.id);
  139.         //从未上传列表中删除
  140.         this.Manager.RemoveQueueWait(this.fileSvr.id);
  141.         //添加到文件列表
  142.         this.FileListMgr.UploadComplete(this.fileSvr);
  143.         this.post_next();
  144.         this.event.fileComplete(this);//触发事件
  145.     };
  146.     this.post_stoped = function (json)
  147.     {
  148.         this.ui.btn.post.show();
  149.         this.ui.btn.del.show();
  150.         this.ui.btn.cancel.hide();
  151.         this.ui.btn.stop.hide();
  152.         this.ui.msg.text("传输已停止....");
  153.         if (this.Config.state.Ready == this.State)
  154.         {
  155.             this.Manager.RemoveQueue(this.fileSvr.id);
  156.             this.post_next();
  157.             return;
  158.         }
  159.         this.State = this.Config.state.Stop;
  160.         //从上传列表中删除
  161.         this.Manager.RemoveQueuePost(this.fileSvr.id);
  162.         this.Manager.AppendQueueWait(this.fileSvr.id);//添加到未上传列表
  163.         //传输下一个
  164.         this.post_next();
  165.     };
  166.     this.post_error = function (json)
  167.     {
  168.         this.svr_update();
  169.         this.ui.msg.text(this.Config.errCode[json.value]);
  170.         this.ui.btn.stop.hide();
  171.         this.ui.btn.post.show();
  172.         this.ui.btn.del.show();
  173.         this.State = this.Config.state.Error;
  174.         //从上传列表中删除
  175.         this.Manager.RemoveQueuePost(this.fileSvr.id);
  176.         //添加到未上传列表
  177.         this.Manager.AppendQueueWait(this.fileSvr.id);
  178.         this.post_next();
  179.     };
  180.     this.md5_process = function (json)
  181.     {
  182.         var msg = "正在扫描本地文件,已完成:" + json.percent;
  183.         this.ui.msg.text(msg);
  184.     };
  185.     this.md5_complete = function (json)
  186.     {
  187.         this.fileSvr.md5 = json.md5;
  188.         this.ui.msg.text("MD5计算完毕,开始连接服务器...");
  189.         this.event.md5Complete(this, json.md5);//biz event
  190.         var loc_path = encodeURIComponent(this.fileSvr.pathLoc);
  191.         var loc_len = this.fileSvr.lenLoc;
  192.         var loc_size = this.fileSvr.sizeLoc;
  193.         var param = jQuery.extend({}, this.fields, { md5: json.md5, id: this.fileSvr.id, lenLoc: loc_len, sizeLoc: loc_size, pathLoc: loc_path, time: new Date().getTime() });
  194.         $.ajax({
  195.             type: "GET"
  196.             , dataType: 'jsonp'
  197.             , jsonp: "callback" //自定义的jsonp回调函数名称,默认为jQuery自动生成的随机函数名
  198.             , url: this.Config["UrlCreate"]
  199.             , data: param
  200.             , success: function (sv)
  201.             {
  202.                 _this.svr_create(sv);
  203.             }
  204.             , error: function (req, txt, err)
  205.             {
  206.                 alert("向服务器发送MD5信息错误!" + req.responseText);
  207.                 _this.ui.msg.text("向服务器发送MD5信息错误");
  208.                 _this.ui.btn.del.text("续传");
  209.             }
  210.             , complete: function (req, sta) { req = null; }
  211.         });
  212.     };
  213.     this.md5_error = function (json)
  214.     {
  215.         this.ui.msg.text(this.Config.errCode[json.value]);
  216.         //文件大小超过限制,文件大小为0
  217.         if ("4" == json.value
  218.              || "5" == json.value)
  219.         {
  220.         this.ui.btn.stop.hide();
  221.         this.ui.btn.cancel.show();
  222.         }
  223.         else
  224.         {           
  225.             this.ui.btn.post.show();
  226.             this.ui.btn.stop.hide();
  227.         }
  228.         this.State = this.Config.state.Error;
  229.         //从上传列表中删除
  230.         this.Manager.RemoveQueuePost(this.fileSvr.id);
  231.         //添加到未上传列表
  232.         this.Manager.AppendQueueWait(this.fileSvr.id);
  233.         this.post_next();
  234.     };
  235.     this.post_next = function ()
  236.     {
  237.         var obj = this;
  238.         setTimeout(function () { obj.Manager.PostNext(); }, 500);
  239.     };
  240.     this.post = function ()
  241.     {
  242.         this.Manager.AppendQueuePost(this.fileSvr.id);
  243.         this.Manager.RemoveQueueWait(this.fileSvr.id);
  244.         if (this.fileSvr.md5.length > 0)
  245.         {
  246.             this.post_file();
  247.         }
  248.         else
  249.         {
  250.             this.check_file();
  251.         }
  252.     };
  253.     this.post_file = function ()
  254.     {
  255.         this.ui.btn.cancel.hide();
  256.         this.ui.btn.stop.show();
  257.         this.State = this.Config.state.Posting;//
  258.         this.app.postFile({ id: this.fileSvr.id, pathLoc: this.fileSvr.pathLoc, pathSvr:this.fileSvr.pathSvr,lenSvr: this.fileSvr.lenSvr, fields: this.fields });
  259.     };
  260.     this.check_file = function ()
  261.     {
  262.         //this.ui.btn.cancel.text("停止").show();
  263.         this.ui.btn.stop.show();
  264.         this.ui.btn.cancel.hide();
  265.         this.State = this.Config.state.MD5Working;
  266.         this.app.checkFile({ id: this.fileSvr.id, pathLoc: this.fileSvr.pathLoc });
  267.     };
  268.     this.stop = function ()
  269.     {
  270.         this.ui.btn.del.hide();
  271.         this.ui.btn.cancel.hide();
  272.         this.ui.btn.stop.hide();
  273.         this.ui.btn.post.hide();
  274.         this.svr_update();
  275.         this.app.stopFile({ id: this.fileSvr.id });       
  276.     };
  277.     //手动停止,一般在StopAll中调用
  278.     this.stop_manual = function ()
  279.     {
  280.         if (this.Config.state.Posting == this.State)
  281.         {
  282.             this.svr_update();
  283.         this.ui.btn.post.show();
  284.         this.ui.btn.stop.hide();
  285.         this.ui.btn.cancel.hide();
  286.             this.ui.msg.text("传输已停止....");
  287.             this.app.stopFile({ id: this.fileSvr.id ,tip:false});
  288.             this.State = this.Config.state.Stop;
  289.         }
  290.     };
  291.     //删除,一般在用户点击"删除"按钮时调用
  292.     this.remove = function ()
  293.     {
  294.         this.Manager.del_file(this.fileSvr.id);
  295.         this.app.delFile(this.fileSvr);
  296.         this.ui.div.remove();
  297.     };
  298. }

这是后台文件块处理代码和截图:

文件上传完毕,f_complete.

文件初始化,f_create

文件块处理,f_post

文件夹上传完毕,fd_complete

文件夹初始化,fd_create

  1. using Newtonsoft.Json;
  2. using Newtonsoft.Json.Linq;
  3. using System;
  4. using System.Web;
  5. using up6.db.biz;
  6. using up6.db.utils;
  7. namespace up6.db
  8. {
  9.     public partial class f_post : System.Web.UI.Page
  10.     {
  11.         bool safe_check(params string[] ps)
  12.         {
  13.             foreach (var v in ps)
  14.             {
  15.                 System.Diagnostics.Debug.Write("参数值:");
  16.                 System.Diagnostics.Debug.WriteLine(v);
  17.                 if (string.IsNullOrEmpty(v)) return false;
  18.             }
  19.             foreach (string key in Request.Headers.Keys)
  20.             {
  21.                 var vs = Request.Headers.GetValues(key);
  22.                 //XDebug.Output(key + " "+String.Join(",", vs));
  23.             }
  24.             return true;
  25.         }
  26.         /// <summary>
  27.         /// 只负责拼接文件块。将接收的文件块数据写入到文件中。
  28.         /// 更新记录:
  29.         ///       2012-04-12 更新文件大小变量类型,增加对2G以上文件的支持。
  30.         ///       2012-04-18 取消更新文件上传进度信息逻辑。
  31.         ///       2012-10-30 增加更新文件进度功能。
  32.         ///       2015-03-19 文件路径由客户端提供,此页面不再查询文件在服务端的路径。减少一次数据库访问操作。
  33.         ///     2016-03-31 增加文件夹信息字段
  34.         ///     2017-07-11 优化参数检查逻辑
  35.         /// </summary>
  36.         /// <param name="sender"></param>
  37.         /// <param name="e"></param>
  38.         protected void Page_Load(object sender, EventArgs e)
  39.         {
  40.             string uid          = Request.Headers["uid"];
  41.             string f_id         = Request.Headers["id"];
  42.             string lenSvr       = Request.Headers["lenSvr"];//已传大小
  43.             string lenLoc       = Request.Headers["lenLoc"];//本地文件大小
  44.             string blockOffset  = Request.Headers["blockOffset"];
  45.             string blockSize    = Request.Headers["blockSize"];//当前块大小
  46.             string blockIndex   = Request.Headers["blockIndex"];//当前块索引,基于1
  47.             string blockMd5     = Request.Headers["blockMd5"];//块MD5
  48.             string complete     = Request.Headers["complete"];//true/false
  49.             string pathSvr      = Request.Form["pathSvr"];//
  50.             pathSvr             = HttpUtility.UrlDecode(pathSvr);
  51.             if( !this.safe_check(lenLoc,uid,f_id,blockOffset,pathSvr)) return;
  52.             //有文件块数据
  53.             if (Request.Files.Count > 0)
  54.             {
  55.                 bool verify = false;
  56.                 string msg = string.Empty;
  57.                 string md5Svr = string.Empty;
  58.                 HttpPostedFile file = Request.Files.Get(0);//文件块
  59.                 //计算文件块MD5
  60.                 if (!string.IsNullOrEmpty(blockMd5))
  61.                 {
  62.                     md5Svr = Md5Tool.calc(file.InputStream);
  63.                 }
  64.                 //文件块大小验证
  65.                 verify = int.Parse(blockSize) == file.InputStream.Length;
  66.                 if (!verify)
  67.                 {
  68.                     msg = "block size error sizeSvr:"+file.InputStream.Length + " sizeLoc:"+blockSize;
  69.                 }
  70.                 //块MD5验证
  71.                 if ( verify && !string.IsNullOrEmpty(blockMd5) )
  72.                 {
  73.                     verify = md5Svr == blockMd5;
  74.                     if(!verify) msg = "block md5 error";
  75.                 }
  76.                 if (verify)
  77.                 {
  78.                     //2.0保存文件块数据
  79.                     FileBlockWriter res = new FileBlockWriter();
  80.                     res.make(pathSvr, Convert.ToInt64(lenLoc));
  81.                     res.write(pathSvr, Convert.ToInt64(blockOffset), ref file);
  82.                     up6_biz_event.file_post_block(f_id,Convert.ToInt32(blockIndex));
  83.                     //生成信息
  84.                     JObject o = new JObject();
  85.                     o["msg"] = "ok";
  86.                     o["md5"] = md5Svr;//文件块MD5
  87.                     o["offset"] = blockOffset;//偏移
  88.                     msg = JsonConvert.SerializeObject(o);
  89.                 }
  90.                 Response.Write(msg);
  91.             }
  92.         }
  93.     }
  94. }

新建文件夹

点击新建文件夹按钮,弹出此窗口,填写新建文件夹名称后点击确定

页面左上角出现刚刚新建的文件夹名称

粘贴上传

复制文件夹、文件或图片

在页面中选择好相应的上传目录,点击粘贴上传按钮,数据即可快速开始上传

 

文件和文件夹批量上传上传功能,在新建文件夹目录内上传文件,选择多个文件或文件夹上传

如果上传的是文件夹,那么左侧的文件夹内会自动添加一个子文件夹,与上传的文件夹相符并可以展开查看文件夹内文件

在哪个目录下上传文件,文件就会存储在哪个目录下

 

点击根目录按钮可以返回根目录

当网络问题导致传输错误时,只需要重传出错分片,而不是整个文件。另外分片传输能够更加实时的跟踪上传进度。

 

上传成功后打开我们的存储文件夹查看,发现自动生成了几个文件夹,打开文件夹确认上传文件成功

 

点击文件夹后的重命名按钮

修改文件名后点击确定

页面左侧文件夹与页面中间的文件夹名称同时改变

 

点击删除按钮

点击确定后,页面中的文件消失

 

文件及文件夹批量下载

这是下载所需的脚本截图和部分代码:

  1. using System;
  2. using System.IO;
  3. using System.Web;
  4. namespace up6.down2.db
  5. {
  6.     public partial class f_down : System.Web.UI.Page
  7.     {
  8.         bool check_params(params string[] vs)
  9.         {
  10.             foreach(string v in vs)
  11.             {
  12.                 if (String.IsNullOrEmpty(v)) return false;
  13.             }
  14.             return true;
  15.         }
  16.         protected void Page_Load(object sender, EventArgs e)
  17.         {
  18.             string id           = Request.Headers["id"];//文件id
  19.             string blockIndex   = Request.Headers["blockIndex"];//基于1
  20.             string blockOffset  = Request.Headers["blockOffset"];//块偏移,相对于整个文件
  21.             string blockSize    = Request.Headers["blockSize"];//块大小(当前需要下载的)
  22.             string pathSvr      = Request.Headers["pathSvr"];//文件在服务器的位置
  23.             pathSvr             = HttpUtility.UrlDecode(pathSvr);
  24.             if ( !this.check_params(id,blockIndex,blockOffset,pathSvr))
  25.             {
  26.                 Response.StatusCode = 500;
  27.                 return;
  28.             }
  29.             Stream iStream = null;
  30.             try
  31.             {
  32.                 // Open the file.
  33.                 iStream = new FileStream(pathSvr, FileMode.Open, FileAccess.Read, FileShare.Read);
  34.                 iStream.Seek(long.Parse(blockOffset),SeekOrigin.Begin);//定位
  35.                 // Total bytes to read:
  36.                 long dataToRead = long.Parse(blockSize);
  37.                 Response.ContentType = "application/octet-stream";
  38.                 Response.AddHeader("Content-Length", blockSize );
  39.                 int buf_size = Math.Min(1048576, int.Parse(blockSize));
  40.                 byte[] buffer = new Byte[ buf_size];
  41.                 int length;
  42.                 while (dataToRead > 0)
  43.                 {
  44.                     // Verify that the client is connected.
  45.                     if (Response.IsClientConnected)
  46.                     {
  47.                         // Read the data in buffer.
  48.                         length = iStream.Read(buffer, 0, buf_size);
  49.                         dataToRead -= length;
  50.                         // Write the data to the current output stream.
  51.                         Response.OutputStream.Write(buffer, 0, length);
  52.                         // Flush the data to the HTML output.
  53.                         Response.Flush();
  54.                     }
  55.                     else
  56.                     {
  57.                         //prevent infinite loop if user disconnects
  58.                         dataToRead = -1;
  59.                     }
  60.                 }
  61.             }
  62.             catch (Exception ex)
  63.             {
  64.                 Response.StatusCode = 500;
  65.                 // Trap the error, if any.
  66.                 Response.Write("Error : " + ex.Message);
  67.             }
  68.             finally
  69.             {
  70.                 if (iStream != null)
  71.                 {
  72.                     //Close the file.
  73.                     iStream.Close();
  74.                 }
  75.             }
  76.         }
  77.     }
  78. }

选择上传后的文件夹内的子目录文件或文件夹下载

 

然后点击下载按钮,设置下载目录文件夹

设置完成后继续点击下载按钮,页面的右下角出现了下载面板,你选择的文件已出现在目录中,然后点击全部下载,或者单个点击继续,自动加载未上传完的任务。在刷新浏览器或重启电脑后任然可以自动加载未完成的任务

 

下载完成后打开我们设置的下载目录文件夹,发现需下载的文件或文件夹确认已下载成功,经确认文件夹内的内容与下载文件夹内容一致

 

数据库记录,支持SQL、MySQL、Oracle

 

控件包下载:

cab(x86):http://t.cn/Ai9pmG8S

cab(x64): http://t.cn/Ai9pm04B

xpi: http://t.cn/Ai9pubUc

crx:http://t.cn/Ai9pmrcy

exe:http://t.cn/Ai9puobe

 

示例下载:

asp.net: http://t.cn/Ai9pue4A

jsp-eclipse: http://t.cn/Ai9p3LSx

jsp-myeclipse:http://t.cn/Ai9p3IdC

php:http://t.cn/Ai9p3CKQ

 

在线文档:

asp.net-测试教程:http://t.cn/Ai9pBM3A

jsp测试教程:http://t.cn/Ai9pB82o

php-测试教程:http://t.cn/Ai9prHik

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/很楠不爱3/article/detail/499438
推荐阅读
相关标签
  

闽ICP备14008679号