SQL Server中的CLR編程——用.NET爲SQL Server編寫存儲過程和函數

很早就知道能夠用.NET爲SQL Server2005及以上版本編寫存儲過程、觸發器和存儲過程的,不過以前開發的系統要麼由於歷史緣由用的是SQL2000要麼根本用不着在SQL Server中啓用CLR,因此一直沒有嘗試。最近由於項目的緣由,在這方面作了一個調研,如今在這裏分享一下心得。
首先要說明的是要在SQL Server中啓用CLR必須是在SQL Server2005及以上版本,其次在默認狀況下是沒有啓用CLR的,必需要顯示設置爲啓用。好比咱們要在ArticleCollectorDB數據庫中運行用.NET編寫的函數或者存儲過程,至少先要進行下面的SQL語句:html

  
  
  
  
  1. exec sp_configure 'clr enabled', 1;--在SQL Server中啓用CLR 
  2. reconfigure; 
  3. go 
  4. --在ArticleCollectorDB數據庫中設置TRUSTWORTHY爲ON 
  5. ALTER DATABASE [ArticleCollectorDB] SET TRUSTWORTHY ON 


這時可能會獲得提示要從新啓動SQL Server,若是有此提示則從新啓動一下。
接着咱們在VS中進行編碼,在這裏咱們將分別編寫一個名爲IsMatch的函數和一個名爲SendMail存儲過程。在VS中建立一個名爲NetSkycn.Data的類庫項目,添加一個SqlCLR的類,代碼以下:web

  
  
  
  
  1. using System.Data.SqlTypes; 
  2. using System.Net; 
  3. using System.Net.Mail; 
  4. using System.Security.Permissions; 
  5. using System.Text.RegularExpressions; 
  6. using Microsoft.SqlServer.Server; 
  7.  
  8. namespace NetSkycn.Data 
  9. /// <summary> 
  10. /// 在SQL Server環境中執行的CLR方法,注意提供給SQL Server調用的方法必須有SqlFunction/SqlProcedure Attribute 
  11. /// 做者:周公 
  12. /// 建立日期:2012-05-09 
  13. /// 博客地址:http://blog.csdn.net/zhoufoxcn http://zhoufoxcn.blog.51cto.com 
  14. /// 新浪微博地址:http://weibo.com/zhoufoxcn 
  15. /// </summary> 
  16. public sealed class SqlCLR 
  17. /// <summary> 
  18. /// 判斷字符串是否匹配正則表達式 
  19. /// </summary> 
  20. /// <param name="source">要匹配的文本</param> 
  21. /// <param name="pattern">進行匹配的正則表達式</param> 
  22. /// <param name="options">正則表達式匹配選項,1爲忽略大小寫,2爲多行匹配,3爲忽略大小寫且多行匹配</param> 
  23. /// <returns></returns> 
  24. [SqlFunction(IsDeterministic = true, DataAccess = DataAccessKind.None)] 
  25. public static SqlBoolean IsMatch(string source, string pattern,int options) 
  26. if (string.IsNullOrEmpty(source) || string.IsNullOrEmpty(pattern)) 
  27. return SqlBoolean.False; 
  28. RegexOptions regexOptions=RegexOptions.None; 
  29. int optionIgnoreCase = 1; 
  30. int optionMultiline = 2; 
  31. if ((options & optionIgnoreCase) != 0) 
  32. regexOptions = regexOptions | RegexOptions.IgnoreCase; 
  33.  
  34. if ((options & optionMultiline) != 0) 
  35. regexOptions = regexOptions | RegexOptions.Multiline; 
  36.  
  37. return (SqlBoolean)(Regex.IsMatch(source, pattern, regexOptions)); 
  38. /// <summary> 
  39. /// 發送郵件 
  40. /// </summary> 
  41. /// <param name="to">收件人郵件地址</param> 
  42. /// <param name="from">發件人郵件地址</param> 
  43. /// <param name="subject">郵件主題</param> 
  44. /// <param name="body">郵件內容</param> 
  45. /// <param name="username">登陸smtp主機時用到的用戶名,注意是郵件地址'@'之前的部分</param> 
  46. /// <param name="password">登陸smtp主機時用到的用戶密碼</param> 
  47. /// <param name="smtpHost">發送郵件用到的smtp主機</param> 
  48. [SqlProcedure] 
  49. [SmtpPermission(SecurityAction.Assert)] 
  50. [SecurityPermission(SecurityAction.Assert)] 
  51. public static void SendMail(string to, string from, string subject, string body, string userName, string password, string smtpHost) 
  52. MailAddress addressFrom = new MailAddress(from); 
  53. MailAddress addressTo = new MailAddress(to); 
  54. MailMessage message = new MailMessage(addressFrom, addressTo); 
  55. message.Subject = subject;//設置郵件主題 
  56. message.IsBodyHtml = true;//設置郵件正文爲html格式 
  57. message.Body = body;//設置郵件內容 
  58.  
  59. SmtpClient client = new SmtpClient(smtpHost); 
  60.  
  61. //設置發送郵件身份驗證方式 
  62. //注意若是發件人地址是abc@def.com,則用戶名是abc而不是abc@def.com 
  63. client.Credentials = new NetworkCredential(userName, password); 
  64. client.Send(message); 
  65.  


編譯經過以後,記住類庫的物理全路徑,好比:F:\VS2008\netskycn\NetSkycn.Data\bin\Release\NetSkycn.Data.dll,在這裏要強調幾點:1、對於未來提供給SQL Server調用的函數或者存儲過程必須是靜態方法,而且還必須帶有SqlFunction或者SqlProcedure屬性;2、對於一些須要訪問外部網絡資源和安全屬性的還必須添加響應的屬性(如本例中的SendMail方法,若是沒有添加響應的屬性在建立SQL Function/Procedure時會出現錯誤提示)。
如今咱們開始遵循先爲SQL Server建立程序集、後建立函數或者存儲過程的順序來操做,在操做過程當中用到的SQL語句以下:正則表達式

  
  
  
  
  1. --在ArticleCollectorDB數據庫中設置TRUSTWORTHY爲ON 
  2. ALTER DATABASE [ArticleCollectorDB] SET TRUSTWORTHY ON 
  3.  
  4. --若是已經存在該對象則刪除 
  5. IF EXISTS(SELECT * FROM SYS.SYSOBJECTS WHERE NAME='SendMail' AND XTYPE='PC')  
  6. DROP PROCEDURE SendMail 
  7.  
  8. --若是已經存在該對象則刪除 
  9. IF EXISTS(SELECT * FROM SYS.SYSOBJECTS WHERE NAME='IsMatch' AND XTYPE='FS')  
  10. DROP FUNCTION IsMatch 
  11.  
  12. --若是已經存在SqlCLR程序集則刪除該程序集 
  13. IF EXISTS(SELECT * FROM SYS.ASSEMBLIES WHERE NAME='SqlCLR')  
  14. DROP ASSEMBLY SqlCLR 
  15.  
  16. --在SQL Server中建立程序集,,建立的程序集名爲SqlCLR 
  17. CREATE ASSEMBLY SqlCLR FROM 'F:\VS2008\netskycn\NetSkycn.Data\bin\Release\NetSkycn.Data.dll' WITH PERMISSION_SET = UNSAFE 
  18. GO 
  19.  
  20. --從CLR程序集中建立函數,函數名爲IsMatch,有三個參數, 
  21. --[SqlCLR]是SQL Server中程序集名 
  22. --[NetSkycn.Data.SqlCLR]是.NET中的類的全名(命名空間及類名) 
  23. --[IsMatch]是.NET中類的函數名 
  24. CREATE FUNCTION [dbo].[IsMatch]  
  25. (  
  26. @source AS NVARCHAR(200), 
  27. @pattern AS NVARCHAR(200), 
  28. @option INT=3 
  29. )  
  30. RETURNS BIT  
  31. AS  
  32. EXTERNAL NAME [SqlCLR].[NetSkycn.Data.SqlCLR].[IsMatch];  
  33. GO 
  34.  
  35. --從CLR程序集中建立函數,函數名爲IsMatch,有三個參數, 
  36. --[SqlCLR]是SQL Server中程序集名 
  37. --[NetSkycn.Data.SqlCLR]是.NET中的類的全名(命名空間及類名) 
  38. --[SendMail]是.NET中類的函數名 
  39. CREATE PROCEDURE [dbo].[SendMail]  
  40. (  
  41. @to AS NVARCHAR(200), 
  42. @from AS NVARCHAR(200), 
  43. @subject AS NVARCHAR(200), 
  44. @body AS NVARCHAR(MAX), 
  45. @userName AS NVARCHAR(200), 
  46. @password AS NVARCHAR(200), 
  47. @smtpHost AS NVARCHAR(200) 
  48. AS  
  49. EXTERNAL NAME [SqlCLR].[NetSkycn.Data.SqlCLR].[SendMail];  
  50. GO 


若是沒有獲得任何錯誤提示,則表示建立函數和存儲過程成功。至此咱們會看到以下情形:
sql


這表示建立成功。
測試建立函數的SQL語句(查找article表中title字段是3至5個字段的數據):數據庫

  
  
  
  
  1. select * from article where dbo.IsMatch(Title,'^[\u4e00-\u9fa5]{3,5}$',3)=1 


測試建立存儲過程的SQL語句:編程

  
  
  
  
  1. exec [dbo].SendMail @to='test@qq.com',@from='webmaster@qq.com',@subject='test',@body='This mail was sent by SQL Procedure',@userName='webmaster',@password='123',@smtpHost='smtp.qq.com' 


以上代碼在SQL Server 2005中文企業版、SQL Server 2008英文企業版測試經過。
能夠看出在一些SQL語句不夠靈活的狀況下,可使用.NET來編寫存儲過程和函數,經過以上步驟以後和調用SQL語句寫的存儲過程和函數沒有區別,極大地方便了編程。

周公
2012-05-12安全

相關文章
相關標籤/搜索