zoukankan      html  css  js  c++  java
  • 转化成Access文件下载的实例

    昨天有位朋友问起一个问题,客户相要一个Access格式的数据,现在的情况是程序已经实现了导出Excel,问题转化成了Excel怎么样转成Access,这点实现起来相对来说比较容易,office本身就有这个功能,但问题是,从web页面导出的Excel不能很兼容的导成Access,这是一点,还有一个方面是让客户去做这个转化工作不是太好,为了提高用户体验,最好是用代码实现了。

    能不能一下子转成Access出来呢?这是我一下想的的问题,如果有的话,客户就省很多力气了,但找了一些相关资料,没有找到把GridView直接导成Access的,也就是客户端比较难实现。想到这里,客户端难实现,服务端应该好实现吧,先把我数据库中的数据转化成一个access的文件,然后客户再下载下来,这个相对来说比较容易实现。

    主要思路:在服务端,把一种数据中的数据,生成access文件,下载给客户商。

    分步实现:

    第一:创建Access文件

    第二:复制数据库格式,在Access文件中创建新表

    第三:复制数据到Access文件中

    第四:实现下载

    创建Access文件

    原本创建Access文件的语法比较简单

    首先引入命名空间using ADOX;    该命名空间位于Interop.ADOX.dll文件中,

    ADOX.CatalogClass cat = new ADOX.CatalogClass();
         cat.Create(@"Provider=Microsoft.Jet.OLEDB.4.0;Data Source=D:\Data.mdb;");

    这样,就能在指定的D盘下创建一个Data.mdb的Access文件。文件是创建好了,但引来了后续的一些问题,因为我们在实现下载时,要用到Response.WriteFile(文件),此时的Response.WriteFile()要求是独立享有这个文件操作权限,但前期创建时的进程一直释放不掉(如果谁有好的方法可以告我如何释放),这里还得考虑多人操作,有可能同时有多要在创建Access文件或释放这个文件。这样就会出现运行时错误,提示该文件被其他进程独占。

    为了解决这个问题,同时还涉及到防止进程的释放带来的其他问题,这里我捌了一下弯来实现,先创建一个Data.mdb存放在网站的App_Data文件夹下,后期所有客户端要下导出Access表,都在这里拷贝一份,然后对新文件进行操作。另外再新建一个DownLoad文件夹用来存用户下载的Access文件。

    代码如下:

    //拷贝Access数据表,不移动

    string newfile=DateTime .Now.ToString ("MMddhhmmss")+".mdb";

    File.Copy(Server.MapPath("~") + @"\App_Data\data.mdb", Server.MapPath("~") + @"\App_Data\" + newfile);

    复制数据库格式,在Access文件中创建新表

    在这里,我是从SQL Server中生表生成Access的表,当然,可以用相关的工具完成些功能,但如果对方的数据库不是SQL Server可能就不太好用了。这里选择SQL Server只是一种演示数据库。

    我们知道,创建一张表有几个要素,表名,字段名,字段类型,字段大小,这里,我们得先从SQL Server中查出该表的这些要素,然后创建Access表,如果是其他数据库,就要相应的转化一下对应的数据类型格式。

    代码实现如下:

    在web.config中创建两个连接字符串

    <add name="AccessConStr" connectionString="Provider=Microsoft.Jet.OLEDB.4.0;Data Source=|DataDirectory|\data.mdb;Persist Security Info=True;" providerName="System.Data.OleDb"/>

    <add name="SqlConStr" connectionString="server=.;database=cum_data;uid=sa;pwd=sa;" providerName="System.Data.SqlClient"/>

    这两个字符,一个用来连接SQL Server数据库一个用来连接我们创建的Access数据库。

    接来查看导库的代码:

    string SqlConstr = ConfigurationManager.ConnectionStrings["SqlConStr"].ConnectionString;

    string sql = "select column_name,data_type,character_maximum_length from information_schema.columns where table_name='kh'";

    SqlConnection con = new SqlConnection(SqlConstr);

    SqlCommand cmd = new SqlCommand(sql, con);

    string AccSql = "";//用来组合生成Access表的SQL语句

    try

    {

    con.Open();

    SqlDataReader DR = cmd.ExecuteReader();

    while (DR.Read())

    {

    if (DR.GetValue(2).ToString() == "")

    {

    AccSql += DR.GetValue(0).ToString() + " " + DR.GetValue(1).ToString() + ",";

    }

    else

    {

    AccSql += DR.GetValue(0).ToString() + " " + DR.GetValue(1).ToString() + "(" + DR.GetValue(2).ToString() + "),";

    }

    }

    }

    catch

    { }

    finally

    {

    con.Close();

    }

    AccSql = AccSql.Substring(0, AccSql.Length - 1);

    string AccConstr = ConfigurationManager.ConnectionStrings["AccessConStr"].ConnectionString.Replace ("data.mdb",newfile );

    OleDbConnection CON = new OleDbConnection(AccConstr);

    OleDbCommand CMD = new OleDbCommand();

    CMD.Connection = CON;

    try

    {

    CON.Open();

    CMD.CommandText = "create table kh(" + AccSql + ")";

    CMD.ExecuteNonQuery();

    }

    catch

    { }

    finally

    {

    CON.Close();

    }

    在上述代码中,SQL语句是比较关键的,“select column_name,data_type,character_maximum_length from information_schema.columns where table_name='kh'”,本SQL查询语句是从把原表的字段名,字段类型,字段大小查询出来,当然,不同的数据库,这个语句不近相同。

    复制数据到Access文件中

    到现在,我们已创建好了数据表格式,接下来,就是要把数据导到Access中。

    代码如下:

    SqlDataAdapter DA = new SqlDataAdapter("select * from kh", SqlConstr);

    DataTable DT = new DataTable();

    DA.Fill(DT);

    foreach (DataRow DRR in DT.Rows)

    {

    DRR.SetAdded();

    }

    OleDbDataAdapter ODA = new OleDbDataAdapter("select * from kh", AccConstr);

    OleDbCommandBuilder cmb = new OleDbCommandBuilder(ODA);

    ODA.Update(DT);

    这里用到了Adapter的一个方法,Update,可以把DataTable中RowState为Add的数据提交会数据库,这样做起来相对要省力些。

    在里,要把有数据的Access文件,移动到DownLoad文件夹下

    File.Move(Server.MapPath("~") + @"\App_Data\"+newfile , Server.MapPath("~") + @"\DownLoad\" + newfile);

    实现下载

    Response.Clear();

    Response.ClearHeaders();

    Response.Charset = "utf-8";

    Response.Buffer = false ;

    this.EnableViewState = false;

    Response.ContentType = "application/vnd.openxmlformats-officedocument.wordprocessingml.document";

    Response.ContentEncoding = System.Text.Encoding.UTF8;

    Response.AppendHeader("Content-Disposition", "attachment;filename=" + newfile);

    Response.WriteFile(Server.MapPath("~") + @"\DownLoad\" + newfile ,true );

    Response.Flush();

    Response.Close();

    Response.End();

    下载代码是通用的代码,这里不作说明。

    整个解决方案就是把文件操作,数据操作,文件下载有机的结合在一起,实现这个用户下载Access文件的需求。

    其实有时候我们去实现一个问题的时候,总体是有一定难度的,但当我们想办法,通过分解问题来实现的时候,相对来说就比较简单的实现了。

  • 相关阅读:
    如何用VSCode手动编译Ace Editor
    libuv源码分析
    二叉平衡查找树---红黑树
    tcp滑动窗口与拥塞控制
    ceph架构剖析
    腾讯面试总结
    协程的实现原理
    dhcp协议交互报文
    libuv源码分析前言
    Protobuf使用规范分享
  • 原文地址:https://www.cnblogs.com/axzxs2001/p/1458171.html
Copyright © 2011-2022 走看看