基于Linq表达式做的一个简单的表达式生成器
using System;
using System.Collections.Generic;
using System.ComponentModel.DataAnnotations.Schema;
using System.Linq;
using System.Linq.Expressions; namespace Model
{
[NotMapped]
public class FilterInfo
{
/// <summary>
/// 字段名称
/// </summary>
public string FieldName { get; set; }
/// <summary>
/// 字段值
/// </summary>
public object FieldValue { get; set; }
/// <summary>
/// 操作符
/// </summary>
public FilterOperator Operator { get; set; }
/// <summary>
/// 构建表达式
/// </summary>
/// <returns></returns>
public Func<T, bool> BuildExpression<T>()
{
var constEexp = Expression.Constant(FieldValue, this.FieldValue.GetType());
var paramExp = Expression.Parameter(typeof(T), "x");
Expression expLeft = null;
var tp = typeof(T);
//基元类型直接去paramExp,自定义类型取PropExp
if (!tp.IsPrimitive && tp != typeof(string))
{
expLeft = (Expression)Expression.Property(paramExp, FieldName);
}
else
{
expLeft = paramExp;
}
Expression bExp = null;
switch (Operator)
{
case FilterOperator.GreaterThan://>
bExp = Expression.GreaterThan(expLeft, constEexp);
break;
case FilterOperator.GreaterThanOrEqual://>=
bExp = Expression.GreaterThanOrEqual(expLeft, constEexp);
break;
case FilterOperator.LessThan://<
bExp = Expression.LessThan(expLeft, constEexp);
break;
case FilterOperator.LessThanOrEqul://<=
bExp = Expression.LessThanOrEqual(expLeft, constEexp);
break;
case FilterOperator.OrElse://!=
bExp = Expression.NotEqual(expLeft, constEexp);
break;
case FilterOperator.Like:
if (this.FieldValue.GetType() != typeof(string))
{
throw (new Exception("Like只能用于string类型的字段操作"));
}
bExp = Expression.Call(typeof(FilterInfo).GetMethod("Contains", new Type[] { typeof(string), typeof(string) }), expLeft, constEexp);
break;
default://=
bExp = Expression.Equal(expLeft, constEexp);
break;
}
var exp = Expression.Lambda<Func<T, bool>>(bExp, paramExp);
Console.WriteLine(exp);
return exp.Compile();
}
/// <summary>
/// 用该静态方法实现like
/// </summary>
/// <param name="a"></param>
/// <param name="b"></param>
/// <returns></returns>
public static bool Contains(string a, string b)
{
if (string.IsNullOrEmpty(a) || string.IsNullOrEmpty(b))
{
return false;
}
return a.Contains(b);
}
}
/// <summary>
/// 操作符枚举
/// </summary>
public enum FilterOperator
{
/// <summary>
/// >
/// </summary>
GreaterThan,
/// <summary>
/// =
/// </summary>
Equal,
/// <summary>
/// <
/// </summary>
LessThan,
/// <summary>
/// <=
/// </summary>
LessThanOrEqul,
/// <summary>
/// >=
/// </summary>
GreaterThanOrEqual,
/// <summary>
/// !=
/// </summary>
OrElse,
/// <summary>
/// contain
/// </summary>
Like
}
}
使用该表达式生成器,可以生成简单的匿名查询函数来过滤查询结果。如果我们在项目中用了EF框架,就可以在业务层中通过这个类来构建查询条件然后传到DAL层来执行,使用方法如下:
//整形数组查询
int[] nums = { , , , , , , , , , };
FilterInfo filterInt = new FilterInfo() { FieldName = "x", FieldValue = , Operator = FilterOperator.LessThanOrEqul };
var intFunc = filterInt.BuildExpression<int>();
var r = nums.Where(intFunc);
foreach (var i in r)
{
Console.WriteLine(i);
}
Console.WriteLine("int array search complete");
//字符串数组查询
string[] strArray = { "string1", "string11", "string2", "string2134", "stringOld" };
FilterInfo f = new FilterInfo() { FieldName = "x", FieldValue = "", Operator = FilterOperator.Like };
var fc = f.BuildExpression<string>();
strArray.Where(fc).ToList().ForEach(s => Console.WriteLine(s));
//对象列表查询
List<Org> orglist = new List<Org>();
orglist.Add(new Org() { Guid = Guid.NewGuid().ToString(), Name = "spring" });
for (int i = ; i < ; i++)
{
orglist.Add(new Org() { Guid = Guid.NewGuid().ToString(), Name = "orgName" + Guid.NewGuid().ToString() });
}
FilterInfo filter = new FilterInfo() { FieldName = "Name", FieldValue = "Name", Operator = FilterOperator.Like };
var func = filter.BuildExpression<Org>();
var orgSearchList = orglist.Where(func);
foreach (var o in orgSearchList)
{
Console.WriteLine(o.Name);
}
Console.WriteLine("org list search Complete");
基于Linq表达式做的一个简单的表达式生成器的更多相关文章
- [Ruby on Rails系列]6、一个简单的暗语生成器与解释器(上)
[0]Ruby on Rails 系列回顾 [Ruby on Rails系列]1.开发环境准备:Vmware和Linux的安装 [Ruby on Rails系列]2.开发环境准备:Ruby on Ra ...
- 基于Spring aop写的一个简单的耗时监控
前言:毕业后应该有一两年没有好好的更新博客了,回头看看自己这一年,似乎少了太多的沉淀了.让自己做一个爱分享的人,好的知识点拿出来和大家一起分享,一起学习. 背景: 在做项目的时候,大家肯定都遇到对一些 ...
- 基于gin框架搭建的一个简单的web服务
刚把go编程基础知识学习完了,学习的时间很短,可能还有的没有完全吸收.不过还是在项目中发现知识,然后在去回顾已学的知识,现在利用gin这个web框架做一个简单的CRUD操作. 1.Go Web框架的技 ...
- 一个简单的ruby生成器例子(用连续体Continuation实现)
ruby中有很多经典的驱动器结构,比如枚举器和生成器等.这次简单介绍下生成器的概念.生成器是按照功能要求,一次产生一个对象,或称之为生成一个对象的方法.ruby中的连续体正好可以用来完成生成器的功能. ...
- 7. Swift 基于Xmpp和openfire实现一个简单的登录注册
1. 基本步骤:首先导入Xmpp框架,配置环境 ->由于我们使用的是OC的Xmpp框架,再进行Swift开发时需要进行桥接. 具体方法就是创建一个基于c的.h的头文件,然后将我们需要编译OC的语 ...
- 基于ACE的TAO开发---一个简单的入门实例-----VS2008(二)
上一节已经说了如何编译idl文件.现在就用编好的文件来写一个最小的corba小程序的.程序分为服务器程序和客户端程序. 说明下,代码是<基于C++CORBA高级编程>一书中的例子. 1.首 ...
- JS 的execCommand 方法 做的一个简单富文本
execCommand 当一个 HTML 文档切换到设计模式(designMode)时,文档对象暴露 execCommand 方法,该方法允许运行命令来操纵可编辑区域的内容.大多数命令影响文档的选择( ...
- 基于ACE的TAO开发---一个简单的入门实例-----VS2008(一)
万事开头难,不管做什么事最开始总是最困难的,一旦上手了就好了. 这也是我自己学习corba编程的一点经验和心得.下面的例子主要是保证读者跟着走能立马看到效果. 1.机器上的TAO是实现已经装好的开发版 ...
- 基于傅里叶变换和PyQt4开发一个简单的频率计数器
小学期的<信号与系统>课,要求写一个频率计数器,下面是我个人理解的频率计数 傅里叶变换的代码: # coding=utf-8 import numpy as np from scipy.i ...
随机推荐
- W25Q64BV(FLASH)(SPI)中文手册
64兆位串行SPI FLASH存储器 1.常规介绍 W25Q64BV(64兆位)串行FLASH存储器为一个空间大小,引脚,功耗限制的系统提供解决方案.25Q系列的灵活性和性能良好超越了普通的串行FLA ...
- maven的核心概念——坐标
7.1 几何中的坐标 [1]在一个平面中使用x.y两个向量可以唯一的确定平面中的一个点. [2]在空间中使用x.y.z三个向量可以唯一的确定空间中的一个点. 7.2 Maven的坐标 使用如下三个向量 ...
- Docker容器Centos不能使用systemctl命令问题
注:本文出自博主 Chloneda:个人博客 | 博客园 | Github | Gitee | 知乎 本文源链接:https://www.cnblogs.com/chloneda/p/bug-dock ...
- vs2015运行时提示未加载vcruntime140.adm64.pb
后调试查看发现 vs2015运行时提示未加载vcruntime140.adm64.pb 解决方案:去微软官网下载安装 vc_redist.exe ,安装就可以了.有64位版和32位版,根据计算机配置进 ...
- codeforceCodeForces - 1107G
单调栈 RMQ #include<iostream> #include<cstdio> #include<cmath> #include<cstring> ...
- R 拼接结果展示
学长教的拼接结果展示 哇,R 简直太有魅力了! 晚一点补充
- gulp常用插件之gulp-rev-collector使用
更多gulp常用插件使用请访问:gulp常用插件汇总 gulp-rev-collector这是一款根据gulp-rev生成的manifest.json文件中的映射, 去替换文件名称, 也可以替换路径. ...
- python三器
1.1 装饰器 1.装饰器的作用 1. 装饰器作用:本质是函数(装饰其他函数)就是为其他函数添加其他功能 2. 装饰器必须准寻得原则: 1)不能修改被装饰函数的源代码 2)不能修改被装饰函数的调用方式 ...
- 设置完代理IP,手机连接WiFi时连不上,一直在转
fiddler抓包,设置代理后,会出现手机连不上网的情况,针对这个情况处理方式为: 首先如果已经连接上WiFi了,但是提示“不可上网”,那就把WiFi断开重新连接下. 不行的话就试试下面的办法 一.配 ...
- 洛谷P1402 酒店之王(网络流)
### 洛谷P1402 题目链接 ### 题目大意:有 n 个人, p 间房间,q 种食物.每个人喜欢一些房间,一些食物,但每间房间.每种食物只能分配给一个人.问最大可以让多少个人满足(当且仅当分配到 ...