大家在学习 entityframework 的时候,都知道那 linq 写的叫一个爽,再也不用区分不同 RDMS 的 sql 版本差异了,但是呢,高效率带来了差灵活性,我们无法控制 sql 的生成策略,所以必须不要让自己缺乏好的工具去监控 sql,本篇给大家介绍的三种监控手段 Log 和 SqlServer profile,ef profile。
一、Log 监控
这个属于 entity framework 自带的一个 Action 方法,它给大家带来了不错的用户体验,我们可以将其输出放到控制台,又或者写入到记事本中。
这里我就通过 EDM 来生成 codefirst,可以看到如下的 Database 的 Log 定义,然后大家就可以给他灌入一个带 string 参数的 Action 方法,比如 Console.WriteLine。
public Action<string> Log { get; set; }
然后写一个 demo 程序
static void Main(string[] args)
{
using (SchoolDB2Entities dbContext = new SchoolDB2Entities())
{
dbContext.Database.Log = Console.WriteLine;
dbContext.Students.Add(new Student()
{
StudentName = "jack123"
});
dbContext.SaveChanges();
}
}
由于 codefirst 初始化生成之时内容太多,无法一一显示出来。
为了方便可灌入自定义方法 AppendLog,比如将其灌入到 File 中。
二、SqlServer Profile
可以看到 SqlServer Profile 是放在 sqlserver 层面上的监控,可以监控到各种 sql 如何流入到 sqlserver 中,但是如果你默认开启的话,会看到很多与 ef 无关的操作,比如下面这样:
那更好的方式是怎么过滤呢?其实也很简单,我们只需要在 ef 的 connectionstring 中塞入一个 App 标记,然后在 sqlserver profile 上面进行筛选就可以了。
1) 第一步:在 connectionstring 中加入 applicationName
<connectionStrings>
<add name="SchoolDB2Entities" connectionString="data source=.;initial catalog=SchoolDB2;integrated security=True;MultipleActiveResultSets=True;App=EntityFramework" providerName="System.Data.SqlClient" />
</connectionStrings>
2) 第二步:在 SqlProfile 中筛选 ApplicationName=EntityFramework 的所有记录
ok,这样我们就配置好了,接下来我们将 codefirst 跑起来,可以清清楚楚的看到,现在的 profile 中仅仅只有 EntityFramework 标记生成的 sql 语句了。
三、Entity Framework Profile
首先这个是一款商业软件,免费试用期是 30 天,不过网上还是能够搜索到各种破解版,废话不多说,我们通过 nuget 下载一下:
下载完之后,再看一下 packages 文件夹中的子文件夹 EntityFrameworkProfiler.3.0.3103.0,找到一个叫做 efprof.exe 的程序,这个就是要开启的监控。
打开软件后,大家可以自己注册一下,生成一个 licence 的 xml,然后大家就可以看到这样的一个??的界面了。
更加??的是,会在我们的 Main 函数中注入一段开启逻辑,并且在 App_Start 中生成了一个 cs,一个 vb 文件,如下:
class Program
{
static void Main(string[] args)
{App_Start.EntityFrameworkProfilerBootstrapper.PreStart();
using (SchoolDB2Entities dbContext = new SchoolDB2Entities())
{
dbContext.Database.Log = AppendLog;
dbContext.Students.Add(new Student()
{
StudentName = "jack123"
});
dbContext.SaveChanges();
}
}
static void AppendLog(string str) => System.IO.File.AppendAllText(Environment.CurrentDirectory + "\1.txt", str, Encoding.Default);
}
既然代码都生成好了。。。那就留给我要做的事情就是 Ctrl + F5 了。
可以看到,efprofile 可以抓拍到各种 create table 语句,包括各种面板信息:Application Statistics,Analysis,Duration 等等。
最重要的我们还能看到 “Query plan”,比如下面我构造一个相对比较复杂的 sql 语句:
class Program
{
static void Main(string[] args)
{
App_Start.EntityFrameworkProfilerBootstrapper.PreStart();
using(SchoolDB2Entities dbContext = new SchoolDB2Entities())
{
var query = (from n in dbContext.Students from m in dbContext.StudentAddresses where n.StudentID == m.StudentID group n.StudentID by n.StudentName into g select new
{
g.Key, count = g.Count()
}).ToList();
dbContext.SaveChanges();
}
}
}
然后执行一下,看看 efprofile 监视到 db 中是如何生成 query plan 的快照的。
转自:一线码农