c# 常用反射和表达式树整理
更新: 2021-06-19
反射 local function
local function 就是 Action or Func 来的.
var www = "";
void Abc(string dada)
{
www = dada;
}
var methodInfo = ((Action<string>)(Abc)).Method;
methodInfo.Invoke(((Action<string>)(Abc)).Target, new object[] { "super man" });
// www = "super man"
可以通过强转换成 Action, 然后通过. Method 获取到 method info, 然后 invoke 的时候它没有 instance 但是有 .Target
想要检查某个 class 是否有实现某个接口, 最好的方式是拿它的 interfaces 出来找
public static bool HasImplementInterface(Type classType, Type interfaceType)
{
var requireIsGenericType = interfaceType.IsGenericType;
var requireIsEmptyGeneric = interfaceType.ContainsGenericParameters; // is empty <,> ?
return classType.GetInterfaces().Where(classInterface =>
{
if (requireIsGenericType && requireIsEmptyGeneric)
{
if (!classInterface.IsGenericType) return false;
return classInterface.GetGenericTypeDefinition() == interfaceType;
}
else
{
return classInterface == interfaceType;
}
}).Count() > 0;
}
更新: 2020-02-22
每次想从 js 写好的代码翻译来 c# 就一定会遇到很多动态语言的问题
想写一个 groupby multiple column
linq 一般上用匿名对象 GroupBy(e => new { e.column1, e.column2 })
但是如果 column 是在 runtime 决定的呢?
直接写不出来, 因为匿名函数不可以简单的用反射写. 匿名函数实在 complie 的时候就创建好的了.
一个简单的解决方案是不要用 column 而是用拼接 string 的方式 group by
https://stackoverflow.com/questions/847066/group-by-multiple-columns
GroupBy(e => e.column1 + "分隔符" + e.column2 );
这个方案好不好,我不是很确定,但是简单是真的, 唯一要注意的是一定要加入分隔符.而且 value 最好都是 string
这种比较简单的场景下会比较安全. 如果不加分隔符 2 个 column 字是有可能撞的,
比如 abc + ef 和 ab + cef 最终都会等于 abcef
所有要特别留意哦.
很久没有写反射了,来整理一下呗.
旧的 : https://www.cnblogs.com/keatkeat/p/4819249.html
创建 Type
public class Abc {}
var type = typeof(Abc); // class to type
var type2 = new Abc().GetType(); // instance to type
有泛型的
public class Abc<T, U> {}
var type = typeof(Abc<,>).MakeGenericType(new[] { typeof(string), typeof(string) });
创建实例
public class Abc{
public string name { get; set; }
}
var abc = Activator.CreateInstance(typeof(Abc)) as Abc;
有参数的
public class Abc
{
public Abc(string name) { }
}
var abc = Activator.CreateInstance(typeof(Abc), new object[] { "name" }) as Abc;
有可选参数的
public class Abc
{
public Abc(string name = "dada") { }
}
var abc = Activator.CreateInstance(typeof(Abc), BindingFlags.OptionalParamBinding, null, new object[] { Type.Missing }, CultureInfo.CurrentCulture) as Abc;
几个点留意一下
一定要有 BindingFlags.OptionalParamBinding, 也有人放完. BindingFlags.CreateInstance | BindingFlags.Public | BindingFlags.Instance
BindingFlags.Instance和BindingFlags.Static二者必须有一项或者都有 (refer: https://blog.csdn.net/weixin_38109688/article/details/80147535)
Type.Missing 表示没有传 value
CultureInfo.CurrentCulture 不清楚用来干嘛.
获取方法
refer : https://stackoverflow.com/questions/3631547/select-right-generic-method-with-reflection 2个高赞的回答
一个是使用了 where 的方式去过滤方法.
方法重载主要是看generic, parameters (return 不看)
generic 只看数量 (where T : class 这个不管的)
parameters 就看数量和类型
步骤大概是把方法拿出来, 找 generic count -> make generic -> 检查所有参数数量类型和返回值. 这样就找到了.
第二种方法比较简单但是前提是你必须知道你要的类.
很巧妙的利用了 new Func 和 new Action 来选择方法, 泛型就用 object 后来补上.
public static MethodInfo GetMethod(Type classType, string methodName, Type[] paramTypes = null!, Type[] genericTypes = null!)
{
paramTypes ??= new Type[] { };
genericTypes ??= new Type[] { };
return classType.GetMethods()
.Where(m => m.Name == methodName && m.GetGenericArguments().Count() == genericTypes.Count() && m.GetParameters().Count() == paramTypes.Count())
.Select(m => genericTypes.Count() > 0 ? m.MakeGenericMethod(genericTypes) : m)
.Single(m => m.GetParameters().Select(p => p.ParameterType).SequenceEqual(paramTypes));
} public static ConstructorInfo GetConstructor(Type classType, Type[] paramTypes = null!) {
paramTypes ??= new Type[] { };
return classType.GetConstructors()
.Where(m => m.GetParameters().Count() == paramTypes.Count())
.Single(m => m.GetParameters().Select(p => p.ParameterType).SequenceEqual(paramTypes));
}
获取还没有泛型的 Type
var isOwnedBuilder = builderType.GetGenericTypeDefinition() == typeof(OwnedNavigationBuilder<,>);
lambda 获取属性
// e =>
var entityAsLambdaParameterExp = Expression.Parameter(entityClrType, "e");
// e.type
var entityDotPropertyExp = Expression.Property(entityAsLambdaParameterExp, property);
// e => e.type
var getPropertyLambdaExp = Expression.Lambda(
entityDotPropertyExp,
entityAsLambdaParameterExp
);
判断值类型
if (property.PropertyType.IsValueType || property.PropertyType == typeof(string))
判断有某个标签
var isHtmlContent = property.GetCustomAttribute<HTMLContentAttribute>() != null;
判断 Enum
if (property.PropertyType.IsEnum)
判断泛型 List
var isSSources = property.PropertyType.IsGenericType &&
property.PropertyType.GetGenericTypeDefinition() == typeof(List<>) &&
property.PropertyType.GetGenericArguments()[0] == typeof(SSource);
c# 常用反射和表达式树整理的更多相关文章
- c#反射优化 表达式树
using System; using System.Collections.Generic; using System.Linq; using System.Linq.Expressions; us ...
- 添加一种emit的应用,反射发出,较直接调用稍慢,但好过反射与表达式树。
System.Reflection.MethodInfo mInfo = typeof(TypeParse).GetMethod("Add", System.Reflection. ...
- C# 反射 表达式树 模糊搜索
反射实体T,非datetime字段反射获取表达式树 public static Expression<Func<T, bool>> GetSearchExpression& ...
- 程序猿修仙之路--数据结构之你是否真的懂数组? c#socket TCP同步网络通信 用lambda表达式树替代反射 ASP.NET MVC如何做一个简单的非法登录拦截
程序猿修仙之路--数据结构之你是否真的懂数组? 数据结构 但凡IT江湖侠士,算法与数据结构为必修之课.早有前辈已经明确指出:程序=算法+数据结构 .要想在之后的江湖历练中通关,数据结构必不可少. ...
- 用lambda表达式树替代反射
本节重点不讲反射机制,而是讲lambda表达式树来替代反射中常用的获取属性和方法,来达到相同的效果但却比反射高效. 每个人都知道,用反射调用一个方法或者对属性执行SetValue和GetValue操作 ...
- Expression 表达式树学习整理
整理了一下表达式树的一些东西,入门足够了 先从ConstantExpression 开始一步一步的来吧 它表示具有常量值的表达式 我们选建一个控制台应用程序 ConstantExpression _ ...
- C#中分别对委托、匿名方法、Lambda表达式、Lambda表达式树以及反射执行同一方法的过程进行比较。
using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.T ...
- (转)Expression 表达式树学习整理
原文地址:http://www.cnblogs.com/li-peng/p/3154381.html 整理了一下表达式树的一些东西,入门足够了 先从ConstantExpression 开始一步一步的 ...
- 利用表达式树Expression优化反射性能
最近做了一个.Net Core环境下,基于NPOI的Excel导入导出以及Word操作的服务封装,涉及到大量反射操作,在性能优化过程中使用到了表达式树,记录一下. Excel导入是相对比较麻烦的一块, ...
- 【C#表达式树 七】 反射在表达式树中的应用 ListInitExpression
以下都是反射在表达式树中的应用 对象初始化 Expression.MemberInit 反射获取成员(字段 或者属性),绑定数据,然后生成 成员表达式节点 class Animal { public ...
随机推荐
- Spring Cloud微服务下如何配置I8n
什么是I8n 国际化(I18n)指的是设计和开发产品的过程,使得它们能够适应多种语言和文化环境,而不需要进行大量的代码更改.这通常涉及到创建一个基础版本的产品,然后通过配置和资源文件来添加对不同语言和 ...
- Apache Hudi X Apache Kyuubi,中国移动云湖仓一体的探索与实践
分享嘉宾:孙方彬 中国移动云能力中心 软件开发工程师 编辑整理:Hoh Xil 出品平台:DataFunTalk 导读:在云原生 + 大数据的时代,随着业务数据量的爆炸式增长以及对高时效性的要求,云原 ...
- WSS SSL HTTPS之间的关系
ssl: secure socket layer 安全套接层,简单来说是一种加密技术,通过它可以在通信的双方上建立一个安全的通信链路,因此数据交互的双方可以安全地通信,而不用担心数据被窃取:wss: ...
- Asp .Net Core 系列:基于 T4 模板生成代码
目录 简介 组成部分 分类 Visual Studio 中使用T4模板 创建T4模板文件 2. 编写T4模板 3. 转换模板 中心控制Manager 根据 MySQL 数据生成 实体 简介 T4模板, ...
- [oeasy]python0072_修改字体前景颜色_foreground_color_font
修改颜色 回忆上次内容 m 可以改变字体样式 0-9 之间设置的都是字体效果 0 重置为默认 1 变亮 2 变暗 3 斜体 4 下划线 5 慢闪 6 快闪 7 前景背景互换 8 隐藏 9 中划线 叠加 ...
- oeasy教您玩转vim - 38 - # 配合移动
快速删除 回忆上节课内容 以前知道可以在插入状态下使用 del.退格 进行删除 现在知道了默认状态下使用通过 x 删除字符 可以在 x 前面使用[count]进行翻倍 如 10x 删除的字符存储在 ...
- OpenStack 基本命令
keystone source /etc/keystone/admin-openrc.sh #登录 openstack user create --password ps1234 --email hq ...
- Jmeter函数助手2-Random
Random函数用于获取随机范围内的正整数或负整数. 一个范围内的最小值:必填,且必须填入整数(正负数都可以) 一个范围内允许的最大值:必填,且必须填入整数(正负数都可以).最大值需大于最小值如[-6 ...
- 9、SpringMVC之处理静态资源
9.1.环境搭建 9.1.1.在project创建新module 9.1.2.选择maven 9.1.3.设置module名称和路径 9.1.4.module初始状态 9.1.5.配置打包方式和引入依 ...
- 【Forza Horizon 5】频繁断网解决办法
参考自文章: https://www.acfun.cn/a/ac32056183_2 简而言之就是玩地平线5的时候不要挂着腾讯的QQ.TIM.微信