SQL Server 2005 - 如何从CLR存储过程传回表格结果与讯息(上)

 

亲爱的朋友们,阅读过实作CLR存储过程》一文之后,相信大家对于如何建立CLR存储过程已经有了清楚的认识。在本文中,我们要继续讨论一个非常重要的课题,那就是如何从CLR存储过程传回表格结果与讯息。

 

除了通过输出参数传回数据之外,我们还可以使用 SqlPipe 对象将表格结果与讯息传回给客户端。一般来说,我们是通过 SqlContext 类别的Pipe 属性取得 SqlPipe 对象,然后呼叫 SqlPipe 对象的 Send 方法将表格结果或讯息传送给客户端。

 

SqlPipe 对象的 Send 方法拥有下列三个多载版本:

 

q          SqlPipe.Send(message As String)

此多载版本的 Send 方法能够将字符串讯息直接传送至客户端。文字讯息最长不可超过 8000 个字符,超出 8000 个字符的部分将会被截掉。

讯息字符串会被当作一个信息讯息传送给客户端,至于客户端要通过哪一种机制来聆听此讯息,则视客户端所采用的数据存取应用程序接口(API)而定。比方说,假设客户端使用 ADO.NET,则讯息会通过事件 InfoMessage 来传送。

 

q          SqlPipe.Send(record As SqlDataRecord)

此多载版本的 Send 方法能够将单一数据列结果集(也就是一个SqlDataRecord 对象)直接传送至客户端。如果您于 Send 方法中所使用的 SqlDataReader 对象拥有隐藏字段,这些字段将不会填入传送给客户端的结果集当中。

以前一篇文章所建立的CLR存储过程 SayHello 而言,它除了使用输出参数来传回文字数据之外,还呼叫此多载版本的 Send 方法将单一数据列结果集直接传送至客户端,如下所示:

SqlContext.Pipe.Send(greetingRecord)

q          SqlPipe.Send(reader As SqlDataReader)

此多载版本的 Send 方法能够将一个多数据列结果集(也就是一个SqlDataReader 对象)直接传送至客户端。

如果您要传送先前通过同处理序(In-Process)提供者所执行的查询结果,或是想要使用 SqlDataReader 的自订实作来预先处理数据,请使此多载版本的 Send 方法来接收一个 SqlDataReader 对象并将其直接传送给客户端。虽然 Send 方法的执行效率稍稍慢于ExecuteAndSend 方法,不过在数据被传送至客户端之前,它提供了最大的弹性来处理数据。

 

请大家注意,除了使用 Send 方法之外,CLR 存储过程还可以使用SqlPipe 对象的 ExecuteAndSend 方法将查询结果传送给客户端。ExecuteAndSend 方法的最大好处,就是它提供了一种最高效率的方式将查询结果传送给客户端,之所以如此,是因为数据是通过网络缓冲区来传送,而不需被复制到受管理的内存中。

 

以下是 ExecuteAndSend 方法的语法:

 

SqlPipi.ExecuteAndSend (command As SqlCommand)

 

从语法可以看出,ExecuteAndSend 方法其实是以一个 SqlCommand对象作为其参数,它会执行 SqlCommand 对象并将结果传送至客户端。大家必须了解,除了所有实际的执行结果之外,其它的讯息与错误也会直接传送给客户端。

 

程序范例

 

以下我们示范如何使用 SqlPipe 对象的 Send 方法将文字符串直接传送至客户端:

 

q          下面这一个CLR存储过程存放于类别库项目 GiveYouString 中,它示范如何使用 SqlPipe 对象的 Send 方法将文字符串直接传送至客户端:

namespace GiveYouString
{
 public class GiveYouStringClass
 {
  [Microsoft.SqlServer.Server.SqlProcedure()]
  public static void GiveYouString()
  {
   // 呼叫 Pipe 对象的 Send 方法将文字符串直接传送给客户端。
   SqlContext.Pipe.Send("章立民研究室");
  }
 }
}

请注意,这一个CLR存储过程并没有任何参数。

q          请执行如下所示之 SQL 指令码,登录组件 GiveYouString.dll、建立一个参考所登录之组件 GiveYouString.dll 的存储过程、然后执行此CLR 存储过程:

USE 北风贸易;
GO

IF EXISTS (SELECT * FROM sys.procedures WHERE [name] =
'GiveYouString')
DROP PROCEDURE GiveYouString;
GO

IF EXISTS (SELECT * FROM sys.assemblies WHERE [name] =
'GiveYouString')
DROP ASSEMBLY GiveYouString;
GO

-- 登录组件GiveYouString.dll
CREATE ASSEMBLY GiveYouString
FROM 'C:\SQL2005Demo\CH13\GiveYouString\GiveYouString\bin\GiveYouString.dll'
WITH permission_set = Safe;
GO

-- 建立一个会参考所登录之组件GiveYouString.dll的存储过程
CREATE PROCEDURE dbo.GiveYouString
AS EXTERNAL NAME GiveYouString.[GiveYouString.GiveYouStringClass].GiveYouString;
GO

EXEC sp_configure 'clr enabled', '1';
GO
RECONFIGURE;
GO

DECLARE @return_value int

-- 执行CLR存储过程
EXECUTE @return_value = dbo.GiveYouString;

q          在您执行了上述 SQL 指令后将会发现, 执行CLR 存储过程GiveYouString 之后并无法取得所传回的文字符串。事实上我们之前就曾经提及,SqlPipe 对象之 Send 方法所传回的讯息字符串会被当作一个信息讯息传送给客户端,至于客户端要通过哪一种机制来聆听此讯息,则视客户端所采用的数据存取应用程序接口(API)而定。如果客户端使用 ADO.NET,则讯息会通过事件 InfoMessage来传送。

为了示范如何在前端应用程序中通过 ADO.NET 取得 SqlPipe 对象之 Send 方法所传回的讯息字符串,我们特别设计了一个Windows应用程序 TakeReturnString 来加以实作。从图表1的执行画面可以看出,我们果然能够顺利取得 CLR 存储过程使用 SqlPipe 对象之 Send 方法所传回的文字符串。

相关的前端程序代码列示如下:

private void btnRun_Click(object sender, EventArgs e)
{
 // 利用 SqlConnectionStringBuilder 对象来构建连接字符串。
 SqlConnectionStringBuilder connectStringBuilder = new
   SqlConnectionStringBuilder();
 connectStringBuilder.DataSource = @"(local)";
 connectStringBuilder.InitialCatalog = "北风贸易";
 connectStringBuilder.IntegratedSecurity = true;
 
 try
 {
  using(SqlConnection northwindConnection = new
    SqlConnection(connectStringBuilder.ConnectionString))
  {
   northwindConnection.InfoMessage += new
     SqlInfoMessageEventHandler(OnInfoMessage);
  
   northwindConnection.Open();
  
   SqlCommand cmdLiming = new SqlCommand("dbo.GiveYouString",
     northwindConnection);
  
   cmdLiming.CommandType = CommandType.StoredProcedure;
   cmdLiming.ExecuteNonQuery();
  }
 }
 catch(Exception ex)
 {
  MessageBox.Show(ex.Message);
 }
}

private static void OnInfoMessage(object sender,
  SqlInfoMessageEventArgs args)
{
 // 显示 CLR 存储过程利用 SqlPipe.Send 方法所传回的文字符串。
 MessageBox.Show(args.Message);
}


图表1

 

参考书籍:
SQL Server 2005 完全实战》
Visual C# 2005 文件 IO 与资料存取秘诀》

posted on 2007-01-15 16:37 章立民研究室 阅读(2447) 评论(3)  编辑 收藏 网摘 所属分类: SQL Server 2005

评论

#1楼 2007-01-15 21:02 sharewind      

章立民老师:
 您好!
 文章算是看得懂,就是不知道在什么场合该使用CLR来写存储过程.
 具体应用在哪里? 谢谢!
  回复  引用  查看    

#2楼[楼主] 2007-01-18 10:35 章立民研究室      

SQL Server 2005在整体开发功能上最大的变革与演进,就是能够与Microsoft .NET Framework Common Language Runtime(CLR)紧密整合,在实际作业上的具体呈现,就是您可以使用Visual Basic 2005与Visual C# 2005等 .NET 程序语言来开发出数据库对象。所有的开发与侦错工作都是在Microsoft Visual Studio 2005的整合式开发环境中进行,使得开发人员可以利用相同的工具来开发出能够存取 .NET 组件与服务的数据库对象。这样的作法,得以弥补Transact-SQL语言仅专注于数据库相关层面的缺憾,并以一种更稳健的方式来扩充SQL Server 2005的功能。

举例来说,假设您想要在SQL Server 2005中撰写一个Stored Procedure来读取与写入数据库字段中的二进制影像数据,就应该建立一个CLR存储过程并使用Visual Basic 2005或Visual C# 2005来撰写相关的程序代码。亦或是您想要利用ADO.NET 2.0的独有功能来进行数据库存取作业,也可以使用CLR存储过程来达成。

章立民 Alex Chang

@sharewind
  回复  引用  查看    

#3楼 2007-03-13 18:07 在北京的湖南人      

SqlDataReader 对象拥有隐藏字段
章老师您提到的整个隐藏字段是什么意思呢?不清楚,请指教!
  回复  引用  查看    

导航

公告


章立民
六度当选MVP
七十本著作酿成
十九载作者生涯
现在有了章立民研究室
依旧陶醉在写作的生活里。

统计

与我联系

搜索

 

常用链接

留言簿

随笔分类(111)

随笔档案(112)

光盘勘误下载

最新随笔

积分与排名

最新评论

阅读排行榜

评论排行榜