<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
首先寫一段簡單的 C# 程式碼,然後把它編譯成 dll。
namespace Bussiness { public class UserFunctions { public static string UserLogin(string username, string password) { var random = new Random(); var isSuccess = random.Next() % 2 == 0; return isSuccess ? "登入成功" : "登入失敗"; } } }
接下來需要做的就是資料庫引數設定,開啟 CLR 支援,並且指定某個資料庫支援 unsafe
模式。
EXEC sp_configure 'clr enabled', 1; RECONFIGURE; GO ALTER DATABASE MyTestDB SET TRUSTWORTHY ON; GO
為了能夠調到 C# 的 UserLogin
方法,需要 SQLSERVER 先匯入這個程式集,然後再以 Function 對映其中方法即可
CREATE ASSEMBLY clr_Bussiness FROM 'D:Bussiness.dll' WITH PERMISSION_SET = UNSAFE; GO CREATE FUNCTION dbo.clr_UserLogin ( @username AS NVARCHAR(100), @password AS NVARCHAR(100) ) RETURNS NVARCHAR(100) AS EXTERNAL NAME clr_Bussiness.[Bussiness.UserFunctions].UserLogin; GO
建立完了之後,可以觀察 assembly
開頭的幾個系統檢視。
SELECT * FROM sys.assemblies SELECT * FROM sys.assembly_files; SELECT * FROM sys.assembly_modules;
接下來呼叫一下剛才建立的 clr_UserLogin
函數。
SELECT dbo.clr_UserLogin(N'jack',N'123456') AS 'State' GO 10
從圖中看登入結果是隨機的,說明 C# 的 Random 函數起到了作用。
從案例的執行結果看,推測在 SQLSERVER 中應該承載了一個 CLR 執行環境,那是不是這樣呢?可以用 WinDbg 附加到 sqlservr.exe
程序,用 lm觀察下模組載入情況。
0:092> lm start end module name ... 00007ff8`d3960000 00007ff8`d3aaf000 clrjit (deferred) 00007ff8`de040000 00007ff8`deb02000 clr (deferred) ... 0:092> !eeversion 4.8.4300.0 free Server mode with 12 gc heaps SOS Version: 4.8.4300.0 retail build
從輸出看果然載入了 clr
和 clrjit
動態連結庫,當前還是 gc server
模式。
接下來再驗證一個問題,既然 clr_UserLogin
函數會顯示 登入成功/登入失敗
,那必然會呼叫 C# 的 UserLogin
方法,可以在 WinDbg
中對 UserLogin
方法下一個斷點觀察一下這個呼叫過程
0:090> !name2ee Bussiness!Bussiness.UserFunctions.UserLogin Module: 00007ff87ee37988 Assembly: Bussiness, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null Token: 0000000006000001 MethodDesc: 00007ff87ee38020 Name: Bussiness.UserFunctions.UserLogin(System.String, System.String) JITTED Code Address: 00007ff87ec560d0 0:090> bp 00007ff87ec560d0 0:090> g
從輸出資訊看 UserLogin 方法已經被 JIT 過了,用 bp 下完斷點之後,繼續 g,然後在 SSMS 上再次執行查詢就可以成功命中啦。
0:090> k # Child-SP RetAddr Call Site 00 000000df`1557ae48 00007ff8`7ee500b6 0x00007ff8`7ec560d0 01 000000df`1557ae50 00007ff8`7ec55ef1 0x00007ff8`7ee500b6 02 000000df`1557aeb0 00007ff8`de04222e 0x00007ff8`7ec55ef1 03 000000df`1557af00 00007ff8`a2b79ff3 clr!UMThunkStub+0x6e 04 000000df`1557af90 00007ff8`a2b741bd sqllang!CallProtectorImpl::CallWithSEH<AppDomainCallTraits,void,FunctionCallBinder_3<void,void (__cdecl*)(void (__cdecl*)(void * __ptr64),void * __ptr64,enum ESqlReturnCode * __ptr64),void (__cdecl*)(void * __ptr64),void * __ptr64,enum ESqlReturnCode * __ptr64> const >+0x23 05 000000df`1557afc0 00007ff8`a2b6bfc4 sqllang!CallProtectorImpl::CallExternalFull<AppDomainUserCallTraits,void,FunctionCallBinder_3<void,void (__cdecl*)(CXVariant * __ptr64,CXVariant * __ptr64,CClrLobContext * __ptr64),CXVariant * __ptr64,CXVariant * __ptr64,CClrLobContext * __ptr64> const >+0x2dd 06 000000df`1557b130 00007ff8`a2bda602 sqllang!CAppDomain::InvokeClrFn+0xd4 07 000000df`1557b1d0 00007ff8`aef51ee7 sqllang!UDFInvokeExternalImpl+0xb72 08 000000df`1557b7e0 00007ff8`9de52e24 sqlTsEs!CEsExec::GeneralEval4+0xe7 09 000000df`1557b8b0 00007ff8`9de52d64 sqlmin!CQScanProjectNew::EvalExprs+0x18f 0a 000000df`1557b920 00007ff8`9ddd8759 sqlmin!CQScanProjectNew::GetRow+0x98 0b 000000df`1557b970 00007ff8`9ddc73de sqlmin!CQScanLightProfileNew::GetRow+0x19 0c 000000df`1557b9a0 00007ff8`a25e51d7 sqlmin!CQueryScan::GetRow+0x80 0d 000000df`1557b9d0 00007ff8`a32a78b2 sqllang!CXStmtQuery::ErsqExecuteQuery+0x3d8 0e 000000df`1557bb40 00007ff8`a2bc2451 sqllang!CXStmtSelect::XretDoExecute+0x342 0f 000000df`1557bc10 00007ff8`a2b733d3 sqllang!UM_LoopbackForStatementExecution+0x191 10 000000df`1557bd00 00007ff8`de48e940 sqllang!AppDomainCallback<FunctionCallBinder_5<void,void (__cdecl*)(CXStmtQuery * __ptr64,CCompExecCtxtStmt const * __ptr64,CMsqlExecContext * __ptr64,unsigned long * __ptr64,enum ESqlReturnCode * __ptr64),CXStmtQuery * __ptr64,CCompExecCtxtStmt const * __ptr64,CMsqlExecContext * __ptr64,unsigned long * __ptr64,enum ESqlReturnCode * __ptr64> >+0x23 11 000000df`1557bd40 00007ff8`de48e193 clr!ExecuteInAppDomainHelper+0x40 12 000000df`1557bd80 00007ff8`a2b79f39 clr!CorHost2::ExecuteInAppDomain+0x3a0 13 000000df`1557c0a0 00007ff8`a2b73a86 sqllang!CallProtectorImpl::CallWithSEH<AppDomainCallTraits,long,MethodCallBinder_3<long,ICLRRuntimeHost,long (__cdecl ICLRRuntimeHost::*)(unsigned long,long (__cdecl*)(void * __ptr64),void * __ptr64) __ptr64,unsigned long,long (__cdecl*)(void * __ptr64),void * __ptr64> >+0x29 14 000000df`1557c0d0 00007ff8`a2b6c2d0 sqllang!CallProtectorImpl::CallExternalFull<AppDomainCallTraits,long,MethodCallBinder_3<long,ICLRRuntimeHost,long (__cdecl ICLRRuntimeHost::*)(unsigned long,long (__cdecl*)(void * __ptr64),void * __ptr64) __ptr64,unsigned long,long (__cdecl*)(void * __ptr64),void * __ptr64> >+0x186 15 000000df`1557c170 00007ff8`a32a72f4 sqllang!CAppDomain::LoopbackForStatementExecution+0x180 16 000000df`1557c230 00007ff8`a32a79ad sqllang!CXStmtQuery::XretCLRExecute+0x104 17 000000df`1557c2a0 00007ff8`a25e4a65 sqllang!CXStmtSelect::XretExecute+0x4a 18 000000df`1557c370 00007ff8`a25e44a8 sqllang!CMsqlExecContext::ExecuteStmts<1,1>+0x8f2 19 000000df`1557cf10 00007ff8`a25e3a2c sqllang!CMsqlExecContext::FExecute+0x936 1a 000000df`1557def0 00007ff8`a25ee67b sqllang!CSQLSource::Execute+0xc5c 1b 000000df`1557e3d0 00007ff8`a25ed815 sqllang!process_request+0xca6 1c 000000df`1557ead0 00007ff8`a25ed5ef sqllang!process_commands_internal+0x4b7 1d 000000df`1557ec00 00007ff8`b1e46523 sqllang!process_messages+0x1d6 1e 000000df`1557ede0 00007ff8`b1e46e6d sqldk!SOS_Task::Param::Execute+0x232 1f 000000df`1557f3e0 00007ff8`b1e46c75 sqldk!SOS_Scheduler::RunTask+0xa5 20 000000df`1557f450 00007ff8`b1e6b160 sqldk!SOS_Scheduler::ProcessTasks+0x39d 21 000000df`1557f570 00007ff8`b1e6aa5b sqldk!SchedulerManager::WorkerEntryPoint+0x2a1 22 000000df`1557f640 00007ff8`b1e6afa4 sqldk!SystemThreadDispatcher::ProcessWorker+0x3ed 23 000000df`1557f940 00007ff8`f6d86fd4 sqldk!SchedulerManager::ThreadEntryPoint+0x3b5 24 000000df`1557fa30 00007ff8`f865cec1 KERNEL32!BaseThreadInitThunk+0x14 25 000000df`1557fa60 00000000`00000000 ntdll!RtlUserThreadStart+0x21
request 請求,然後達到了託管方法 UserLogin
,頂部的三行執行緒棧可以用 !clrstack
0:090> !clrstack OS Thread Id: 0x6df4 (90) Child SP IP Call Site 000000df1557ae48 00007ff87ec560d0 AQMN.Bussiness.UserFunctions.UserLogin(System.String, System.String) 000000df1557ae50 00007ff87ee500b6 DynamicClass.SQLCLR_Eval(IntPtr, IntPtr, IntPtr) 000000df1557aeb0 00007ff87ec55ef1 DomainBoundILStubClass.IL_STUB_ReversePInvoke(Int64, Int64, Int64) 000000df1557bf18 00007ff8de04222e [ContextTransitionFrame: 000000df1557bf18]
看來SQLSERVER 內嵌了 CLR,讓 sqlservr 程序成了一種託管和非託管的混合環境,託管的 C#,VB,F# 可以助 SQLSERVER 更加強大。
到此這篇關於SQLSERVER呼叫C#的程式碼實現的文章就介紹到這了,更多相關SQLSERVER呼叫C#內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!
相關文章
<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
综合看Anker超能充系列的性价比很高,并且与不仅和iPhone12/苹果<em>Mac</em>Book很配,而且适合多设备充电需求的日常使用或差旅场景,不管是安卓还是Switch同样也能用得上它,希望这次分享能给准备购入充电器的小伙伴们有所
2021-06-01 09:31:42
除了L4WUDU与吴亦凡已经多次共事,成为了明面上的厂牌成员,吴亦凡还曾带领20XXCLUB全队参加2020年的一场音乐节,这也是20XXCLUB首次全员合照,王嗣尧Turbo、陈彦希Regi、<em>Mac</em> Ova Seas、林渝植等人全部出场。然而让
2021-06-01 09:31:34
目前应用IPFS的机构:1 谷歌<em>浏览器</em>支持IPFS分布式协议 2 万维网 (历史档案博物馆)数据库 3 火狐<em>浏览器</em>支持 IPFS分布式协议 4 EOS 等数字货币数据存储 5 美国国会图书馆,历史资料永久保存在 IPFS 6 加
2021-06-01 09:31:24
开拓者的车机是兼容苹果和<em>安卓</em>,虽然我不怎么用,但确实兼顾了我家人的很多需求:副驾的门板还配有解锁开关,有的时候老婆开车,下车的时候偶尔会忘记解锁,我在副驾驶可以自己开门:第二排设计很好,不仅配置了一个很大的
2021-06-01 09:30:48
不仅是<em>安卓</em>手机,苹果手机的降价力度也是前所未有了,iPhone12也“跳水价”了,发布价是6799元,如今已经跌至5308元,降价幅度超过1400元,最新定价确认了。iPhone12是苹果首款5G手机,同时也是全球首款5nm芯片的智能机,它
2021-06-01 09:30:45