- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我有一个复杂的 IQueryable,我希望 EF 用单个数据库查询填充它,以便我可以将它用于延迟执行。请考虑以下示例。
对于这些模型:
public enum AlphaState { Unknown = '\0', Good = 'G', Bad = 'B' }
[Table("MY_ALPHA")]
public class Alpha
{
[Key]
[Column("alpha_index")]
public long Index { get; set; }
[Column("alpha_id")] // user-editable field that users call ID
public string AlphaId { get; set; }
[Column("deleted")]
public char? Deleted { get; set; }
[Column("state")]
public AlphaState State { get; set; }
[InverseProperty("Alpha")]
public ICollection<Bravo> Bravos { get; set; }
}
[Table("MY_BRAVO")]
public class Bravo
{
[Key]
[Column("bravo_index")]
public long BravoIndex { get; set; }
[ForeignKey("Alpha")]
[Column("alpha_index")] // actually a 1:0..1 relationship
public long? AlphaIndex { get; set; }
public virtual Alpha Alpha { get; set; }
[InverseProperty("Bravo")]
public ICollection<Charlie> Charlies { get; set; }
}
[Table("MY_CHARLIE_VIEW")]
public class Charlie
{
[Key]
[Column("charlie_index")]
public int CharlieIndex { get; set; }
[Column("deleted")]
public char? Deleted { get; set; }
[Column("created_at")]
public DateTime CreatedAt { get; set; }
[ForeignKey("Bravo")]
[Column("bravo_index")]
public long BravoIndex { get; set; }
public virtual Bravo Bravo { get; set; }
[ForeignKey("Delta")]
[Column("delta_index")]
public long DeltaIndex { get; set; }
public virtual Delta Delta { get; set; }
[InverseProperty("Charlie")]
public virtual ICollection<Delta> AllDeltas { get; set; }
}
[Table("MY_DELTA")]
public class Delta
{
[Key]
[Column("delta_index")]
public long DeltaIndex { get; set; }
[ForeignKey("Charlie")]
[Column("charlie_index")]
public long CharlieIndex { get; set; }
public virtual Charlie Charlie { get; set; }
[InverseProperty("Delta")] // actually a 1:0..1 relationship
public ICollection<Echo> Echoes { get; set; }
}
public enum EchoType { Unknown = 0, One = 1, Two = 2, Three = 3 }
[Table("MY_ECHOES")]
public class Echo
{
[Key]
[Column("echo_index")]
public int EchoIndex { get; set; }
[Column("echo_type")]
public EchoType Type { get; set; }
[ForeignKey("Delta")]
[Column("delta_index")]
public long DeltaIndex { get; set; }
public virtual Delta Delta { get; set; }
}
...考虑这个查询:
IQueryable<Alpha> result = context.Alphas.Where(a => a.State == AlphaState.Good)
.Where(a => !a.Deleted.HasValue)
.Where(a => a.Bravos.SelectMany(b => b.Charlies)
.Where(c => !c.Deleted.HasValue)
.Where(c => c.Delta.Echoes.Any())
.OrderByDescending(c => c.CreatedAt).Take(1)
.Any(c => c.Delta.Echoes.Any(e => e.Type == EchoType.Two)))
var query = result as System.Data.Objects.ObjectQuery;
string queryString = query.ToTraceString();
注意:Charlie 实际上是表上的一个 View ; Delta 具有到 Charlie 的表的 FK,但 View 为链接到该 Charlie 的最新 Delta 提供了一个假 FK,因此该模型使用它,因为计划是将 EF 仅用于查询,从不用于更新。
我希望此查询由单个数据库查询填充,但正如所写的那样,实际情况并非如此。我如何修改此查询以获得相同的结果,但让 EF 简单地将条件构建到 results
IQueryable 而不是为其预取数据?
我怎么知道它使用了两个查询
我确定它会拆分为多个查询,因为出于超出此问题范围的原因,我故意为上下文提供了错误的连接字符串。 result
是一个 IQueryable,因此它应该使用延迟执行,并且在使用它之前不会实际尝试检索任何数据,但是我一声明它就收到连接失败异常。
背景
我们有一个现有的数据库结构、数据库访问层,以及使用上述结构和 DAL 的数十万行代码。我们想要添加一个 UI 以允许用户构建他们自己的复杂查询,而 EF 似乎是为此构建基础模型的好方法。不过,我们以前从未使用过 EF,因此当权者已声明它永远无法连接到数据库;我们应该使用 EF 生成 IQueryable,从中提取查询字符串,并使用我们现有的 DAL 来运行查询。
最佳答案
How I Know It's Using Two Queries
您观察到的不是 EF 开始运行您的查询。在您将查询分配给您的 result
之后变量,您仍然有一个查询定义,而不是结果集。如果将探查器附加到数据库,您会看到没有 SELECT
语句已针对您的查询执行。
那么,为什么是作为与数据库的连接?原因是您第一次为给定的派生 DbContext
构建查询类型,EF 为该类型构建并缓存其内存中模型。它通过对您定义的类型、特性和属性应用各种约定来实现这一点。理论上这个过程不需要连接到数据库,但 SQL Server 的提供者无论如何都会这样做。它这样做是为了确定您正在使用的 SQL Server 版本,以便确定它是否可以在其构建的模型中使用更新的 SQL Server 功能。
有趣的是,这个模型是为类型 缓存的,而不是上下文的实例。你可以通过处理你的 context
来看到这一点然后创建一个新的并重复构建查询的代码行。在第二种情况下,您不会看到与数据库的连接,因为 EF 会将其缓存模型用于您的上下文类型。
the Powers That Be have declared that it cannot ever connect to the database; we should use EF to generate an IQueryable, extract the query string from it, and use our existing DAL to run the query.
由于您需要完全避免让 EF 连接到数据库,您可以查看我的帖子 here其中包含有关如何在代码中预先提供此信息的详细信息。
此外,请注意,还有另一个原因可能导致 EF 在第一次遇到您的 DbContext
时连接到您的服务器。类型:初始化。除非您禁用了初始化(使用类似 Database.SetInitializer<MyContext>(null)
的东西),否则它将检查数据库是否存在,如果不存在则尝试创建它。
请注意,您可以调用 ToString()
直接在 EF 代码优先查询上获取 T-SQL 查询。你不需要通过中间 ObjectQuery.ToTraceString()
方法,它实际上是遗留 EF API 的一部分。但是,这两种方法都用于调试和记录目的。使用 EF 构建查询而不执行查询是很不寻常的。您可能会遇到这种方法的问题——最明显的是当 EF 确定它应该生成参数化查询时。此外,无法保证不同版本的 EF 会为相同的输入生成类似的 T-SQL,因此当您更新到较新的 EF 版本时,您的代码最终可能会变得相当脆弱。确保你有足够的测试!
如果您担心让用户直接连接到数据库——这是一个完全合理的安全问题——您可以考虑另一种方法。我没有太多经验,但似乎OData可能很合适。这允许您在客户端上构建查询,通过远程连接序列化它们,然后在您的服务器上重新创建它们。然后,在服务器上,您可以针对您的数据库执行它们。您的客户需要对数据库一无所知。
如果您确实决定(或被指示)坚持您在问题中详述的方法,请花时间学习 how to profile a SQL Server connection .这将是您了解 EF 如何翻译您的查询的绝对必要工具。
关于c# - 如何强制 EF 使用连接而不是拆分复杂的查询?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/30652083/
EF POCO 和 EF Code First 有什么区别? 如果我一开始只使用 POCO,我可以将 EF 与它们一起使用吗? 最佳答案 如果您首先使用 EF 代码,您将拥有 POCO 对象,并且数据
EF POCO 和 EF Code First 有什么区别? 如果我一开始只使用 POCO,我可以将 EF 与它们一起使用吗? 最佳答案 如果您首先使用 EF 代码,您将拥有 POCO 对象,并且数据
我有一个基于 .NET 4.8 和 EF 6.4.4 的项目。我们正在逐步迁移到 .Net Core,但在此过程中我可以创建一个 .NET Core 数据上下文类 EF Core 并将两者指向相同的实
我有以下 Entity Framework 5 代码第一类 public class Airplane { public int Id { get; set; } public int
我正在尝试使用 Entity Framework Core 对现有数据库进行逆向工程.我试着按照指示from Microsoft但我遇到了错误: Unable to find provider ass
当数据库不是由 EF 代码首先创建时,有没有办法检查 DbContext 是否与数据库匹配? 我正在寻找与 Database.CompatibleWithModel 类似的功能但没有元数据。 最佳答案
目前,我正在重构我的上下文方法的测试,以不再需要真正的数据库。我使用 Ef Core。 所以我通读了 Microsoft 文档如何测试上下文方法。我首先找到了 EF6 测试的文档,然后阅读了 EfCo
我正在使用 EF 6 Database.SqlQuery 语句来执行存储过程,并实现错误和事务处理,打开和关闭事务(处理多个记录,如果一个记录有错误,仅回滚此,并提交其他任何内容无错误完成)。 Lis
我首先使用 EF 数据库,因为我喜欢在 SQL Management Studio 中设计我的数据库,坦率地说,让 Visual Studio 直接从数据库创建所有实体非常容易,而无需执行任何代码。
我的项目中有几个迁移文件,由于我对上次迁移进行了手动修改,所以我不想使用“程序包管理器控制台”重新生成它。我只需要添加 1 列。所以在之前的迁移中手动添加了这个(我可以这样做,因为还没有人升级过)。
原文:https://bit.ly/2umidlb 作者:jon p smith 翻译:王亮 声明:我翻译技术文章不是逐句翻译的,而是根据我自己的理解来表述的。其中可能会去除一些本人实在不知道如何组
我们想开始一个新项目,我们决定从一开始就在一些表中使用 Columnstore 索引和聚簇索引,我们如何使用 Code First EF Core 3.1 做到这一点? 最佳答案 您应该将主键更改为非
我在 Entity Framework 6.0 上。这是一个开发问题,而不是生产问题。 我想我有一个相互矛盾的策略。 目前,我设置了 DropCreateDatabaseIfModelChanges
我在 VS 2012 RTM 上,使用 EF 5。我在做代码优先,但由于我只是在开发中,所以试图忽略代码迁移。为了避免它们,我有这一套 Database.SetInitializer(new Drop
我有复杂的审计字段类型 我的复杂类型: [ComplexType] public class AuditData { [Column("CreatorUserId")] public
我已经阅读了很多关于如何在关系中指定外键名称的帖子,没有遇到任何问题。不过,我想知道的是,有没有办法更改主键和关系的命名方式? 例如,您有一个以 UserId 作为主键的 User 表。 EF 代码第
我刚刚安装了新的Entity Framework 4.1 NuGet包,因此根据NuGet指令和this article of Scott Hanselman替换了EFCodeFirst包。 现在,想
我的应用程序基于 .NET 4.0 和 EF 4。我现在正在考虑升级到最新版本。 是否有任何可能对我的申请产生不利影响的重大变化或行为差异? 升级路径有多简单?升级到 EF 5 是否需要任何代码更改或
假设您必须使用 EF Code First 和 POCO 类开发支持多种语言的网站,您将如何对 POCO 类进行建模以支持这种情况? 通过多语言支持,我的意思不仅是拥有例如在一些资源文件中为您的 UI
我首先使用 EF 4.1 代码。鉴于以下类片段: public class Doctor { public virtual ICollection Hospitals { get; set;
我是一名优秀的程序员,十分优秀!