zoukankan      html  css  js  c++  java
  • Linq在路上(外一)获取SQL语句

      Linq方便了我们对数据库的操作,直接用C#的语法操作数据库。但Linq也隐藏了实际执行的SQL语句,封装是好事,可是有时候还是不得不了解Linq具体对数据库的操作。比如,只有查看实际的SQL语句才能对数据库或查询语句做优化,而且在调试时不能查看SQL是很郁闷的事情。

    有下面几种方法来挖掘出操作中所使用的Sql语句:

    1、获取Query所对应的SqlCommand对象:
      在开发过程中,我们可以通过Query获得对应的Sql Command对象。请看如下代码:
    AdventureWorksDataContext db = new AdventureWorksDataContext();
    var products = from p in db.Products
                   where p.ProductID == 3
                   select new { p.ProductID, p.Name };
    foreach (var p in products)
    {
        Console.WriteLine(p);
    }
     
    DbCommand cmd = db.GetCommand(products);
     
    Console.WriteLine("------------");
    Console.WriteLine("Command Text: \n{0}", cmd.CommandText);
     
    Console.WriteLine("------------");
    Console.WriteLine("Command Type: \n{0}", cmd.CommandType);
     
    Console.WriteLine("------------");
    Console.WriteLine("Command Parameters:");
    foreach (DbParameter p in cmd.Parameters)
    {
        Console.WriteLine("{0}: {1}", p.ParameterName, p.Value);
    }
     
    Console.ReadLine();
      
    输出结果如下:

    Command Text:
    SELECT [t0].[ProductID], [t0].[Name]
    FROM [Production].[Product] AS [t0]
    WHERE [t0].[ProductID] = @p0
    ------------
    Command Type:
    Text
    ------------
    Command Parameters:
    @p0: 3
      
    可以看到,无论是Sql语句或是参数都被打印了出来。事实上,由于我们得到了完整的SqlCommand对象,我们可以获取的信息并不止上述这些。

    2、使用LINQ to SQL Debug Visualizer:
      使用LINQ to SQL Debug Visiualizer,我们可以在调试程序时直观地获得Query所对应的Sql语句以及参数,而不必获得SqlCommand对象并打印信息。具体使用方法详见Scott Gu的这篇博文

    3、使用DataContext的Log功能:
      DataContext自带的Log属性为一个TextWriter类型的对象,如果我们设置了这个属性,则DataContext所有的 操作将会通过这个TextWriter对象输出。与比上述两种方法相比,这个方法的优势在于DataContext所执行的所有语句,无论SELECT、 INSERT、UPDATE或者是DELETE都会被输出;而上面的两种做法只能得到Query的信息,也就是SQL语句的SELECT操作。
      请看如下代码,下面的代码将AdventureWorksDataContext对象的所有操作输出至Console:
    AdventureWorksDataContext db = new AdventureWorksDataContext();
    db.Log = Console.Out;
     
    Product product = (from p in db.Products
                       where p.ProductID == 1
                       select p).First();
     
    product.Name = "Hello World";
    db.SubmitChanges();
     
    Console.ReadLine();
      
    输出结果如下:

    SELECT TOP (1) [t0].[ProductID], [t0].[Name], [t0].[ProductNumber], [t0].[MakeFl
    ...
    edDate], [t0].[rowguid], [t0].[ModifiedDate]
    FROM [Production].[Product] AS [t0]
    WHERE [t0].[ProductID] = @p0
    -- @p0: Input Int (Size = 0; Prec = 0; Scale = 0) [1]
    -- Context: SqlProvider(Sql2005) Model: AttributedMetaModel Build: 3.5.21004.1

    UPDATE [Production].[Product]
    SET [Name] = @p11
    WHERE ([ProductID] = @p0) AND ([Name] = @p1) AND ([ProductNumber] = @p2) AND (NO
    ...
    NULL) AND ([rowguid] = @p9) AND ([ModifiedDate] = @p10)
    -- @p0: Input Int (Size = 0; Prec = 0; Scale = 0) [1]
    ...
    -- @p11: Input NVarChar (Size = 11; Prec = 0; Scale = 0) [Hello World]
    -- Context: SqlProvider(Sql2005) Model: AttributedMetaModel Build: 3.5.21004.1
      
    在这里我省略了大部分的输出,不过从上面的片断中我们已经可以看到SELECT和UPDATE操作所使用的Sql语句以及参数都被打印了出来。这就是我们可以利用的调试信息。

    4、使用Sql Server Profiler:
      这个是我推荐的方法,比1,3方便,而且不可能漏掉任何信息。Sql Profiler的用法就不说了,大家应该都用过,但和监视一般的SQL还是有点地方需要注意。
    aaaDataContext db = new aaaDataContext();
    aaa a = new aaa();
    var q =
                from p in context.aaa
                where p.a.StartsWith("a") == true
                select p.id;

    int? a1 = q.First();

    上面这段代码Linq不是直接将它转换成"select id from aaa where....."而是转换成

      exec sp_executesql N'SELECT TOP (1) [t0].[id]
    FROM [dbo].[aaa] AS [t0]
    WHERE [t0].[a] LIKE @p0',N'@p0 nvarchar(2)',@p0=N'a%'

      使用了存储过程sp_executesql以及动态代码的方式,因此,想要获取上述这段SQL需要打开profiler事件选择中stored procedures下的RPC:Completed和SP:Completed事件。而且由于Linq使用延迟加载机制来减少性能消耗,因此只有在int? a1 = q.First();执行时,sql语句才真正被执行。


    鼓励作者写出更好的文章



    QQ讨论群612280956
      


    每个人都会经过这个阶段,见到一座山,就想知道山后面是什么。我很想告诉他,可能翻过山后面,你会发现没什么特别。回望之下,可能会觉得这一边更好。 每个人都会坚持自己的信念,在别人看来,是浪费时间,她却觉得很重要。
  • 相关阅读:
    QQ浏览器X5内核问题汇总
    jQuery全屏滚动插件fullPage.js
    CSS3 Animation
    CSS3 Transition
    CSS3 Transform
    HTML5学习笔记(2):input type file的特性
    HTML5学习笔记(1):HTML5介绍与语法
    你必须知道的28个HTML5特征、窍门和技术
    Java内存释放——《Thinking in Java》随笔004
    构造器调用构造器——《Thinking in Java》随笔003
  • 原文地址:https://www.cnblogs.com/amingo/p/1562524.html
Copyright © 2011-2022 走看看