通过CLR同步SQL Server和Sharepoint List数据(二)

本文涉及的产品
云数据库 RDS SQL Server,基础系列 2核4GB
RDS SQL Server Serverless,2-4RCU 50GB 3个月
推荐场景:
简介:

写在前面

    本系列文章一共分为四部分:

        1. CLR概述。

        2. 在Visual Studio中进行CLR集成编程并部署到SQL Server,包括存储过程、触发器、自定义函数、自定义类型和聚合。

        3. CLR集成编程的调试和所遇到的问题。

        4. 利用CLR同步SQL Server表和Sharepoint List(来源于实际项目应用)。

 

    本系列文章建立在以下软件环境的基础上:

  • Windows Server 2003 Enterprise Edition Service Pack 2
  • Microsoft Visual Studio Team System 2008
  • Microsoft SQL Server 2008
  • Microsoft Office Sharepoint Server 2007

 

准备工作

    默认情况下SQL Server对CLR的允许状态是关闭的,我们需要在查询分析器中打开它,将CLR设置为允许,这样该服务器将对所有数据库实例开放CLR执行条件。可以按照下面的步骤进行:

3-20-2009 5-10-59 PM

Exec  sp_configure  ' clr enabled '
        该系统存储过程用于查看当前CLR的状态,返回的结果中如果config_value字段的值为0则表示禁止CLR在该服务器上运行,值为1则为允许。增加第二个参数到该存储过程,然后再执行。
Exec  sp_configure  ' clr enabled ' 1 -- 1 enabled, 0 disabled
        查询分析器的消息窗口中给出提示: Configuration option 'clr enabled' changed from 0 to 1. Run the RECONFIGURE statement to install.  按提示运行reconfigure命令进行安装。
reconfigure   with  override
        现在SQL Server服务器已经允许执行CLR托管代码了,接下来我们只需要在Visual Studio中编写代码,将生成的dll部署到SQL Server中即可。按下图,创建SQL Server Project并设置好数据库连接,然后就可以编写用户自定义类型、存储过程、用户自定义函数、聚合和触发器了。

3-19-2009 2-23-30 PM

3-19-2009 2-27-35 PM

开始

一个简单的自定义函数

    现在我们已经可以在Visual Studio中开始CLR项目了,在Solution Explorer中右键单击项目文件,点击Add,在其中选择你所要添加的类型。我们从最简单的类型开始,选中User-Defined Function…,取名为DateTimePre.cs,该自定义函数主要实现在用户给定的字符串数据前加上系统当前时间前缀,编写代码如下。

3-19-2009 2-47-15 PM

复制代码
1  using  System; 
2  using  System.Data; 
3  using  System.Data.SqlClient; 
4  using  System.Data.SqlTypes; 
5  using  Microsoft.SqlServer.Server; 
6   
7  public   partial   class  UserDefinedFunctions 
8 
9      [Microsoft.SqlServer.Server.SqlFunction] 
10       public   static  SqlString DateTimePre( string  input,  string  format) 
11      { 
12           string  sRst  =   string .Format( " {0}:{1} " , DateTime.Now.ToString(format), input); 
13           return   new  SqlString(sRst); 
14      } 
15  };
复制代码
        代码很简单,就是在用户给定的文本前加上当前时间前缀,第二个参数用于指定时间显示的样式。先说明一下代码的结构。SQLCLR中定义的类(用户自定义的类除外)都以partial关键字开头,表示它是一个分部类,这个我们不需要去改它,默认的类名可以修改,同一个类中可以有多个自定义的类型(自定义函数、触发器、存储过程等),但都必须用特征属性显示指明类型,如SqlFunction、SqlProcedure、SqlTrigger、SqlUserDefinedType等,这些类型必须定义为public类型的而且必须为static,以向SQL Server完全公开,有些类型必须要有返回值,如自定义函数,如果不需要类型返回值,则可以定义为存储过程类型,这个后面再举例。

    编译代码,部署到SQL Server服务器,这个服务器的地址取决于你在Visual Studio中所指定的数据库服务器,就是我们在创建项目前所指定的那个数据库服务器。在Solution Explorer中右键单击项目文件,点击Deploy,Visual Studio会自动为你进行程序集的部署。因为我们的这个示例很简单,也没有涉及到访问外部资源的代码,所以不用考虑程序集的访问级别和安全性,让Visual Studio按照默认的情况自动进行就可以了。我会在后面专门介绍如何设置程序集的访问级别以及其中遇到的问题。

    部署成功后转到SQL Server Management Studio,打开你所连接的数据库,依次选择Programmability—Functions—Scalar-valued Functions,在下面可以找到我们刚创建的这个类型,表示一切顺利!

3-19-2009 3-31-50 PM    接下来我们在查询分析器中执行它,看一下执行结果,OK。一个简单的自定义函数就完成了,因为有Visual Studio的帮助,使得SQLCLR的编写变得非常简单,否则,你需要在命令行方式下手动进行部署,这个过程比较复杂,也不是本文的重点。

Select  dbo.DateTimePre( ' Begin ' '' )

3-19-2009 3-34-14 PM

流式表值函数(TVF)的实现

    前面我在介绍CLR集成性能的时候提到了流式表值函数(TVF),它返回IEnumerable接口的托管函数,通过特征属性指定返回结果的样式和定义方法,将结果以数据表的形式在SQL Server的查询分析器中输出。它的性能将优于在SQL Server查询分析器中使用扩展存储过程的性能。

    在刚才创建的class文件中再写一个方法,用于实现流式表值函数。示例给出的是在用户指定的位置搜索出用户指定类型的所有文件,然后以规定的表格样式将结果在SQL Server的查询分析器中输出。

复制代码
1  [SqlFunction(FillRowMethodName  =   " BuildRow "
2       TableDefinition  =   " Name nvarchar(32), Length bigint, Modified DateTime " )] 
3       public   static  IEnumerable FileListCs( string  directoryName,  string  pattern) 
4      { 
5          FileInfo[] files; 
6           // 模拟当前SQL安全上下文  
7          WindowsImpersonationContext contect  =  SqlContext.WindowsIdentity.Impersonate(); 
8           try  
9          { 
10              DirectoryInfo di  =   new  DirectoryInfo(directoryName); 
11              files  =  di.GetFiles(pattern); 
12          } 
13           finally  
14          { 
15               if  (contect  !=   null
16              { 
17                  contect.Undo(); 
18              } 
19          } 
20           return  files; 
21      } 
22   
23       private   static   void  BuildRow( object  Obj, 
24             ref  SqlString fileName, 
25             ref  SqlInt64 fileLength, 
26             ref  SqlDateTime fileModified) 
27      { 
28           if  (Obj  !=   null
29          { 
30              FileInfo file  =  (FileInfo)Obj; 
31              fileName  =  file.Name; 
32              fileLength  =  file.Length; 
33              fileModified  =  file.LastWriteTime; 
34          } 
35           else  
36          { 
37              fileName  =  SqlString.Null; 
38              fileLength  =  SqlInt64.Null; 
39              fileModified  =  SqlDateTime.Null; 
40          } 
41      }
复制代码

     对代码的说明。特征属性中的FillRowMethodName用来告诉该函数,输出的结果需要用它指定的函数进行格式化,而格式化的样式(即Table的定义)由TableDefinition的值指定。这也就是说,我们需要自己编写FillRowMethodName所指定的函数,即代码中的BuildRow方法。该方法有一个输入型参数Obj,用于接收流式表值函数的返回值,另外几个ref型的参数必须与TableDefinition所规定的内容一致,并且必须是ref类型的参数。经过这样的规定,流式表值函数就会按照我们定义好的表结构在SQL Server的查询分析器中输出结果。WindowsImpersonationContext对象用于在SQL Server中模拟当前安全上下文,以SQL Server进程的身份执行程序代码,在访问外部资源时(尤其是网络资源),如果当前SQL Server没有访问权限,则依附于它的CLR程序集的访问也会失败。注意在对CLR进行安全上下文模拟之后必须进行Undo操作!

    有一个前提条件需要说明一下。前面我在CLR集成安全性中提到,当CLR的程序集需要访问外部资源时(例如我们这里所访问的磁盘文件),需要设置程序集的访问级别为External,而且需要将宿主数据库的权限级别设置为EXTERNAL_ACCESS,否则SQL Server的查询分析器会提示错误。

Use  Test 
Go  
Alter   Database  Test  Set  trustworthy  On  
Go

3-19-2009 5-14-39 PM     编译、部署,转到SQL Server查询分析器中,在Programmability—Functions—Table valued Functions下可以看到我们刚创建的流式表值函数。执行它!

3-19-2009 5-18-53 PM

Select   *   From  Test.dbo.FileListCs( ' D:\TreeView ' ' *.* ' )

    出现了错误!原来我们在执行流式表值函数的特征属性时少了一个DataAccess,下面补上。再运行,函数给出了正确的结果。

Msg 6522, Level 16, State 2, Line 1 
A .NET Framework error occurred during execution of user-defined routine or aggregate "FileListCs": 
System.InvalidOperationException: Data access is not allowed in this context.  Either the context is a function or method not marked with DataAccessKind.Read or SystemDataAccessKind.Read, is a callback to obtain data from FillRow method of a Table Valued Function, or is a UDT validation method. 
System.InvalidOperationException: 
   at System.Data.SqlServer.Internal.ClrLevelContext.CheckSqlAccessReturnCode(SqlAccessApiReturnCode eRc) 
   at System.Data.SqlServer.Internal.ClrLevelContext.GetCurrentContext(SmiEventSink sink, Boolean throwIfNotASqlClrThread, Boolean fAllowImpersonation) 
   at Microsoft.SqlServer.Server.InProcLink.GetCurrentContext(SmiEventSink eventSink) 
   at Microsoft.SqlServer.Server.SmiContextFactory.GetCurrentContext() 
   at Microsoft.SqlServer.Server.SqlContext.get_CurrentContext() 
   at Microsoft.SqlServer.Server.SqlContext.get_WindowsIdentity() 
   at UserDefinedFunctions.FileListCs(String directoryName, String pattern) 
.

[SqlFunction(DataAccess  =  DataAccessKind.Read, 
        FillRowMethodName 
=   " BuildRow "
        TableDefinition 
=   " Name nvarchar(32), Length bigint, Modified DateTime " )]

3-19-2009 5-28-40 PM

 

存储过程的实现

    在Visual Studio中重新创建一个类型为Stored Procedure的class,编写代码如下。

复制代码
1  [Microsoft.SqlServer.Server.SqlProcedure] 
2       public   static   void  GetData( string  tbName) 
3      { 
4           using  (SqlConnection cn  =   new  SqlConnection( " context connection=true " )) 
5          { 
6               using  (SqlCommand cmd  =  cn.CreateCommand()) 
7              { 
8                  cmd.CommandText  =   string .Format( " Select * from {0} " , tbName); 
9                  cn.Open(); 
10                  SqlContext.Pipe.Send(cmd.ExecuteReader()); 
11              } 
12          } 
13      }
复制代码

     按照我在前面CLR集成安全性中介绍的,CLR在EXTERNAL_ACCESS访问模式下默认以SQL Server当前的服务账户运行,所以我们可以利用当前上下文来获取数据库连接字符串,并进行相关数据处理(如输出Message,访问站点等)。SqlContext.Pipe.Send方法用于在当前上下文中输出结果到SQL Server查询分析器的Results窗口中,它可以接收SqlDataReader、SqlDataRecord和string类型的参数,同时也可以SqlContext.Pipe.ExecuteAndSend(SqlCommand)这样来用。

    编译、部署,转到SQL Server查询分析器中,刚刚编写的存储过程出现在Programmability—Stored Procedure下,直接调用该存储过程,得到结果。

3-19-2009 5-57-07 PM

 

触发器的实现

    触发器的实现比较简单,主要还是方法前的特征属性需要描述清楚,这里给出一个示例,当对表Area进行更新操作的时候会在表Region中更新相应的值,相关截图和代码如下。

3-20-2009 10-17-05 AM   3-20-2009 10-18-08 AM
表Area(修改前)   表Region(修改前)
复制代码
1  [Microsoft.SqlServer.Server.SqlTrigger(Name  =   " Triggers " , Target  =   " Area " , Event  =   " FOR UPDATE " )] 
2       public   static   void  TriggersTest() 
3      { 
4           string  comText  =   @"  Declare @oldTitle varchar(50) 
5                              Declare @newTitle varchar(50) 
6                                                      
7                              Select @oldTitle = Title From Deleted 
8                              Select @newTitle = Title From Inserted 
9   
10                              Update Region Set Area = @newTitle Where Area = @oldTitle "
11   
12           using  (SqlConnection cn  =   new  SqlConnection()) 
13          { 
14              cn.ConnectionString  =   " context connection=true "
15              cn.Open(); 
16               using  (SqlCommand cmd  =  cn.CreateCommand()) 
17              { 
18                  cmd.CommandText  =  comText; 
19                  SqlContext.Pipe.ExecuteAndSend(cmd); 
20              } 
21          } 
22      }
复制代码
         编译并部署,触发器出现在相关表下面的Triggers目录下,并且该触发器的图标上有一个小锁,表示该触发器是由CLR生成的,并且不能被修改。现在update表Area的一条数据,会发现Region表中相关的记录也发生了变化,表示触发器已经生效了。
Update  Area  Set  Title  =   ' APAC '   Where  Title  =   ' APAC1 '
3-20-2009 10-34-18 AM   3-20-2009 10-34-45 AM
表Area(修改后)   表Region(修改后)

 

用户自定义类型的实现

    用户自定义类型理解起来可能稍微有点复杂,在实际应用当中可能也很少用到,我在这里直接引用了MSDN上的一个例子,了解一下。实际上,当我们在Visual Studio上创建一个UserDefinedType时,IDE已经为我们做了很多事情了,剩下的工作只需要以填空的方式完善代码即可,这样说来,实际上也不是那么复杂啊,至少Visual Studio在为我们搭建好的代码结构中已经有了不少注释,我们应该知道怎么去做。

    这个示例中定义了一个UserDefinedType类型,用户使用的时候可以给定一个字符串值,然后通过内部的转换显示出相应的信息。如给定1:2,则表示right top,给定-1:3,则表示left top等,内部的转换规则需要自己去实现,当然,你也可以实现更加复杂的结构。

复制代码
  1  [Serializable] 
  2  [Microsoft.SqlServer.Server.SqlUserDefinedType(Format.Native)] 
  3  public   struct  UserDefinedType : INullable 
  4 
  5       public   override   string  ToString() 
  6      { 
  7           if  ( this .IsNull) 
  8          { 
  9               return   " NULL "
10          } 
11           else  
12          { 
13               return   this .m_x  +   " : "   +   this .m_y; 
14          } 
15      } 
16   
17       public   bool  IsNull 
18      { 
19           get  {  return   this .m_Null; } 
20           set  {  this .m_Null  =  value; } 
21      } 
22   
23       public   static  UserDefinedType Null 
24      { 
25           get  
26          { 
27              UserDefinedType h  =   new  UserDefinedType(); 
28              h.m_Null  =   true
29               return  h; 
30          } 
31      } 
32   
33       public   static  UserDefinedType Parse(SqlString s) 
34      { 
35           if  (s.IsNull) 
36               return  Null; 
37   
38           string  str  =  Convert.ToString(s); 
39           string [] xy  =  str.Split( ' : ' ); 
40   
41          UserDefinedType u  =   new  UserDefinedType(); 
42          u.X  =  Convert.ToInt32(xy[ 0 ]); 
43          u.Y  =  Convert.ToInt32(xy[ 1 ]); 
44   
45           return  u; 
46      } 
47   
48       public  SqlString Quadrant() 
49      { 
50           if  (m_x  ==   0   &&  m_y  ==   0
51          { 
52               return   " centered "
53          } 
54   
55          SqlString stringReturn  =   ""
56   
57           if  (m_x  ==   0
58          { 
59              stringReturn  =   " center "
60          } 
61           else   if  (m_x  >   0
62          { 
63              stringReturn  =   " right "
64          } 
65           else   if  (m_x  <   0
66          { 
67              stringReturn  =   " left "
68          } 
69   
70           if  (m_y  ==   0
71          { 
72              stringReturn  =  stringReturn  +   "  center "
73          } 
74           else   if  (m_y  >   0
75          { 
76              stringReturn  =  stringReturn  +   "  top "
77          } 
78           else   if  (m_y  <   0
79          { 
80              stringReturn  =  stringReturn  +   "  bottom "
81          } 
82   
83           return  stringReturn; 
84      } 
85   
86       //  This is a place-holder field member  
87       public   int  X  
88      { 
89           get  {  return   this .m_x; } 
90           set  {  this .m_x  =  value; } 
91      } 
92   
93       public   int  Y 
94      { 
95           get  {  return   this .m_y; } 
96           set  {  this .m_y  =  value; } 
97      } 
98   
99       //  Private member  
100       private   int  m_x; 
101       private   int  m_y; 
102       private   bool  m_Null; 
103  }
复制代码

 

    编译部署,在SQL Server的查询分析器中打开Programmability—Types—User-Defined Types,可以看到刚创建的类型,执行后可以看到结果。

复制代码
CREATE   TABLE  test_table (column1 UserDefinedType) 
go  

INSERT   INTO  test_table (column1)  VALUES  ( ' 1:2 '
INSERT   INTO  test_table (column1)  VALUES  ( ' -2:3 '
INSERT   INTO  test_table (column1)  VALUES  ( ' -3:-4 '

Select  column1.Quadrant()  From  test_table
复制代码

3-20-2009 12-56-05 PM

 

聚合的实现

    与自定义类型类似,在Visual Studio中创建聚合(Aggreagte),IDE同样已经为我们做好了准备。这里我也给出一个MSDN上的例子(不过对于嵌套循环稍做了修改),用来在聚合中计算字符串数据类型中的原音字符的个数。

复制代码
1  [Serializable] 
2  [Microsoft.SqlServer.Server.SqlUserDefinedAggregate(Format.Native)] 
3  public   struct  Aggregate 
4 
5       public   void  Init() 
6      { 
7          countOfVowels  =   0
8      } 
9   
10       public   void  Accumulate(SqlString value) 
11      { 
12           //  list of vowels to look for  
13          List < string >  vowels  =   new  List < string > (); 
14          vowels.Add( " a " ); 
15          vowels.Add( " e " ); 
16          vowels.Add( " i " ); 
17          vowels.Add( " o " ); 
18          vowels.Add( " u " ); 
19   
20           //  for each character in the given parameter  
21           for  ( int  i  =   0 ; i  <  value.ToString().Length; i ++
22          { 
23               if  (vowels.Contains(value.Value.Substring(i,  1 ).ToLower())) 
24              { 
25                   //  it is a vowel, increment the count  
26                  countOfVowels  +=   1
27              } 
28          } 
29      } 
30   
31       public   void  Merge(Aggregate value) 
32      { 
33          Accumulate(value.Terminate()); 
34      } 
35   
36       public  SqlString Terminate() 
37      { 
38           return  countOfVowels.ToString(); 
39      } 
40   
41       //  This is a place-holder member field  
42       private  SqlInt32 countOfVowels; 
43  }
复制代码

 

    编译部署,在SQL Server的查询分析器中打开Programmability—Functions—Aggreate Functions,可以看到刚创建的聚合函数,下面是执行后的结果。

3-20-2009 1-13-59 PM

 

结语

    CLR创建SQL Server对象应该还不止上面提到的这几种类型,但常用的基本都在这里了,尤其是存储过程、函数和触发器。利用C#来编写这些类型,灵活性更大,可操控性也更强了。下一篇我将会介绍如何在Visual Studio中进行CLR调试,对程序集的分发和手动部署,以及常见问题的解决办法。

示例代码下载

1 2 3

4



本文转自Jaxu博客园博客,原文链接:http://www.cnblogs.com/jaxu/archive/2009/03/20/1416634.html,如需转载请自行联系原作者


相关实践学习
使用SQL语句管理索引
本次实验主要介绍如何在RDS-SQLServer数据库中,使用SQL语句管理索引。
SQL Server on Linux入门教程
SQL Server数据库一直只提供Windows下的版本。2016年微软宣布推出可运行在Linux系统下的SQL Server数据库,该版本目前还是早期预览版本。本课程主要介绍SQLServer On Linux的基本知识。 相关的阿里云产品:云数据库RDS&nbsp;SQL Server版 RDS SQL Server不仅拥有高可用架构和任意时间点的数据恢复功能,强力支撑各种企业应用,同时也包含了微软的License费用,减少额外支出。 了解产品详情:&nbsp;https://www.aliyun.com/product/rds/sqlserver
相关文章
|
3月前
|
SQL
简单练习Microsoft SQL Server MERGE同步两个表
【10月更文挑战第13天】本文介绍了在Microsoft SQL Server中使用`MERGE`语句同步两个表的步骤。首先创建源表`SourceTable`和目标表`TargetTable`并分别插入数据,然后通过`MERGE`语句根据ID匹配行,实现更新、插入和删除操作,最后验证同步结果。此方法可根据需求调整以适应不同场景。
173 1
|
6月前
|
SQL 关系型数据库 索引
关系型数据库SQLserver插入数据
【7月更文挑战第28天】
65 4
|
2月前
|
SQL 开发框架 .NET
突破T-SQL限制:利用CLR集成扩展RDS SQL Server的功能边界
CLR集成为SQL Server提供了强大的扩展能力,突破了T-SQL的限制,极大地拓展了SQL 的应用场景,如:复杂字符串处理、高性能计算、图像处理、机器学习集成、自定义加密解密等,使开发人员能够利用 .NET Framework的丰富功能来处理复杂的数据库任务。
|
2月前
|
SQL 存储 缓存
SQL Server 数据太多如何优化
11种优化方案供你参考,优化 SQL Server 数据库性能得从多个方面着手,包括硬件配置、数据库结构、查询优化、索引管理、分区分表、并行处理等。通过合理的索引、查询优化、数据分区等技术,可以在数据量增大时保持较好的性能。同时,定期进行数据库维护和清理,保证数据库高效运行。
|
4月前
|
Python
Python量化炒股的获取数据函数— get_billboard_list()
Python量化炒股的获取数据函数— get_billboard_list()
55 0
|
4月前
|
Java API 开发者
代码小妙招:用Java轻松获取List交集数据
在Java中获取两个 `List`的交集可以通过 `retainAll`方法和Java 8引入的流操作来实现。使用 `retainAll`方法更为直接,但会修改原始 `List`的内容。而使用流则提供了不修改原始 `List`、更为灵活的处理方式。开发者可以根据具体的需求和场景,选择最适合的方法来实现。了解和掌握这些方法,能够帮助开发者在实际开发中更高效地处理集合相关的问题。
113 1
|
5月前
|
SQL 关系型数据库 MySQL
“震撼揭秘!Flink CDC如何轻松实现SQL Server到MySQL的实时数据同步?一招在手,数据无忧!”
【8月更文挑战第7天】随着大数据技术的发展,实时数据同步变得至关重要。Apache Flink作为高性能流处理框架,在实时数据处理领域扮演着核心角色。Flink CDC(Change Data Capture)组件的加入,使得数据同步更为高效。本文介绍如何使用Flink CDC实现从SQL Server到MySQL的实时数据同步,并提供示例代码。首先确保SQL Server启用了CDC功能,接着在Flink环境中引入相关连接器。通过定义源表与目标表,并执行简单的`INSERT INTO SELECT`语句,即可完成数据同步。
528 1
|
5月前
|
SQL 关系型数据库 数据库
|
5月前
|
Java 测试技术 容器
从零到英雄:Struts 2 最佳实践——你的Web应用开发超级变身指南!
【8月更文挑战第31天】《Struts 2 最佳实践:从设计到部署的全流程指南》深入介绍如何利用 Struts 2 框架从项目设计到部署的全流程。从初始化配置到采用 MVC 设计模式,再到性能优化与测试,本书详细讲解了如何构建高效、稳定的 Web 应用。通过最佳实践和代码示例,帮助读者掌握 Struts 2 的核心功能,并确保应用的安全性和可维护性。无论是在项目初期还是后期运维,本书都是不可或缺的参考指南。
59 0
|
5月前
|
SQL 存储 开发框架
Entity Framework Core 与 SQL Server 携手,高级查询技巧大揭秘!让你的数据操作更高效!
【8月更文挑战第31天】Entity Framework Core (EF Core) 是一个强大的对象关系映射(ORM)框架,尤其与 SQL Server 数据库结合使用时,提供了多种高级查询技巧,显著提升数据操作效率。它支持 LINQ 查询,使代码简洁易读;延迟加载与预先加载机制优化了相关实体的加载策略;通过 `FromSqlRaw` 或 `FromSqlInterpolated` 方法支持原始 SQL 查询;可调用存储过程执行复杂任务;利用 `Skip` 和 `Take` 实现分页查询,便于处理大量数据。这些特性共同提升了开发者的生产力和应用程序的性能。
261 0