EntityFramework4.5使用Expression类创建动态查询及动态查询导航属性
创建动态查询
想在项目中实现一个灵活的动态查询类,参考http://www.cnblogs.com/lyj/archive/2008/03/25/1122157.html和http://www.cnblogs.com/killuakun/archive/2008/08/03/1259389.html后写了一段Demo,发现代码在VS2012 EF4.5中会抛如下异常:

相同的代码在VS2008 EF3.5中是可以正常运行的:

纠结万分后找到解决方法,代码如下:
- OscarEntities db = new OscarEntities();
- IQueryable<City> cities = db.Citys;
- ParameterExpression param = Expression.Parameter(typeof(City), "c");
- Expression left = Expression.Property(param, typeof(City).GetProperty("Name"));
- Expression right = Expression.Constant("北京市");
- Expression filter = Expression.Equal(left, right);
- //Expression pred = Expression.Lambda(filter, param);
- //Expression expr = Expression.Call(typeof(Queryable), "Where", new Type[] { typeof(City) },
- // Expression.Constant(cities), pred);
- //var result = db.Citys.AsQueryable().Provider.CreateQuery<City>(expr);
- var result = db.Citys.Where(Expression.Lambda<Func<City, bool>>(filter, param));
- list.DataSource = result.ToList();
- list.DisplayMember = "Name";
- OscarEntities db = new OscarEntities();
- IQueryable<City> cities = db.Citys;
- ParameterExpression param = Expression.Parameter(typeof(City), "c");
- Expression left = Expression.Property(param, typeof(City).GetProperty("Name"));
- Expression right = Expression.Constant("北京市");
- Expression filter = Expression.Equal(left, right);
- //Expression pred = Expression.Lambda(filter, param);
- //Expression expr = Expression.Call(typeof(Queryable), "Where", new Type[] { typeof(City) },
- // Expression.Constant(cities), pred);
- //var result = db.Citys.AsQueryable().Provider.CreateQuery<City>(expr);
- var result = db.Citys.Where(Expression.Lambda<Func<City, bool>>(filter, param));
- list.DataSource = result.ToList();
- list.DisplayMember = "Name";
OscarEntities db = new OscarEntities();
IQueryable<City> cities = db.Citys;
ParameterExpression param = Expression.Parameter(typeof(City), "c");
Expression left = Expression.Property(param, typeof(City).GetProperty("Name"));
Expression right = Expression.Constant("北京市");
Expression filter = Expression.Equal(left, right);
//Expression pred = Expression.Lambda(filter, param);
//Expression expr = Expression.Call(typeof(Queryable), "Where", new Type[] { typeof(City) },
// Expression.Constant(cities), pred);
//var result = db.Citys.AsQueryable().Provider.CreateQuery<City>(expr);
var result = db.Citys.Where(Expression.Lambda<Func<City, bool>>(filter, param));
list.DataSource = result.ToList();
list.DisplayMember = "Name";

动态查询导航属性
实体关系如图:

如何拼接出 db.Citys.Where(x => x.Province.Name == "湖南省") 呢?,代码如下:
- OscarEntities db = new OscarEntities();
- IQueryable<City> cities = db.Citys;
- ParameterExpression param = Expression.Parameter(typeof(City), "c");
- Expression left = Expression.Property(param, typeof(City).GetProperty("Province")); //先得到导航属性Province
- Expression leftproperty = Expression.Property(left, "Name"); //再得到Province.Name
- Expression right = Expression.Constant("湖南省");
- Expression filter = Expression.Equal(leftproperty, right);
- var result = db.Citys.Where(Expression.Lambda<Func<City, bool>>(filter, param));
- list.DataSource = result.ToList();
- list.DisplayMember = "Name";
- 网上的中文资料很少,我对着Expression.Property方法琢磨了老半天才整明白,写在这方便大家吧。
- 执行结果:
- OscarEntities db = new OscarEntities();
- IQueryable<City> cities = db.Citys;
- ParameterExpression param = Expression.Parameter(typeof(City), "c");
- Expression left = Expression.Property(param, typeof(City).GetProperty("Province")); //先得到导航属性Province
- Expression leftproperty = Expression.Property(left, "Name"); //再得到Province.Name
- Expression right = Expression.Constant("湖南省");
- Expression filter = Expression.Equal(leftproperty, right);
- var result = db.Citys.Where(Expression.Lambda<Func<City, bool>>(filter, param));
- list.DataSource = result.ToList();
- list.DisplayMember = "Name";
- 网上的中文资料很少,我对着Expression.Property方法琢磨了老半天才整明白,写在这方便大家吧。
- 执行结果:
OscarEntities db = new OscarEntities();
IQueryable<City> cities = db.Citys;
ParameterExpression param = Expression.Parameter(typeof(City), "c");
Expression left = Expression.Property(param, typeof(City).GetProperty("Province")); //先得到导航属性Province
Expression leftproperty = Expression.Property(left, "Name"); //再得到Province.Name
Expression right = Expression.Constant("湖南省");
Expression filter = Expression.Equal(leftproperty, right);
var result = db.Citys.Where(Expression.Lambda<Func<City, bool>>(filter, param));
list.DataSource = result.ToList();
list.DisplayMember = "Name"; 网上的中文资料很少,我对着Expression.Property方法琢磨了老半天才整明白,写在这方便大家吧。 执行结果:

再贴上自己项目中用的方法
- public Expression GetProperty(Expression source, ParameterExpression para, string Name)
- {
- string[] propertys = Name.Split('.');
- if (source == null)
- {
- source = Expression.Property(para, typeof(City).GetProperty(propertys.First()));
- }
- else source = Expression.Property(source, propertys.First());
- foreach (var item in propertys.Skip(1))
- {
- source = GetProperty(source , para, item);
- }
- return source;
- }
- public Expression GetProperty(Expression source, ParameterExpression para, string Name)
- {
- string[] propertys = Name.Split('.');
- if (source == null)
- {
- source = Expression.Property(para, typeof(City).GetProperty(propertys.First()));
- }
- else source = Expression.Property(source, propertys.First());
- foreach (var item in propertys.Skip(1))
- {
- source = GetProperty(source , para, item);
- }
- return source;
- }
public Expression GetProperty(Expression source, ParameterExpression para, string Name)
{
string[] propertys = Name.Split('.');
if (source == null)
{
source = Expression.Property(para, typeof(City).GetProperty(propertys.First()));
}
else source = Expression.Property(source, propertys.First());
foreach (var item in propertys.Skip(1))
{
source = GetProperty(source , para, item);
}
return source;
}
使用方法:
- ParameterExpression param = Expression.Parameter(typeof(City), "x");
- Expression left = GetProperty(null, param, "Province.Name"); //得到查询条件属性
- Expression right = Expression.Constant("湖南省");
- Expression filter = Expression.Equal(left,right);
- ParameterExpression param = Expression.Parameter(typeof(City), "x");
- Expression left = GetProperty(null, param, "Province.Name"); //得到查询条件属性
- Expression right = Expression.Constant("湖南省");
- Expression filter = Expression.Equal(left,right);
EntityFramework4.5使用Expression类创建动态查询及动态查询导航属性的更多相关文章
- 关于Entity Framework自动关联查询与自动关联更新导航属性对应的实体注意事项说明
一.首先了解下Entity Framework 自动关联查询: Entity Framework 自动关联查询,有三种方法:Lazy Loading(延迟加载),Eager Loading(预先加载) ...
- spring AbstractBeanDefinition创建bean类型是动态代理类的方式
1.接口 Class<?> resourceClass 2.获取builder BeanDefinitionBuilder builder = BeanDefinitionBuilder. ...
- 8.mybatis动态SQL模糊查询 (多参数查询,使用parameterType)
多参数查询,使用parameterType.实例: 用户User[id, name, age] 1.mysql建表并插入数据 2.Java实体类 public class User { public ...
- JAVAEE——Mybatis第二天:输入和输出映射、动态sql、关联查询、Mybatis整合spring、Mybatis逆向工程
1. 学习计划 1.输入映射和输出映射 a) 输入参数映射 b) 返回值映射 2.动态sql a) If标签 b) Where标签 c) Sql片段 d) Foreach标签 3.关联查询 a) 一对 ...
- MyBatis学习总结(三)——多表关联查询与动态SQL
在上一章中我们学习了<MyBatis学习总结(二)——MyBatis核心配置文件与输入输出映射>,这一章主要是介绍一对一关联查询.一对多关联查询与动态SQL等内容. 一.多表关联查询 表与 ...
- mybatis第二天——动态SQL与关联查询
大纲摘要: 1.输入映射和输出映射 a) 输入参数映射 b) 返回值映射 2.动态sql a) If b) Where c) Foreach d) Sql片段 3.关联查询 a) 一对一关联 b) 一 ...
- MongoDB动态条件之分页查询
一.使用QueryByExampleExecutor 1. 继承MongoRepository public interface StudentRepository extends MongoRepo ...
- Asp.net Core C#进行筛选、过滤、使用PredicateBuilder进行动态拼接lamdba表达式树并用作条件精准查询,模糊查询
在asp.net core.asp.net 中做where条件过滤筛选的时候写的长而繁琐不利于维护,用PredicateBuilder进行筛选.过滤.LInq配合Ef.core进行动态拼接lamdba ...
- ASP.NET MVC with Entity Framework and CSS一书翻译系列文章之第二章:利用模型类创建视图、控制器和数据库
在这一章中,我们将直接进入项目,并且为产品和分类添加一些基本的模型类.我们将在Entity Framework的代码优先模式下,利用这些模型类创建一个数据库.我们还将学习如何在代码中创建数据库上下文类 ...
随机推荐
- 一、数据库表中字段的增删改查,二、路由基础.三、有名无名分组.四、多app共存的路由分配.五、多app共存时模板冲突问题.六、创建app流程.七、路由分发.八、路由别名,九、名称空间.十、反向解析.十一、2.x新特性.十二、自定义转换器
一.数据库表中字段的增删改查 ''' 直接在modules中对字段进行增删改查 然后在tools下点击Run manage.py Task执行makemigrations和migrate 注意在执行字 ...
- centOS6.8开放防火墙端口
1.找到防火墙配置文件,/etc/sysconfig/iptables.如果是新装的linux系统,防火墙默认是被禁掉的,没有配置任何防火墙策略,所以不存在iptables.需要在控制台使用iptab ...
- ODAC(V9.5.15) 学习笔记(三)TOraSession(1)
1. 连接相关 名称 类型 说明 ConnectDialog 执行连接对话框控件 Connected Boolean 连接状态,通过函数Connect和Disconnect连接或关闭数据库连接,并触发 ...
- dede的使用-2
自己开发站点的规律, 跟cms开发的规律应该是一样的, 而且应该学习和借鉴cms的开发规律方法和思想.比如也要分站点的内容性质, 是文章类站点, 还是图片类, 还是shopping类等. 也要分栏目. ...
- 【做题】neerc2017的A、C、I、L
A - Archery Tournament 一开始往化简公式的方向去想,结果没什么用. 考虑与一条垂线相交的圆的个数.不难YY,当圆的个数最多时,大概就是这个样子的: 我们稍微推一下式子,然后就能发 ...
- js动画(速度)
<!DOCTYPE html> <html> <head> <meta charset="utf-8" /> <meta ht ...
- 51nod 四级题 汇总
51Nod-1060-最复杂的数 #include <bits/stdc++.h> using namespace std; typedef unsigned long long ull; ...
- centos 查看USB接口的版本
# lsusbBus 001 Device 001: ID 1d6b:0002 Linux Foundation 2.0 root hubBus 002 Device 001: ID 1d6b:000 ...
- 【C#】 Method invocation is skipped
相信大家看到这个标题也是一头雾水了. 这个问题主要是我在项目中遇到了一个问题, 然后我通过搜索引擎搜索的关键词进而找到了answer, 我先描述一下我遇到的问题: 做项目的时候我发现log时常没有输出 ...
- BZOJ 1143: [CTSC2008]祭祀river(二分图最大点独立集)
http://www.lydsy.com/JudgeOnline/problem.php?id=1143 题意: 思路: 二分图最大点独立集,首先用floyd判断一下可达情况. #include< ...