从实体框架获取 SQL 查询 IQueryable帮助我解决了这个问题以满足稍微不同的要求,我需要访问 TraceString和底层参数,以便我可以将一些IQueryable<>
表达式注入到.SqlQuery<>()
当您IQueryable
在调试会话期间检查 an 时,您可以看到它在内部使用 ObjectQuery,但它不从 ObjectQuery继承,因此您不能直接转换它。然而,我们可以使用反射来访问内部查询对象。
这是 Steve Fenton 的代码封装到扩展方法中:
/// <summary>
/// Extract the Internal Object Query from an IQueryable, you might do this to access the internal parameters for logging or injection purposes
/// </summary>
/// <remarks>Sourced from https://www.stevefenton.co.uk/2015/07/getting-the-sql-query-from-an-entity-framework-iqueryable/ </remarks>
/// <typeparam name="T">Entity Type that is the target of the query</typeparam>
/// <param name="query">The query to inspect</param>
/// <returns>Object Query that represents the same IQueryable Expression</returns>
public static System.Data.Entity.Core.Objects.ObjectQuery<T> ToObjectQuery<T>(this IQueryable<T> query)
{
// force the query to be cached, otherwise parameters collection will be empty!
string queryText = query.ToString();
queryText = queryText.ToLower(); // stop compiler from optimising this code away because we don't do anything with queryText parameter!
var internalQueryField = query.GetType().GetProperties(System.Reflection.BindingFlags.NonPublic | System.Reflection.BindingFlags.Instance).Where(f => f.Name.Equals("InternalQuery")).FirstOrDefault();
var internalQuery = internalQueryField.GetValue(query);
var objectQueryField = internalQuery.GetType().GetFields(System.Reflection.BindingFlags.NonPublic | System.Reflection.BindingFlags.Instance).Where(f => f.Name.Equals("_objectQuery")).FirstOrDefault();
return objectQueryField.GetValue(internalQuery) as System.Data.Entity.Core.Objects.ObjectQuery<T>;
}
现在您的代码可能如下所示:
IEnumerable<Coupon> coupons =
(from x in ctx.Coupons
where x.LanguageCode.Equals("EN", StringComparison.InvariantCultureIgnoreCase) && x.CategoryId == MainCategoryId && x.Date >= fromDate && x.Date <= toDate
select x).ToObjectQuery();