获取数据库结构的奥义......(无码,完全版,未删节)

2023-10-30 00:59

本文主要是介绍获取数据库结构的奥义......(无码,完全版,未删节),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

看过《朝花夕拾:代码生成器的基础——获取数据源的架构信息》一文,感觉,方法正确但是点到即止,没有完全说透,所以来个补全版。经验之谈,欢迎拍砖

 

首先是获取的方法,没错,都是采用的Connection的GetSchema方法,但是这个方法对于不通的Connection来说,也就是底层连接的数据库有所不同的话,返回的DataTable其实是有区别的。有的是返回所有表和视图,有的是返回系统表+用户表+视图,注意,这个跟数据库有关,而不是XXXConnection,用OledbConnection连接Access和SqlServer返回的就是完全不同的。比如OledbConnection连接Access返回的DataTable里面第4列的值必须是“Table”的才是用户表,其余的都是系统表。如果用MysqlConnection连接Mysql的话第四列必须等于“BASE TABLE”的才是用户表。就算使用SqlClient去连接SqlServer2000和SqlServer2005也是有区别的。前者的第四列是“Base Table”而后者根本不需要判断,返回的都是用户表。

SqlServer2005:

 

ContractedBlock.gif ExpandedBlockStart.gif SqlServer2005
<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />--> 1 using (SqlConnection conn = new SqlConnection(ConnStr))
 2ExpandedBlockStart.gifContractedBlock.gif                {
 3                    if (conn.State == ConnectionState.Closed)
 4ExpandedSubBlockStart.gifContractedSubBlock.gif                    {
 5                        conn.Open();
 6                    }

 7                    DataTable sc = conn.GetSchema("Tables");
 8                    conn.Close();
 9                    foreach (DataRow row in sc.Rows)
10ExpandedSubBlockStart.gifContractedSubBlock.gif                    {
11                        Tables.Add(row[2].ToString());
12                    }

13                }
 
14

 

               SqlServer2000:

 

ContractedBlock.gif ExpandedBlockStart.gif SqlServer2000
<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />--> 1 using (SqlConnection conn = new SqlConnection(ConnStr))
 2ExpandedBlockStart.gifContractedBlock.gif                {
 3                    if (conn.State == ConnectionState.Closed)
 4ExpandedSubBlockStart.gifContractedSubBlock.gif                    {
 5                        conn.Open();
 6                    }
 7                    DataTable sc = conn.GetSchema("Tables");
 8                    conn.Close();
 9                    foreach (DataRow row in sc.Rows)
10ExpandedSubBlockStart.gifContractedSubBlock.gif                    {
11                        if (row[3].ToString().Equals("BASE TABLE"))
12ExpandedSubBlockStart.gifContractedSubBlock.gif                        {
13                            Tables.Add(row[2].ToString());
14                        }

15                    }

16                }
 
17

 

 

 

               MySql:

 

ContractedBlock.gif ExpandedBlockStart.gif MySql
<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />--> 1               using (MySqlConnection conn = new MySqlConnection(ConnStr))
 2ExpandedBlockStart.gifContractedBlock.gif                {
 3                    if (conn.State == ConnectionState.Closed)
 4ExpandedSubBlockStart.gifContractedSubBlock.gif                    {
 5                        conn.Open();
 6                    }

 7                    DataTable sc = conn.GetSchema("Tables");
 8                    conn.Close();
 9                    foreach (DataRow row in sc.Rows)
10ExpandedSubBlockStart.gifContractedSubBlock.gif                    {
11                        if (row[3].ToString().Equals("BASE TABLE"))
12ExpandedSubBlockStart.gifContractedSubBlock.gif                        {
13                            Tables.Add(row[2].ToString());
14                        }

15                    }

16                }
 
17

 

 Access:

 

ContractedBlock.gif ExpandedBlockStart.gif Access
<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />--> 1using (OleDbConnection conn = new OleDbConnection(ConnStr))
 2ExpandedBlockStart.gifContractedBlock.gif{
 3    if (conn.State == ConnectionState.Closed)
 4ExpandedSubBlockStart.gifContractedSubBlock.gif    {
 5        conn.Open();
 6    }
 
 7
 8    DataTable sc = conn.GetSchema("Tables");
 9    conn.Close();
10    foreach (DataRow row in sc.Rows)
11ExpandedSubBlockStart.gifContractedSubBlock.gif    {
12        if (row[3].ToString().Equals("TABLE"))
13ExpandedSubBlockStart.gifContractedSubBlock.gif        {
14            Tables.Add(row[2].ToString());
15        }

16    }

17}
 
18
19

 

获取到表名后就可以通过表名去获取表的列信息,但是这个时候并不是单纯的说通过DataReader去GetSchama。这样子说纯粹是让人走弯路。

using (SqlDataReader rd = cmd.ExecuteReader(CommandBehavior.KeyInfo))
                {
                    Schema = rd.GetSchemaTable();
                    rd.Close();
                }

注意需要CommandBehavior.KeyInfo这个参数,加了这个参数获得的DataReader才能够通过GetSchemaTable()获得列信息。

同样的,通过不通的DataReader和不通的数据库,所得到的DataTable的数据还是有很多的差别,而且差别足够大到影响我们开发了,我不是单单获取一个列名就够了。

我们这里假设我们要获取的关键数据有

列名

类型(Varchar之类)

数据类型(程序里的类型,比如String)

是否主键

是否自动增加的列

我这里定义了一个类,也算是一个数据结构来获取数据表里的这些数据

 

ContractedBlock.gif ExpandedBlockStart.gif ColumnProperty
<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />--> 1public class ColumnProperty
 2ExpandedBlockStart.gifContractedBlock.gif{
 3ExpandedSubBlockStart.gifContractedSubBlock.gif    /**//// <summary>
 4    /// Column Name of DataTable
 5    /// 列名
 6    /// </summary>

 7ExpandedSubBlockStart.gifContractedSubBlock.gif    public string ColumnName getset; }
 8ExpandedSubBlockStart.gifContractedSubBlock.gif    /**//// <summary>
 9    /// Data Column's DataType
10    /// 用数值表示的数据列类型(其实也就是枚举的值)
11    /// </summary>

12ExpandedSubBlockStart.gifContractedSubBlock.gif    public int TypeNumber getset; }
13ExpandedSubBlockStart.gifContractedSubBlock.gif    /**//// <summary>
14    /// Property's Datatype in program
15    /// 实体字段的类型(string或者int之类)
16    /// </summary>

17ExpandedSubBlockStart.gifContractedSubBlock.gif    public string DataType getset; }
18ExpandedSubBlockStart.gifContractedSubBlock.gif    /**//// <summary>
19    /// if is Primary Key
20    /// 是否是主键
21    /// </summary>

22ExpandedSubBlockStart.gifContractedSubBlock.gif    public bool IsKey getset; }
23ExpandedSubBlockStart.gifContractedSubBlock.gif    /**//// <summary>
24    /// if is auto upgrede
25    /// 是否是自增字段(表示不能插入数据)
26    /// </summary>

27ExpandedSubBlockStart.gifContractedSubBlock.gif    public bool IsAuto getset; } 
28
29ExpandedSubBlockStart.gifContractedSubBlock.gif    /**//// <summary>
30    /// print schema result,this method used for test
31    /// </summary>
32    /// <returns>schema infomation</returns>

33}
 
34
35

 

 

这里注意第二个field,这个属性是数据库里的类型,也就是用来给Parameter对象使用的,也就是数据定义在数据库里的类型。在程序里一般使用SqlDbType这个枚举来标示,其实这里我们用int来表示也就是说这个int的值和枚举的值是一一对应的。我们得到这个数值之后只需要

(SqlDbType)System.Enum.Parse(typeof(SqlDbType), “数值”)就能够得到对应的Type了。

 

下面的代码是不同数据库读取关键信息的不通点,注意看读取的列序号的不同

SqlServer2005:

 

ContractedBlock.gif ExpandedBlockStart.gif SqlServer2005
<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />--> 1foreach (DataRow row in Schema.Rows)
 2ExpandedBlockStart.gifContractedBlock.gif            {
 3                ppt = new ColumnProperty();
 4                ppt.ColumnName = row[0].ToString();
 5                ppt.TypeNumber = int.Parse(row[29].ToString());
 6                ppt.DataType = row[12].ToString();
 7                ppt.IsKey = bool.Parse(row[6].ToString().ToLower());
 8                ppt.IsAuto = bool.Parse(row[17].ToString().ToLower());
 9                Rs.Add(ppt);
10            }
 

 

SqlServer2000:

 

ContractedBlock.gif ExpandedBlockStart.gif SqlServer2000
<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />--> 1foreach (DataRow row in Schema.Rows)
 2ExpandedBlockStart.gifContractedBlock.gif            {
 3                ppt = new ColumnProperty();
 4                ppt.ColumnName = row[0].ToString();
 5                ppt.TypeNumber = int.Parse(row[29].ToString());
 6                ppt.DataType = row[12].ToString();
 7                ppt.IsKey = bool.Parse(row[6].ToString().ToLower());
 8                ppt.IsAuto = bool.Parse(row[17].ToString().ToLower());
 9                Rs.Add(ppt);
10            } 
11
12

 

 

MySql:

 

ContractedBlock.gif ExpandedBlockStart.gif Mysql
<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />--> 1foreach (DataRow row in Schema.Rows)
 2ExpandedBlockStart.gifContractedBlock.gif            {
 3                ppt = new ColumnProperty();
 4                ppt.ColumnName =

这篇关于获取数据库结构的奥义......(无码,完全版,未删节)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/304573

相关文章

mysql通过frm和ibd文件恢复表_mysql5.7根据.frm和.ibd文件恢复表结构和数据

《mysql通过frm和ibd文件恢复表_mysql5.7根据.frm和.ibd文件恢复表结构和数据》文章主要介绍了如何从.frm和.ibd文件恢复MySQLInnoDB表结构和数据,需要的朋友可以参... 目录一、恢复表结构二、恢复表数据补充方法一、恢复表结构(从 .frm 文件)方法 1:使用 mysq

golang获取prometheus数据(prometheus/client_golang包)

《golang获取prometheus数据(prometheus/client_golang包)》本文主要介绍了使用Go语言的prometheus/client_golang包来获取Prometheu... 目录1. 创建链接1.1 语法1.2 完整示例2. 简单查询2.1 语法2.2 完整示例3. 范围值

SpringBoot使用Jasypt对YML文件配置内容加密的方法(数据库密码加密)

《SpringBoot使用Jasypt对YML文件配置内容加密的方法(数据库密码加密)》本文介绍了如何在SpringBoot项目中使用Jasypt对application.yml文件中的敏感信息(如数... 目录SpringBoot使用Jasypt对YML文件配置内容进行加密(例:数据库密码加密)前言一、J

javaScript在表单提交时获取表单数据的示例代码

《javaScript在表单提交时获取表单数据的示例代码》本文介绍了五种在JavaScript中获取表单数据的方法:使用FormData对象、手动提取表单数据、使用querySelector获取单个字... 方法 1:使用 FormData 对象FormData 是一个方便的内置对象,用于获取表单中的键值

Python调用Orator ORM进行数据库操作

《Python调用OratorORM进行数据库操作》OratorORM是一个功能丰富且灵活的PythonORM库,旨在简化数据库操作,它支持多种数据库并提供了简洁且直观的API,下面我们就... 目录Orator ORM 主要特点安装使用示例总结Orator ORM 是一个功能丰富且灵活的 python O

使用 sql-research-assistant进行 SQL 数据库研究的实战指南(代码实现演示)

《使用sql-research-assistant进行SQL数据库研究的实战指南(代码实现演示)》本文介绍了sql-research-assistant工具,该工具基于LangChain框架,集... 目录技术背景介绍核心原理解析代码实现演示安装和配置项目集成LangSmith 配置(可选)启动服务应用场景

Python中顺序结构和循环结构示例代码

《Python中顺序结构和循环结构示例代码》:本文主要介绍Python中的条件语句和循环语句,条件语句用于根据条件执行不同的代码块,循环语句用于重复执行一段代码,文章还详细说明了range函数的使... 目录一、条件语句(1)条件语句的定义(2)条件语句的语法(a)单分支 if(b)双分支 if-else(

如何利用Java获取当天的开始和结束时间

《如何利用Java获取当天的开始和结束时间》:本文主要介绍如何使用Java8的LocalDate和LocalDateTime类获取指定日期的开始和结束时间,展示了如何通过这些类进行日期和时间的处... 目录前言1. Java日期时间API概述2. 获取当天的开始和结束时间代码解析运行结果3. 总结前言在J

使用Navicat工具比对两个数据库所有表结构的差异案例详解

《使用Navicat工具比对两个数据库所有表结构的差异案例详解》:本文主要介绍如何使用Navicat工具对比两个数据库test_old和test_new,并生成相应的DDLSQL语句,以便将te... 目录概要案例一、如图两个数据库test_old和test_new进行比较:二、开始比较总结概要公司存在多

MySQL数据库函数之JSON_EXTRACT示例代码

《MySQL数据库函数之JSON_EXTRACT示例代码》:本文主要介绍MySQL数据库函数之JSON_EXTRACT的相关资料,JSON_EXTRACT()函数用于从JSON文档中提取值,支持对... 目录前言基本语法路径表达式示例示例 1: 提取简单值示例 2: 提取嵌套值示例 3: 提取数组中的值注意