EntityFramework4.5使用Expression类创建动态查询及动态查询导航属性
2012-10-22 13:01
459 查看
创建动态查询
想在项目中实现一个灵活的动态查询类,参考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中是可以正常运行的:
纠结万分后找到解决方法,代码如下:
[csharp]
view plaincopyprint?
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 == "湖南省") 呢?,代码如下:
[csharp]
view plaincopyprint?
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方法琢磨了老半天才整明白,写在这方便大家吧。
执行结果:
再贴上自己项目中用的方法
[csharp]
view plaincopyprint?
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;
}
使用方法:
[csharp]
view plaincopyprint?
ParameterExpression param = Expression.Parameter(typeof(City),
"x");
Expression left = GetProperty(null, param,
"Province.Name"); //得到查询条件属性
Expression right = Expression.Constant("湖南省");
Expression filter = Expression.Equal(left,right);
想在项目中实现一个灵活的动态查询类,参考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中是可以正常运行的:
纠结万分后找到解决方法,代码如下:
[csharp]
view plaincopyprint?
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 == "湖南省") 呢?,代码如下:
[csharp]
view plaincopyprint?
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方法琢磨了老半天才整明白,写在这方便大家吧。 执行结果:
再贴上自己项目中用的方法
[csharp]
view plaincopyprint?
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; }
使用方法:
[csharp]
view plaincopyprint?
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类创建动态查询及动态查询导航属性
- EntityFramework4.5使用Expression类创建动态查询及动态查询导航属性
- PHP使用方法重载实现动态创建属性的get和set方法
- 创建动态导航(使用SqlSiteMapProvider)
- PHP使用方法重载实现动态创建属性的get和set方法
- 使用VBA动态创建SQL查询语句
- 《Entity Framework 6 Recipes》中文翻译系列 (26) ------ 第五章 加载实体和导航属性之延缓加载关联实体和在别的LINQ查询操作中使用Include()方法
- 使用 JdbcTemplate 动态创建表并添加数据 动态连表查询
- PHP使用方法重载实现动态创建属性的get和set方法
- 使用运行时动态创建属性
- 形成查询结果(实体框架)- 使用导航属性导航关系
- 使用innerHTML属性创建动态内容
- 在Django中使用自定义Tag创建动态导航栏
- 使用ascx动态创建webpart
- 使用动态属性来自定制QT样式表单
- 使用Visual Studio创建简单的自定义Web Part 部件属性
- 如何使用CCRenderTexture创建动态纹理 Cocos2d-x 2.1.4 含iOS版源代码
- 使用jQuery+CSS如何创建流动导航菜单-Fluid Navigation
- 泛型约束where条件的使用(可以通过类型参数动态反射创建实例)
- 如何创建和使用动态链接(dll)和静态链接(lib),全面总结