ExcelHome技术论坛

 找回密码
 免费注册

QQ登录

只需一步,快速开始

快捷登录

搜索
EH技术汇-专业的职场技能充电站 妙哉!函数段子手趣味讲函数 Excel服务器-会Excel,做管理系统 效率神器,一键搞定繁琐工作
HR薪酬管理数字化实战 Excel 2021函数公式学习大典 Excel数据透视表实战秘技 打造核心竞争力的职场宝典
让更多数据处理,一键完成 数据工作者的案头书 免费直播课集锦 ExcelHome出品 - VBA代码宝免费下载
用ChatGPT与VBA一键搞定Excel WPS表格从入门到精通 Excel VBA经典代码实践指南
查看: 3435|回复: 6

从微软下载中心copy几个SQL的函数,我是不甚了解,现贴了?以后再用

[复制链接]

TA的精华主题

TA的得分主题

发表于 2005-5-4 14:55 | 显示全部楼层 |阅读模式

Option Explicit

Dim sStoredSQL As String

Function QueryRefreshInternal(Optional Ref As Variant) As Variant Dim Ref1 As Object On Error GoTo ReturnPoundRef If IsMissing(Ref) Then Set Ref1 = ActiveCell Else Set Ref1 = Ref End If If Application.DataEntryMode <> xlOff Then GoTo ReturnPoundRef If Application.ActiveWindow.SelectedSheets.Count > 1 Then GoTo ReturnPoundRef If TypeName(Ref1) <> "Range" Then GoTo ReturnPoundRef If Ref1.Areas.Count > 1 Then GoTo ReturnPoundRef If Ref1.Rows.Count > 1 Or Ref1.Columns.Count > 1 Then GoTo ReturnPoundRef ' If the single cell is within a querytable range, then refresh that range synchronously. ' Accessing the QueryTable property for a range that isn't within a querytable ' will generate a runtime exception that will cause #REF to be returned. Failure ' to refresh will generate a runtime exception that will cause #REF to be returned. Ref1.QueryTable.Refresh False QueryRefreshInternal = True Exit Function ReturnPoundRef: QueryRefreshInternal = CVErr(xlErrRef) Exit Function End Function

Private Function ValidateBooleanArgument(arg As Variant, fDefault As Boolean, fResult As Boolean) As Boolean ValidateBooleanArgument = True If IsMissing(arg) Then fResult = fDefault Else On Error GoTo CannotConvertToBool fResult = CBool(arg) End If Exit Function CannotConvertToBool: ValidateBooleanArgument = False Exit Function End Function

Function QueryGetDataDialogInternal(Optional ConnectionStr As Variant, Optional QueryText As Variant, Optional KeepQueryDef As Variant, Optional FieldNames As Variant, Optional RowNumbers As Variant, Optional Destination As Variant, Optional Execute As Variant, Optional KeepPassword As Variant) As Variant Dim sConnectionString As String Dim sSQL As String ' GetDataDialog doesn't actually work for Excel97, since there's no interactive form. ' Fail out if the connection string and/or querytext string are missing or empty On Error GoTo ReturnPoundRef If IsMissing(ConnectionStr) Then GoTo ReturnPoundRef sConnectionString = CStr(ConnectionStr) If Len(sConnectionString) = 0 Then GoTo ReturnPoundRef If IsMissing(QueryText) Then GoTo ReturnPoundRef sSQL = CStr(QueryText) If Len(sSQL) = 0 Then GoTo ReturnPoundRef QueryGetDataDialogInternal = QueryGetDataValidateArgs(sConnectionString, sSQL, KeepQueryDef, FieldNames, RowNumbers, Destination, Execute, KeepPassword) Exit Function ReturnPoundRef: QueryGetDataDialogInternal = CVErr(xlErrRef) Exit Function End Function

Function QueryGetDataXLMStub(ConnectionStr As Variant, QueryText As Variant, Optional KeepQueryDef As Variant, Optional FieldNames As Variant, Optional RowNumbers As Variant, Optional Destination As Variant, Optional Execute As Variant, Optional KeepPassword As Variant) As Variant Dim sConnectionString As String Dim sSQL As String Dim v As Variant On Error GoTo ReturnPoundRef If Not IsArray(ConnectionStr) Then sConnectionString = CStr(ConnectionStr) Else For Each v In ConnectionStr sConnectionString = sConnectionString & CStr(v) Next End If If Not IsArray(QueryText) Then sSQL = CStr(QueryText) Else For Each v In QueryText sSQL = sSQL & CStr(v) Next End If QueryGetDataXLMStub = QueryGetDataValidateArgs(sConnectionString, sSQL, KeepQueryDef, FieldNames, RowNumbers, Destination, Execute, KeepPassword) Exit Function ReturnPoundRef: QueryGetDataXLMStub = CVErr(xlErrRef) Exit Function End Function

Function QueryGetDataValidateArgs(ConnectionStr As String, QueryText As String, Optional KeepQueryDef As Variant, Optional FieldNames As Variant, Optional RowNumbers As Variant, Optional Destination As Variant, Optional Execute As Variant, Optional KeepPassword As Variant) As Variant Dim fKeepQueryDef As Boolean Dim fFieldNames As Boolean Dim fRowNumbers As Boolean Dim fExecute As Boolean Dim fKeepPassword As Boolean Dim sSQL As String Dim rDestination As Object If Not ValidateBooleanArgument(KeepQueryDef, True, fKeepQueryDef) Then GoTo ReturnPoundRef If Not ValidateBooleanArgument(FieldNames, True, fFieldNames) Then GoTo ReturnPoundRef If Not ValidateBooleanArgument(RowNumbers, True, fRowNumbers) Then GoTo ReturnPoundRef If Not ValidateBooleanArgument(Execute, True, fExecute) Then GoTo ReturnPoundRef If Not ValidateBooleanArgument(KeepPassword, True, fKeepPassword) Then GoTo ReturnPoundRef ' handle building up the array of SQL. Instead of keeping around an array like the Excel5 implementation, ' we keep the SQL in a single string for convenience. If fExecute Then If Len(sStoredSQL) > 0 Then sSQL = sStoredSQL & QueryText Else sSQL = QueryText End If sStoredSQL = "" ' now that we have the SQL text, validate the destination range object If Application.DataEntryMode <> xlOff Then GoTo ReturnPoundRef If Application.ActiveWindow.SelectedSheets.Count > 1 Then GoTo ReturnPoundRef On Error GoTo ReturnPoundRef If IsMissing(Destination) Then Set rDestination = ActiveCell Else Set rDestination = Destination End If If TypeName(rDestination) <> "Range" Then GoTo ReturnPoundRef QueryGetDataValidateArgs = QueryGetDataFetch(ConnectionStr, sSQL, fKeepQueryDef, fFieldNames, fRowNumbers, rDestination, fKeepPassword) Else ' not executing, just building the SQL string. sStoredSQL = sStoredSQL & QueryText QueryGetDataValidateArgs = True End If Exit Function ReturnPoundRef: QueryGetDataValidateArgs = CVErr(xlErrRef) Exit Function End Function

Private Function QueryGetDataFetch(sConnectionString As String, sSQL As String, fKeepQueryDef As Boolean, fFieldNames As Boolean, fRowNumbers As Boolean, rDestination As Range, fKeepPassword As Boolean) As Variant Dim qt As QueryTable On Error GoTo NotInQueryTable Set qt = rDestination.QueryTable If Not (qt Is Nothing) Then ' QueryTable already covers the destination range. Modify it to have the new connectionstring ' and query parameter On Error GoTo ReturnPoundRef qt.Connection = "ODBC;" & sConnectionString qt.Sql = sSQL Else CreateNewQueryTable: On Error GoTo ReturnPoundRef Set qt = rDestination.Worksheet.QueryTables.Add("ODBC;" & sConnectionString, rDestination, sSQL) End If qt.RowNumbers = fRowNumbers qt.FieldNames = fFieldNames qt.SavePassword = fKeepPassword On Error GoTo RefreshFailed qt.Refresh False If Not fKeepQueryDef Then qt.Delete End If QueryGetDataFetch = True Exit Function NotInQueryTable: Resume CreateNewQueryTable ReturnPoundRef: QueryGetDataFetch = CVErr(xlErrRef) Exit Function RefreshFailed: qt.Delete GoTo ReturnPoundRef End Function

Sub Compiler() Dim rLanguageRange As Range Dim sLang As String Dim wsLocal As Worksheet Set wsLocal = ThisWorkbook.Worksheets("Loc Table") sLang = wsLocal.Range("SelectedLanguage").Value Set rLanguageRange = wsLocal.Range("USAColumnStart") ' Do the USA VBA functions, which are present in addition to the localized ' VBA functions If StrComp(sLang, "USA", 1) <> 0 Then CompileVBAFunctionNames rLanguageRange ' and do the rest of the compilation for the selected language Do Until IsEmpty(rLanguageRange.Value) If StrComp(rLanguageRange.Value, sLang, 1) = 0 Then ' Compile the VBA Function names CompileVBAFunctionNames rLanguageRange ' Compile the XLM Function names CompileXLMFunctionNames rLanguageRange ' Compile the XLM Function arguments CompileXLMFunctionArgs rLanguageRange ' Setup the summary info using this language CompileSummaryInfo rLanguageRange Exit Do End If Set rLanguageRange = rLanguageRange.Offset(0, 1) Loop End Sub

Sub CompileVBAFunctionNames(rLocRange As Range) ' QueryRefresh Application.MacroOptions rLocRange.Offset(6, 0).Value, Description:=rLocRange.Offset(5, 0).Value, _ hasmenu:=False, hasshortcutkey:=False, Category:=6

' QueryGetData Application.MacroOptions rLocRange.Offset(9, 0).Value, Description:=rLocRange.Offset(8, 0).Value, _ hasmenu:=False, hasshortcutkey:=False, Category:=6 ' QueryGetDataDialog Application.MacroOptions rLocRange.Offset(12, 0).Value, Description:=rLocRange.Offset(11, 0).Value, _ hasmenu:=False, hasshortcutkey:=False, Category:=6 End Sub

Sub CompileXLMFunctionNames(rLocRange As Range) Dim sSheetPrefix As String Dim wsXLQUERY As Worksheet Dim sMacroName As String Set wsXLQUERY = ThisWorkbook.Excel4IntlMacroSheets("XLQUERY") sSheetPrefix = "=XLQUERY!" ' the QUERY.REFRESH name is offset down by 4 rows sMacroName = rLocRange.Offset(4, 0).Value ThisWorkbook.Names.Add sMacroName, _ sSheetPrefix & wsXLQUERY.Range("QueryRefreshLocation").Address(True, True, xlA1, False), _ MacroType:=2, Category:=6 Application.MacroOptions sMacroName, Description:=rLocRange.Offset(5, 0).Value, _ hasmenu:=False, hasshortcutkey:=False, Category:=6 ' the QUERY.GET.DATA name is offset down by 6 rows sMacroName = rLocRange.Offset(7, 0).Value ThisWorkbook.Names.Add sMacroName, _ sSheetPrefix & wsXLQUERY.Range("QueryGetDataLocation").Address(True, True, xlA1, False), _ MacroType:=2, Category:=6 Application.MacroOptions sMacroName, Description:=rLocRange.Offset(8, 0).Value, _ hasmenu:=False, hasshortcutkey:=False, Category:=6

' the QUERY.GET.DATA? name is offset down by 8 rows sMacroName = rLocRange.Offset(10, 0).Value ThisWorkbook.Names.Add sMacroName, _ sSheetPrefix & wsXLQUERY.Range("QueryGetDataDialogLocation").Address(True, True, xlA1, False), _ MacroType:=2, Category:=6 Application.MacroOptions sMacroName, Description:=rLocRange.Offset(11, 0).Value, _ hasmenu:=False, hasshortcutkey:=False, Category:=6 End Sub

Sub CompileXLMFunctionArgs(rLocRange As Range) Dim wsXLQUERY As Worksheet Dim rDest As Range Dim rSrc As Range Dim i As Integer Set wsXLQUERY = ThisWorkbook.Excel4IntlMacroSheets("XLQUERY") Set rDest = wsXLQUERY.Range("mg00s.addin1") Set rSrc = rLocRange.Offset(13, 0) For i = 0 To 7 rDest.Offset(i, 0).Value = rSrc.Offset(i, 0).Value Next End Sub

Sub CompileSummaryInfo(rLocRange As Range) Dim sOldUserName As String ThisWorkbook.Title = rLocRange.Offset(1, 0).Value ThisWorkbook.Author = rLocRange.Offset(2, 0).Value ThisWorkbook.BuiltinDocumentProperties("Company").Value = rLocRange.Offset(2, 0).Value ThisWorkbook.Comments = rLocRange.Offset(3, 0).Value sOldUserName = Application.UserName Application.UserName = rLocRange.Offset(2, 0).Value ThisWorkbook.IsAddin = True 'ThisWorkbook.SaveCopyAs "C:\xlquery.xla" 'Application.Dialogs(xlDialogSaveCopyAs).Show ThisWorkbook.IsAddin = False Application.UserName = sOldUserName End Sub

-------完完完完完完完完完-完完完完完完完完完-------

[此贴子已经被作者于2005-5-4 16:38:27编辑过]

TA的精华主题

TA的得分主题

 楼主| 发表于 2005-5-4 16:15 | 显示全部楼层
从 .NET 应用程序访问 Microsoft Office 数据发布日期: 8/12/2004 | 更新日期: 8/12/2004

Christa Carpentiere

适用范围:

Microsoft® Access

Microsoft® Excel

Microsoft® Office

Microsoft® Visual Basic® .NET

Microsoft® Visual Studio® .NET

摘要:查看如何从 Office 文件中获取 Microsoft Access 和 Excel 数据并将其放入 .NET 应用程序中,并查看 OLE DB 提供程序怎样用于编写使处理 .xls 或 .mdb 文件更容易的通用代码。

引言

在 Microsoft .NET 应用程序中使用 Microsoft Office 数据时所需的大多数任务与使用任何其他数据时的任务相同 - 创建连接、根据使用数据时的需要创建 DataReader 或 DataAdapter、创建一个或多个 DataSet 来封装相关数据的 DataTable,等等。因此,我不打算在此重新编写 ADO.NET 文档。本文的目的在于,提供按 Office 数据源编写基础数据检索代码时所需的缺失或广为分布的信息段。好像那些知道 Jet 的奇特之处的人士却不是很熟悉 ADO.NET,而大部分的 ADO.NET 专业人士不是很了解支持 Office 可编程序性的技术。因此提供一个快速的概述,希望可以解答我所见到的这两方的常见问题。

首先让我们讨论一下连接到 Microsoft Excel 和 Microsoft Access 的数据并从中检索数据的基础。我认为此领域中存在最大的问题 - 通常一旦人们能够解决该问题,Office 特定问题将会得到解决,而且其余问题主要集中于如何使用 ADO.NET。下面我将讨论如何使用 OleDbConnection 对象的 GetOleDbSchemaTable 方法来检索关于运行时 Access 或 Excel 数据的架构信息。这样,您可以避开关于数据源的硬编码信息;数据源非常有用,从中可以获取数据。

返回页首

TA的精华主题

TA的得分主题

 楼主| 发表于 2005-5-4 16:16 | 显示全部楼层
建立连接

如果要连接到 Excel 电子表格或 Access 数据库,则需要创建一个 OleDbConnection 对象,并向其传递一个带有目标数据源详细信息的连接字符串。您需要为 Provider 属性指定 Jet 4.0 OLE DB 提供程序。

连接到 Access

让我们看一看用于 Access 数据库的典型连接字符串:

"Provider=Microsoft.Jet.OLEDB.4.0;Data Source= C:\Databases\mymusic.mdb"

这非常简单 - 它仅指定 Provider 和 Data Source 属性。如上所述,Provider 是 Jet 4.0 提供程序,而 Data Source 属性包含数据库的完全限定路径。

当然,您甚至还需要确保 Access 数据库的安全,对不对?这样,您还需要指定工作组信息文件(默认情况下为 system.mdw)的路径,并提供用户 ID 和密码:

"Provider=Microsoft.Jet.OLEDB.4.0;Data " & _ "Source=C:\Databases\mymusic.mdb; " & _"Jet OLEDB:System database=" & _"C:\Databases\system.mdw; " & _"User ID=UserX;Password=UserXPassword"

请注意,指定控制 Access 数据库安全性的 .mdw 文件时需要使用 Jet OLE DB 特定的属性。如果希望更好地控制数据库的行为,则可以浏览 Jet OLE DB 提供程序(该提供程序位于 Appendix B:Microsoft Jet 4.0 OLE DB Properties Reference)所提供的其他属性,来控制锁定行为和如何处理特定类型的失败,等等。请注意,只有那些可以在连接字符串中设置的属性才可以访问;指定提供程序的属性之前需要连接处于打开状态,否则将无法设置这些属性。

可以使用 Microsoft® Visual Studio® .NET,来获取包括所有的 Jet OLE DB 提供程序设置、完整的 Access 连接字符串的模板。在服务器资源管理器中创建一个与 Access 数据库的数据连接,然后使用工具箱的数据部分创建一个 OleDbConnection 对象。得到的对象的 ConnectionString 属性会包含默认的所有 Jet OLE DB 提供程序属性。

TA的精华主题

TA的得分主题

 楼主| 发表于 2005-5-4 16:16 | 显示全部楼层
连接到 Excel

现在,让我们看看 Excel 电子表格的典型连接字符串的外观:

"Provider=Microsoft.Jet.OLEDB.4.0;" & _"Data Source=C:\Spreadsheets\calculations.xls;" & _"Extended Properties=Excel 8.0"

在连接到 Excel 文件的情况下,可以看到我们必须在连接字符串中使用 Extended Properties 属性以及 Provider 和 Data Source。如果是 Excel 8.0 和更高版本,则使用“Excel 8.0”设置。如果希望获取此属性的其他可接受值的更多信息,请参阅 ADO Provider Properties and Settings 的“Extended Properties Property Settings”部分。

您会说:“啊,但是 Excel 中的安全性怎么样?”嗯,恐怕没有激动人心的消息。您无法打开与受密码保护的电子表格的连接,除非已手动地在 Excel 中打开此电子表格(有关详细信息,请参阅 XL2000:"Could Not Decrypt File" Error with Password Protected File)。描述的错误出现于 Excel ODBC 提供程序,但是在 Jet 4.0 OLE DB 提供程序中该行为相同。其他选项为,删除电子表格的密码并使用某些其他安全机制(例如,限制文件所在文件夹的权限)来控制访问。

不幸的是,您也无法使用 Visual Studio .NET 获取 Excel 连接字符串的模板。进行一些小小的尝试,即可创建 Excel 数据连接;但是您会发现其属性不可编辑,而且 ConnectionString 属性将保留空白 - 这是 IDE(集成的开发环境)中的奇特之处。有关详细信息,请参阅 PRB:Cannot Configure Data Connection to Excel Files in Visual Studio .NET。这样,对于 Excel 连接,基本上都需要自己进行工作,但在这种情况下对连接进行编码就像在用户界面 (UI) 中创建连接一样简单。

返回页首检索数据

现在已经明白了如何建立与 Office 数据源的连接,就让我们看看它是怎样检索数据的。出于简单性的考虑,我打算使用 OleDbConnection/OleDbCommand/OleDbDataReader 数据检索方案。相同的方法在稍稍调整之后,可以用于构建 OleDbDataAdapter 并填充 DataSet。如果想要大体上了解一下 ADO.NET,请参阅《.NET Framework Developer''s Guide》的 Accessing Data with ADO.NET 部分,它会为您带来阅读的乐趣。

检索 Access 数据

编写 Access 数据检索代码时请切记,指定 SQL 时必须遵循的语法具有某些特性。无法在图形 UI 中创建 Access 查询,无法访问 SQL 视图,无法复制得到的 SQL 语句并将其粘贴到代码中。不,这本应非常简单的。生成的 SQL 代码通常具有一部分但不是全部的所需语法。必须在 Access 开发环境中编写代码的任何人都会知道这一点,但是对于一般的 .NET 客户端应用程序开发人员而言,这算的上是新闻。最让人烦心的是条件表达式,它需要以某种方式分隔 WHERE 字句中的特定类型的数据。日期和时间值必须使用数字记号 (#) 进行分隔。文本值必须使用单引号 ('') 进行分隔。例如:

SELECT City, Neighborhood, SalePrice, MonthsOnMarketFROM RealEstateWHERE ListingDate > #1/1/04#

有关这些问题的详细信息,请参阅 Date and Time Criteria Expressions。

需要记住的另一点看上去好像显而易见,但是却使人们感到困惑,所以我将讨论这一点:请确保 Access 表的列名称不使用保留字。可以在 SQL Reserved Words 处查询保留字。如果使用任何保留字,我想说的是如果可能就重命名列。我知道如果您已使用数据库,该操作则不太可能,所以必要时可以创建查询并将其用作有问题列的表的替换查询。仅使用 AS 重命名列,就如:

SELECT Artists.ArtistName, Genres.Genre, Labels.Label, Tracks.Public AS Track, Releases.ReleaseNameFROM (Labels INNER JOIN ((Artists...

记住这些要点,让我们来看一个示例:

Imports System Imports System.Data Imports System.Data.OleDb ... Public Function GetAccessData(ByVal UID As String, _ ByVal pwd As String, ByVal artist As String) Dim conn As New OleDbConnection Dim musicReader As OleDbDataReader Dim cmd As New OleDbCommand Dim connString As String Dim i As Integer Try ' 设置连接字符串。 connString = "Jet OLEDB:System database=" & _ "C:\Databases\system.mdw;" & _ "Data Source=C:\Databases\mymusic.mdb;" & _ "Provider=Microsoft.Jet.OLEDB.4.0;" & _ "User ID=" & UID & ";Password=" & pwd ' 打开连接。 conn.ConnectionString = connString conn.Open() '设置命令属性。 cmd.Connection = conn cmd.CommandText = "SELECT * from music " & _ "WHERE ArtistName = '" & artist & "'" ' 获取 OleDbDataReader ' 并对其进行一些处理。 musicReader = _ cmd.ExecuteReader(CommandBehavior.CloseConnection) Try While (musicReader.Read) '处理数据。 End While Finally musicReader.Close() End Try Catch ex As Exception '错误处理 End Try End Function 'GetAccessData

-------------------------------------------------------------

[此贴子已经被作者于2005-5-4 16:30:53编辑过]

TA的精华主题

TA的得分主题

 楼主| 发表于 2005-5-4 16:32 | 显示全部楼层
[广告] Excel易用宝 - 提升Excel的操作效率 · Excel / WPS表格插件       ★免费下载 ★       ★ 使用帮助
检索 Excel 数据

关于列名称,Excel 和 Access 一样具有相同的保留字限制。一般而言,如果记住 SQL 保留字,并在创建可能用作数据源的任何对象时避开保留字,则会更好。还有一点需要考虑。

Excel 的语法同样也有奇特之处。对代码影响最大的项是用于引用要返回的数据集的语法。

注意:对于最简单的 Excel 数据检索,请使用维护类似表格式的电子表格。

第一个选项是指定工作表和该表中的单元格集(可选)。需要确保工作表名称后跟美元符号和单元格集(可选)。通过使用冒号分隔集合中的起始单元格和终止单元格,来指定此单元格集。然后,使用括号将整个数据标识字符串括起。使用此类型语法的 SELECT 语句可能如下所示:

SELECT SalesMonth, TotalSales, PercentageChange1Year FROM [Sheet1$A1:E24]

另一个选项是在 Excel 中创建一个命名范围,它将起到类似于表的作用。要创建命名范围,请参阅 Create named cell references or ranges。要使用的范围名称就像 SELECT 语句中的表名称:

SELECT SalesMonth, TotalSales, PercentageChange1Year FROM SalesHighlights

记住这些要点,让我们来看一个示例:

Imports System Imports System.Data Imports System.Data.OleDb ... Public Function GetExcelData() Dim conn As New OleDbConnection Dim salesReader As OleDbDataReader Dim connString As String Dim cmd As New OleDbCommand Try ' 设置连接字符串。 connString = "Data Source=" & _ "C:\Spreadsheets\calculations.xls;" & _ "Provider=Microsoft.Jet.OLEDB.4.0;" & _ "Extended Properties=Excel 8.0;" ' 打开连接。 conn.ConnectionString = connString conn.Open() '设置命令属性。 cmd.Connection = conn cmd.CommandText = "SELECT SalesMonth, " & _ "TotalSales, PercentageChange1Year, " & _ "VolumeDiscounts, Profit from [Sheet1$]" ' 获取 OleDbDataReader ' 并对其进行一些处理。 salesReader = _ cmd.ExecuteReader(CommandBehavior.CloseConnection) Try While (salesReader.Read) '处理数据。 End While Finally salesReader.Close() End Try Catch ex As Exception '错误处理 End Try End Function 'GetExcelData
[此贴子已经被作者于2005-5-4 16:32:27编辑过]

TA的精华主题

TA的得分主题

 楼主| 发表于 2005-5-4 16:33 | 显示全部楼层
对通用数据检索使用元数据

现在可以连接到 Excel 或 Access 数据源并检索数据,让我们再深入一步。让我们看看从其中一个数据源检索元数据时需要执行的操作,随后您可以使用数据源构造数据访问代码。如果希望创建的过程提供某些通用功能,并且不希望将这些功能绑定在特定数据源上,则该进程可能会有用处。

使用 OleDbConnection.GetOleDbSchemaTable

要获取所需的元数据,则需要使用 OleDbConnection.GetOleDbSchemaTable 方法。此方法的构造函数采用一个代表 OLE DB 架构行集的 OleDbSchemaGuid 对象,和一组代表要返回的架构信息的选择条件基本内容的对象。

注意:对于那些不熟悉 OLE DB 架构行集的人而言,它们基本上是由 ANSI SQL-92 定义的数据库构造的标准化架构。每个架构行集具有为指定构造提供定义元数据的一组列(称作 .NET 文档中的“限制列”)。这样,如果请求架构信息(例如,列的架构信息或排序规则的架构信息),则您会明确知道可以得到哪种类型的数据。如果希望了解更多信息,请访问 Appendix B:Schema Rowsets。

Object 数组在文档中定义为“限制值的数组”。它用于确定(即限制)返回的数据集,有些类似于 SQL 中的 WHERE 子句。例如,连接到具有工作表 Alpha、Beta 和 Pi 的工作簿。需要架构信息来确定工作表 Beta 所包含的列。您的代码将如下所示:

schemaTable = conn.GetOleDbSchemaTable(OleDbSchemaGuid.Columns, _New Object() {Nothing, Nothing, _"Beta", Nothing})

使用 OleDbSchemaGuid.Columns 字段,来表示因需要返回列信息而应使用 COLUMNS 架构。COLUMNS 架构包含 TABLE_CATALOG、TABLE_SCHEMA、TABLE_NAME 和 COLUMN_NAME 限制列,并且需要提供代表数组中每个限制列的限制值的对象。通过将“Beta”指定为 TABLE_NAME 值,可以将返回的列信息限制为仅来自于“表”的信息。

现在,您非常熟悉我们的朋友 GetOleDbSchemaTable,让我们来看看它是如何实现的。通过使用它遍历数据源中的表和列,可以获取检索数据时所需的所有信息,而无须提前熟悉架构。让我们来看看使用 Excel 的一个示例:

Imports System Imports System.Data Imports System.Data.OleDb ... Public Function GetExcelSchema(ByVal xlsPath As String) As DataSet Dim schemaTable As New DataTable Dim workAdapter As New OleDbDataAdapter Dim workSet As New DataSet Dim conn As New OleDbConnection Dim i As Integer Dim x As Integer Dim charArray As Char() = {",", " "} Dim charArray2 As Char() = {"$"} Dim cmdString As String Dim cmdString2 As String Dim cmd As New OleDbCommand Dim tableName As String workSet.DataSetName = "excelData" Try ' 设置连接字符串。 Dim connString As String = _ "Data Source=" & xlsPath & _ ";Provider=Microsoft.Jet.OLEDB.4.0;" & _ "Extended Properties=Excel 8.0" ' 打开连接。 conn.ConnectionString = connString conn.Open() ' 使用数据源表中的架构信息 ' 填充 DataTable。 schemaTable = _ conn.GetOleDbSchemaTable(OleDbSchemaGuid.Tables, _ New Object() {Nothing, Nothing, Nothing, "TABLE"}) ' 使用表名称填充数组。 i = schemaTable.Rows.Count - 1 Dim tablesArray(i) As String For i = 0 To schemaTable.Rows.Count - 1 tablesArray(i) = schemaTable.Rows(i).Item("Table_Name") Next ' 清除 DataTable schemaTable.Clear() ' 使用表名称和列架构 ' 信息来构造 SELECT 语句, ' 并为数据源中的每个表返回数据。 For i = 0 To tablesArray.GetLength(0) - 1 ' 使用数据源列中的架构信息 ' 填充 DataTable。 schemaTable = _ conn.GetOleDbSchemaTable(OleDbSchemaGuid.Columns, _ New Object() {Nothing, Nothing, _ tablesArray(i).ToString(), Nothing}) ' 逐个查看列名称,并将其附加到 ' SELECT 语句中 cmdString = "SELECT " For x = 0 To schemaTable.Rows.Count - 1 cmdString = cmdString & _ schemaTable.Rows(x).Item("Column_Name") & _ ", " Next cmdString2 = cmdString.TrimEnd(charArray) ' 请注意,无须将“$”附加到 ' 表名称中 - 它已包括在内。 cmdString2 = cmdString2 & " FROM [" & _ tablesArray(i).ToString() & "]" ' 使用 SELECT 命令和 ' OleDbDataAdapter 填充 DataSet。 cmd.CommandText = cmdString2 workAdapter.SelectCommand = cmd workAdapter.SelectCommand.Connection = conn tableName = _ tablesArray(i).ToString().TrimEnd(charArray2) workAdapter.Fill(workSet, tableName) schemaTable.Clear() Next Catch ex As Exception '错误处理 Finally conn.Close() End Try Return workSet End Function 'GetExcelSchema

正如您所看到的,涉及的代码相当简单。如果想对 Access 数据库进行同样的操作,唯一真正的不同之处在于连接字符串,而无须将表名称格式化为工作表并使其可在 SELECT 语句中使用。

另外,可以将此方法用于 SQL Server™ 数据库,效果也很好;或者可以对其进行修改,使其根据从数据源中获取的架构信息执行其他操作。如果要尝试执行任何类型的查找或文档管理,这将非常有用。

[此贴子已经被作者于2005-5-4 16:33:51编辑过]

TA的精华主题

TA的得分主题

发表于 2011-11-30 15:41 | 显示全部楼层
[广告] VBA代码宝 - VBA编程加强工具 · VBA代码随查随用  · 内置多项VBA编程加强工具       ★ 免费下载 ★      ★使用手册
这么复杂,留个记号先
您需要登录后才可以回帖 登录 | 免费注册

本版积分规则

手机版|关于我们|联系我们|ExcelHome

GMT+8, 2025-1-11 01:38 , Processed in 0.023512 second(s), 9 queries , Gzip On, MemCache On.

Powered by Discuz! X3.4

© 1999-2023 Wooffice Inc.

沪公网安备 31011702000001号 沪ICP备11019229号-2

本论坛言论纯属发表者个人意见,任何违反国家相关法律的言论,本站将协助国家相关部门追究发言者责任!     本站特聘法律顾问:李志群律师

快速回复 返回顶部 返回列表