Debug 对 Linq to Sql 有什么影响吗?

本文探讨了在使用Table<T>.Attach()方法时遇到的异常问题,解释了如何通过修改实体类来避免这一问题,并提供了一个具体的解决方案。

关于 Table<TEntity>.Attach(),据说它的作用就是可以在不同的DataContext当中附加数据,让被附加的数据(实体)即使在外部(DataContext上下文之外)修改也可以让别的DataContext中的对应数据得以更新。

假设我已经有一个Category的实体类与对应的数据库表。
相关类代码如下:

 1    public class DataAccess
 2    {
 3        private static DLContext dtx;
 4
 5        static DataAccess()
 6        {
 7            dtx = new DLContext();
 8            dtx.Log = Console.Out;
 9        }

10
11        public static DLContext Instance
12        {
13            get
14            {
15                return dtx;
16            }

17        }

18     
19        public static void Attach(Category cy)
20        {
21            dtx.GetTable<Category>().Attach(cy, true);
22        }

23
24    }


    public class DLContext:DataContext
    
{
        
private static MappingSource mappingSource = new AttributeMappingSource();
        
private static string connectionString=@"Server=.\SQLExpress;Database=Helloworld;User ID=xxx;Password=xxxx;";

        
public DLContext()
            : 
base(connectionString, mappingSource)
        
{
        }

    }

 1        static void Main(string[] args)
 2        {
 3            TestAttach();
 4            Console.Read();
 5        }

 6
 7        public static void TestAttach()
 8        {
 9            using (DLContext ctx = new DLContext())
10            {
11                var category = (from c in ctx.GetTable<Category>() select c).FirstOrDefault();
12                string name = category.Name;
13                DataAccess.Attach(category);
14
15                Console.WriteLine("Category ID={0},Name={1}", category.KeyID, category.Name);
16
17                category.Name = "After modifying" + DateTime.Now.ToString();
18
19                var category2 = (from c in DataAccess.Instance.GetTable<Category>() where c.KeyID == category.KeyID select c).SingleOrDefault();
20                Console.WriteLine("After changing the value of property.");
21                Console.WriteLine("Category ID={0},Name={1}", category.KeyID, category.Name);
22                Console.WriteLine("Category2 ID={0},Name={1}", category2.KeyID, category2.Name);
23                Console.WriteLine("Reference equeals ={0}", Object.ReferenceEquals(category, category2));
24            }

25        }


为什么当我不设断点的时候,会出现Exception:
"An attempt has been made to Attach or Add an entity that is not new, perhaps having been loaded from another DataContext.  This is not supported."
当我在13行  DataAccess.Attach(category);处设了一个断点,而且用鼠标放在category上查看了一下属性值,继续运行,它就不会出错,一切正常了!!!!

是什么原因啊?????

---------------------------------------------------------------------

经大侠指引,终天知道原来因为Category 表中有一个外键,对于实体类Category, 有一个字段:

private EntityRef<Parent> _Parent;

必须在调用 Attach()方法之前,先令
_Parent=default(EntityRef<Parent> );

综上所述,为了方便使用,我在Category类中加了一个Detach()方法:

public void Detach()
{
   _Parent 
= default(EntityRef<Parent>);
}

必须在调用 Table<TEntity>.Attach()方法之前,先调用category.Detach()方法。



 

转载于:https://www.cnblogs.com/vivid-stanley/archive/2008/01/30/1058128.html

<think>嗯,用户问的是如何在使用LINQ的扩展方法编写SQL语句时,查看翻译后的SQL。首先,我得确定用户对LINQ和EF Core的基础了解程度。可能他们已经在用LINQ写查询,但想调试生成的SQL,可能是为了优化性能或者验证是否正确转换。 接下来,我需要回忆不同的方法。比如,在EF Core中常用的LogTo或者日志记录,还有DbContext的Logger属性。另外,可能还要提到像LINQPad这样的工具,或者ToString()方法,不过ToString()可能在某些情况下不生成完整SQL,需要说明限制。 然后,要考虑用户的使用场景。他们可能是在开发环境中调试,所以推荐的方法应该简单易行,比如LogTo控制台输出,或者用DebugView。如果是生产环境,可能需要不同的方法,但用户可能更关注开发阶段的调试。 还要注意用户可能没有提到但相关的内容,比如异步查询的处理,或者在旧版本的EF中如何使用?可能需要提到不同版本的差异,比如EF Core之前的版本是否有不同的日志配置方式。 另外,是否需要给出具体的代码示例?比如如何在DbContextOptions中启用日志记录,或者如何通过DbContext的Database.Log属性设置。用户可能需要一步步的指导,所以示例代码会很有帮助。 还要考虑用户可能的误区,比如直接调用ToString()可能不会触发查询生成,只有在执行时才会生成SQL。需要指出必须确保查询被执行,比如通过ToList()或FirstOrDefault(),才能捕获日志。 最后,是否需要推荐第三方工具,比如EF Core的扩展库或者分析器,这些可能提供更详细的SQL信息?或者提到拦截器(Interceptors)来记录SQL? 总结下来,回答应该分步骤,介绍几种主要方法,给出具体示例,并提醒注意事项,比如执行查询的必要性,以及不同方法的适用场景。确保用户能根据具体情况选择合适的方式查看生成的SQL。</think>在 C# 中使用 LINQ 扩展方法编写查询时,查看最终生成的 SQL 语句有以下几种常用方法: --- ### **方法 1:EF Core 的日志记录(推荐)** 适用于 Entity Framework Core: ```csharp var options = new DbContextOptionsBuilder<YourDbContext>() .UseLoggerFactory(LoggerFactory.Create(builder => builder.AddConsole())) // 输出到控制台 .EnableSensitiveDataLogging() // 显示参数值 .Options; using var context = new YourDbContext(options); var query = context.Users.Where(u => u.Age > 18); var sql = query.ToQueryString(); // 直接获取生成的 SQL Console.WriteLine(sql); ``` --- ### **方法 2:通过 DebugView 查看(LINQ to SQL)** 适用于 LINQ to SQL 或 EF Core: ```csharp var query = dbContext.Users.Where(u => u.Name.Contains("John")); var debugView = query.Expression.DebugView; // 输出查询的表达式树和翻译后的 SQL(需要调试时查看) ``` --- ### **方法 3:拦截数据库命令(通用方法)** 通过监听 EF Core 的数据库命令: ```csharp // 在 DbContext 配置中添加事件监听 protected override void OnConfiguring(DbContextOptionsBuilder optionsBuilder) { optionsBuilder.UseSqlServer(connectionString) .LogTo(Console.WriteLine, // 输出到控制台 new[] { DbLoggerCategory.Database.Command.Name }, LogLevel.Information); } ``` --- ### **方法 4:第三方工具(如 LINQPad)** 使用 LINQPad 工具可直接查看生成的 SQL: 1. 在 LINQPad 中编写 LINQ 查询 2. 点击左下角的 `SQL` 按钮即可显示实际生成的 SQL --- ### **注意事项** 1. 必须触发查询执行(如调用 `.ToList()`)才能生成 SQL 2. 参数化查询中的实际值可能以 `@p0` 占位符形式显示 3. 复杂查询(如包含 `Include()` 或导航属性)可能导致 SQL 嵌套 --- **示例输出结果:** ```sql SELECT [u].[Id], [u].[Name], [u].[Age] FROM [Users] AS [u] WHERE [u].[Age] > 18 ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值