专栏首页Danny的专栏ASP.NET实现文件的上传和下载

ASP.NET实现文件的上传和下载

版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/huyuyang6688/article/details/40785429

       最近做的一个高校网站中涉及到了上传和下载文件的需求(具体需求为:网站公布的通知,在后台要能给每个通知添加附件,在前台要能显示并下载附件),之前只是学习过关于上传的 理论知识,这里实践了一下下,与大家分享一下成果。

       事先说明:这个例子采用的是简单的三层结构,层与层之间是用实体来传值。而且这种方法不但在本地测试时可以成功,并且可以部署在服务器上,供异地上传和下载文件。

       专门做了一个数据库表用来存储附件的相关信息:

字段

说明

AnnexID

附件ID

AnnexName

附件名称

AnnexAddress

存储附件的地址

NoticeID

附件所属“通知”的ID

       ASP.NET实现上传文件

前端

       界面十分简单,只是放一个file类型的<input>和一个按钮,并且为这个按钮添加点击事件(btnUpLoad_Click),如下图:

       代码:

    <input id="UpLoad" type="file" runat="server" />
    <asp:Button runat="server" Text="上传" ID="btnUpLoad" OnClick="btnUpLoad_Click" />

后台

       再就是在后台编写上传按钮点击事件UpLoad_Click里的代码,先大体说一下思路:

       1、根据file类型的<input>控件获得将要上传文件在本机的物理路径;

       2、在这个物理路径中用截取字符串的方法获得文件名(第一步中取得的路径为本机的绝对路径,在服务器上是无效的,所以这里我们只需要获取文件名);

       3、利用file类型的<input>控件属性PostedFile的SaveAs()方法将相应文件存储到服务器中指定的文件夹中。

       核心代码:

    protected void btnUpLoad_Click(object sender, EventArgs e)
        {
            //取出所选文件的本地路径
            string fullFileName = this.UpLoad.PostedFile.FileName;
            //从路径中截取出文件名
            string fileName = fullFileName.Substring(fullFileName.LastIndexOf("\\") + 1);
            //限定上传文件的格式
            string type = fullFileName.Substring(fullFileName.LastIndexOf(".") + 1);
            if (type == "doc" || type == "docx" || type == "xls" || type == "xlsx" || type == "ppt" || type == "pptx" || type == "pdf" || type == "jpg" || type == "bmp" || type == "gif" || type == "png" || type == "txt" || type == "zip" || type == "rar")
            {
                //将文件保存在服务器中根目录下的files文件夹中
                string saveFileName = Server.MapPath("/files") + "\\" + fileName;
                UpLoad.PostedFile.SaveAs(saveFileName);
                Page.ClientScript.RegisterStartupScript(Page.GetType(), "message", "<script language='javascript' defer>alert('文件上传成功!');</script>");

                //向数据库中存储相应通知的附件的目录
                BLL.news.InsertAnnexBLL insertAnnex = new BLL.news.InsertAnnexBLL();
                AnnexEntity annex=new AnnexEntity();     //创建附件的实体
                annex.AnnexName=fileName;               //附件名
                annex.AnnexContent=saveFileName;        //附件的存储路径
                annex.NoticeId = noticeId;              //附件所属“通知”的ID在这里为已知
                insertAnnex.InsertAnnex(annex);         //将实体存入数据库(其实就是讲实体的这些属性insert到数据库中的过程,具体BLL层和DAL层的代码这里不再多说)
            }
            else
            {
                Page.ClientScript.RegisterStartupScript(Page.GetType(), "message", "<script language='javascript' defer>alert('请选择正确的格式');</script>");
            }
        }

ASP.NET实现下载文件

       上述操作已经可以实现将一个个附件存入数据库,在数据库中存储的情况给大家截了个图:

       下面就要把这些附件在页面上显示,页面显示效果为:

       点击附件,浏览器提示下载:

前台:       

       按照需求来说,每则发布的通知可以包含若干个附件,所一前台用了repeter控件来显示多个附件:    

       代码:

    <asp:Repeater ID="rptAnnex" runat="server">
         <ItemTemplate>
             <%--为repeter添加序号--%>
             附件:<%#Container.ItemIndex + 1 %>       
             <asp:LinkButton ID="lbtnDownLoad" runat="server" OnCommand="lbtnDownLoad_Command" CommandArgument="<%#((Model.AnnexEntity)Container.DataItem).AnnexContent %>"><%#((Model.AnnexEntity)Container.DataItem).AnnexName %></asp:LinkButton>
             <br />
         </ItemTemplate>
     </asp:Repeater>

后台

       ASP.NET可以采用多种方式下载文件(详情可参考《ASP.NET下载文件的几种方式》),这里采用了流式的下载方式(参考文章《Asp.net下载实例》):

 using System.IO;
    protected void lbtnDownLoad_Command(object sender, CommandEventArgs e)
        {
            // 定义文件名  
            string fileName = "";
            // 获取文件在服务器的地址  
            string url = e.CommandArgument.ToString();

            // 判断传输地址是否为空  
            if (url == "")
            {
                // 提示“该文件暂不提供下载”  
                Page.ClientScript.RegisterStartupScript(Page.GetType(), "message", "<script defer>alert('该文件暂不提供下载!');</script>");
                return;
            }
            // 判断获取的是否为地址,而非文件名  
            if (url.IndexOf("\\") > -1)
            {
                // 获取文件名  
                fileName = url.Substring(url.LastIndexOf("\\") + 1);
            }
            else
            {
                // url为文件名时,直接获取文件名  
                fileName = url;
            }
            // 以字符流的方式下载文件  
            FileStream fileStream = new FileStream(@url, FileMode.Open);
            byte[] bytes = new byte[(int)fileStream.Length];
            fileStream.Read(bytes, 0, bytes.Length);
            fileStream.Close();
            Response.ContentType = "application/octet-stream";

            // 通知浏览器下载 
            Response.AddHeader("Content-Disposition", "attachment; filename=" + fileName);
            Response.BinaryWrite(bytes);
            Response.Flush();
            Response.End();     
        }

控制上传文件的大小     

       前面的两个步骤基本上已经可以实现文件的上传和下载,除了这些,还需要控制上传文件的大小,默认情况下上传文件大小限制为4M,这里可以在配置文件web.config中修改,在httpRuntime节点中加入如下属性即可:

<configuration>
  <system.web>  
    <httpRuntime executionTimeout="300" maxRequestLength="51200" useFullyQualifiedRedirectUrl="false" />
  </system.web>
</configuration>

executionTimeout 属性的值是 ASP.NET 关闭前允许发生的上载秒数,maxRequestLength指限制上传文件的大小,useFullyQualifiedRedirectUrl指示客户端重定向是否是完全限定的,或者指示是否代之以将相对重定向发送到客户端。

       到这里就大功告成了,欢迎分享更好的方法!

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • System.Data.SqlClient.SqlException: 将截断字符串或二进制数据

    版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/huyuyang6688/article/...

    DannyHoo
  • 探秘BOF 和EOF

    版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/huyuyang6688/article/...

    DannyHoo
  • Java多编程知识点精华汇总

    DannyHoo
  • nginx服务的快速搭建(跨域)

    原文地址https://github.com/gs3170981/nginx_quick 1.下载以后有一个server.js 文件

    河湾欢儿
  • 讲座笔记:图匹配 Graph Matching 问题 | 机器学习&组合优化

    Multi-Graph Matching via Affinity Optimization with Graduated Consistency Regula...

    Piper蛋窝
  • 算法篇:利用map求数组交集

    题目1: https://leetcode-cn.com/problems/intersection-of-two-arrays/

    灰子学技术
  • shell实现脚本监控服务器及web应用

    实际工作中我们需要知道部署在服务器上的应用有没有问题,但是人为的操作太麻烦有咩有简单的方式呢shell来监控我们服务器运行状态以及服务器上部署的应用,如果出现异...

    菲宇
  • 【Web技术】913- 谈谈你对前端路由的理解

    好了不装了,今天我就化身性感面试官在线问大家一个问题,“谈谈你对前端路由的理解”。看到这个问题,那回答可多了去了。但是换位思考一下,你问候选人这个问题的时候,你...

    pingan8787
  • 机器人抓取领域性能评估标准

    机器人抓取涉及检测、分割、姿态估计、抓取点检测、路径规划等任务,本文主要介绍这些任务的评估标准。

    3D视觉工坊
  • 2-SAT速成

    本文只做总结性说明 2-SAT 2-SAT是k-SAT问题的一种,k-SAT问题在k>=3时已经被证明是NP完全问题 2-SAT问题定义比较简单 有n个布尔变量...

    attack

扫码关注云+社区

领取腾讯云代金券