麒麟v10 上部署 TiDB v5.1.2 生产环境优化实践
546
2023-04-17
***居然也能调 C# 代码 ?
一:背景
1. 讲故事
前些天看到一个奇怪的 Function 函数,调用的是 C# 链接库中的一个 UserLogin 方法,参考代码如下:
CREATE FUNCTION dbo.clr_UserLogin( @name AS NVARCHAR(100), @password AS NVARCHAR(100))RETURNS INT ASEXTERNAL NAME asmXXX.[xxx.CLRFunctions].UserLogin;GO
这就让我产生了很大的兴趣,众所周知 SQLSERVER 是 C++ 写的,那这里的 C++ 怎么和 C# 打通呢?而且 C# 是一门托管语言,需要 JIT 将其 native 化,这个 JIT 又在哪里呢?带着这些疑问一起研究下吧。
二:互通原理研究
1. 一个简单的例子
首先写一段简单的 C# 代码,然后把它编译成 dll。
namespace AQMN.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;GOALTER DATABASE MyTestDB SET TRUSTWORTHY ON;GO
为了能够调到 C# 的 UserLogin 方法,需要 SQLSERVER 先导入这个程序集,然后再以 Function 映射其中方法即可,参考代码如下:
CREATE ASSEMBLY clr_AQMN_BussinessFROM 'D:\net6\SQLCrawl\AQMN.Bussiness\bin\Debug\AQMN.Bussiness.dll'WITH PERMISSION_SET = UNSAFE;GOCREATE FUNCTION dbo.clr_UserLogin( @username AS NVARCHAR(100), @password AS NVARCHAR(100))RETURNS NVARCHAR(100)ASEXTERNAL NAME clr_AQMN_Bussiness.[AQMN.Bussiness.UserFunctions].UserLogin;GO
创建完了之后,可以观察 assembly 开头的几个系统视图。
SELECT * FROM sys.assembliesSELECT * 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 函数起到了作用,非常有意思。
2. WinDbg 观察
从案例的运行结果看,推测在 SQLSERVER 中应该承载了一个 CLR 运行环境,那是不是这样呢?可以用 WinDbg 附加到 sqlservr.exe 进程,用 lm 观察下模块加载情况。
0:092> lmstart end module name...00007ff8`d3960000 00007ff8`d3aaf000 clrjit (deferred) 00007ff8`de040000 00007ff8`deb02000 clr (deferred) ...0:092> !eeversion4.8.4300.0 freeServer mode with 12 gc heapsSOS Version: 4.8.4300.0 retail build
版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。