C#进阶之路(七)反射的应用

Wesley13
• 阅读 603

反射在C#中的应用还是很多的,但它对代码的性能有一定影响。

反射的性能:

  使用反射来调用类型或者触发方法,或者访问一个字段或者属性时clr 需要做更多的工作:校验参数,检查权限等等,所以速度是非常慢的。所以尽量不要使用反射进行编程,对于打算编写一个动态构造类型(晚绑定)的应用程序,可以采取以下的几种方式进行代替:

1、通过类的继承关系。让该类型从一个编译时可知的基础类型派生出来,在运行时生成该类型的一个实例,将对其的引用放到其基础类型的一个变量中,然后调用该基础类型的虚方法。

2、通过接口实现。在运行时,构建该类型的一个实例,将对其的引用放到其接口类型的一个变量中,然后调用该接口定义的虚方法。

3、通过委托实现。让该类型实现一个方法,其名称和原型都与一个在编译时就已知的委托相符。在运行时先构造该类型的实例,然后在用该方法的对象及名称构造出该委托的实例,接着通过委托调用你想要的方法。这个方法相对与前面两个方法所作的工作要多一些,效率更低一些。

反射创建实例

  System.Activator提供了方法来根据类型动态创建对象,比如创建一个DataTable:

Type  t  =  Type.GetType("System.Data.DataTable,System.Data,Version=1.0.3300.0,  Culture=neutral,  PublicKeyToken=b77a5c561934e089");
DataTable  table  =  (DataTable)Activator.CreateInstance(t);

例二:根据有参数的构造器创建对象

namespace TestSpace
{
    public class TestClass
    {
        private string _value;
        public TestClass(string value)
        {
            _value = value;
        }
    }
}
…
Type  t  =  Type.GetType(“TestSpace.TestClass”);
Object[]  constructParms  =  new  object[]  {“hello”};  //构造器参数
TestClass  obj  =  (TestClass)Activator.CreateInstance(t,constructParms);
…

把参数按照顺序放入一个Object数组中即可

反射执行方法

//获取类型信息
Type t = Type.GetType("TestSpace.TestClass");
//构造器的参数
object[] constuctParms = new object[] { "timmy" };
//根据类型创建对象
object dObj = Activator.CreateInstance(t, constuctParms);
//获取方法的信息
MethodInfo method = t.GetMethod("GetValue");
//调用方法的一些标志位,这里的含义是Public并且是实例方法,这也是默认的值
BindingFlags flag = BindingFlags.Public | BindingFlags.Instance;
//GetValue方法的参数
object[] parameters = new object[] { "Hello" };
//调用方法,用一个object接收返回值
object returnValue = method.Invoke(dObj, flag, Type.DefaultBinder, parameters, null);

动态创建委托

  委托是C#中实现事件的基础,有时候不可避免的要动态的创建委托,实际上委托也是一种类型:System.Delegate,所有的委托都是从这个类派生的

System.Delegate提供了一些静态方法来动态创建一个委托,比如一个委托:

namespace TestSpace
{
    delegate string TestDelegate(string value);
    public class TestClass
    {
        public TestClass()
        {
        }
        public string GetValue(string value)
        {
            return value;
        }
    }
}

使用示例:

TestClass  obj  =  new  TestClass();
//获取类型,实际上这里也可以直接用typeof来获取类型
Type  t  =  Type.GetType(“TestSpace.TestClass”);
//创建代理,传入类型、创建代理的对象以及方法名称
TestDelegate  method  =  (TestDelegate)Delegate.CreateDelegate(t,obj,”GetValue”);
String  returnValue  =  method(“hello”);

批量生成插入SQL

这里要注意的是,你传入的T模型与你数据库中的模型要是相同的。

/// <summary>
/// 批量加入MYSQL数据库
/// </summary>
/// <param name="list"></param>
/// <param name="connectionString"></param>
/// <param name="tableName"></param>
/// <returns></returns>
public static int InsertByList<T>(List<T> list, string connectionString, string tableName)
{
    int count = 0;
    if (list == null || list.Count <= 0) throw new Exception("List无任何数据");
    if (string.IsNullOrEmpty(tableName)) throw new Exception("添加失败!请先设置插入的表名");
    // 构建INSERT语句
    var sb = new StringBuilder();
    sb.Append("Insert into " + tableName + "(");
    Type type = typeof(T);
    foreach (var item in type.GetProperties())
    {
        if (item.Name == "UpdateTime")
        {
            continue;
        }
        sb.Append(item.Name + ",");
    }
    sb.Remove(sb.ToString().LastIndexOf(','), 1);
    sb.Append(") VALUES ");
    foreach (var item in list)
    {
        sb.Append("(");
        foreach (var pi in type.GetProperties())
        {
            if (pi.Name == "UpdateTime")
            {
                continue;
            }

            if (pi.PropertyType.Name == "Nullable`1")//可空类型,判定是否为空
            {
                var aa = type.GetProperty(pi.Name)?.GetValue(item, null);
                if (aa == null)
                {
                    sb.Append("null,");
                }
                else
                {
                    sb.Append("'" + type.GetProperty(pi.Name)?.GetValue(item, null)?.ToString().Replace(@"\",@"\\").Replace("'", @"\'") + "',");
                }
            }
            else
            {
                sb.Append("'" + type.GetProperty(pi.Name)?.GetValue(item, null)?.ToString().Replace(@"\", @"\\").Replace("'", @"\'") + "',");
            }

        }
        sb.Remove(sb.ToString().LastIndexOf(','), 1);
        sb.Append("),");

    }
    sb.Remove(sb.ToString().LastIndexOf(','), 1);
    sb.Append(";");
    sb.Append("select @@IDENTITY");

    using (var con = new MySqlConnection(connectionString))
    {
        con.Open();
        using (var cmd = new MySqlCommand(sb.ToString(), con))
        {
            try
            {
                count = Convert.ToInt32(cmd.ExecuteScalar());
            }
            catch (Exception ex)
            {
                LogHelper.Error("批量sql插入操作失败:" + ex.Message + "--" + ex.Source + "---" + ex.StackTrace);
            }
        }
    }
    return count;
}

判断对象是否已赋值

这里model是要验证的实例

Type type  = model.GetType();
bool res= false;
//判断实例是否是已经赋值, false 未赋值   true 已经赋值
foreach (var item in type.GetProperties())
{
    List<string> skipList=new List<string>{"UpdateTime","Id"};
    if (skipList.Contains(item.Name))
    {
        continue;
    }
    var typeThis =item.PropertyType;
    var valueDefault = typeThis.IsValueType ? Activator.CreateInstance(typeThis) : null;
    var gggggg = item.GetValue(model);

    if (valueDefault==null)
    {
        if (gggggg!=null)
        {
            res = true;
            break;
        }
    }
    //如果直接使用== 其实是对引用地址的对比
    else if (valueDefault.ToString() !=gggggg.ToString())
    {
        res = true;
        break;
    }
}
点赞
收藏
评论区
推荐文章
blmius blmius
3年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
Karen110 Karen110
3年前
一篇文章带你了解JavaScript日期
日期对象允许您使用日期(年、月、日、小时、分钟、秒和毫秒)。一、JavaScript的日期格式一个JavaScript日期可以写为一个字符串:ThuFeb02201909:59:51GMT0800(中国标准时间)或者是一个数字:1486000791164写数字的日期,指定的毫秒数自1970年1月1日00:00:00到现在。1\.显示日期使用
皕杰报表之UUID
​在我们用皕杰报表工具设计填报报表时,如何在新增行里自动增加id呢?能新增整数排序id吗?目前可以在新增行里自动增加id,但只能用uuid函数增加UUID编码,不能新增整数排序id。uuid函数说明:获取一个UUID,可以在填报表中用来创建数据ID语法:uuid()或uuid(sep)参数说明:sep布尔值,生成的uuid中是否包含分隔符'',缺省为
Jacquelyn38 Jacquelyn38
3年前
2020年前端实用代码段,为你的工作保驾护航
有空的时候,自己总结了几个代码段,在开发中也经常使用,谢谢。1、使用解构获取json数据let jsonData  id: 1,status: "OK",data: 'a', 'b';let  id, status, data: number   jsonData;console.log(id, status, number )
Easter79 Easter79
2年前
sql注入
反引号是个比较特别的字符,下面记录下怎么利用0x00SQL注入反引号可利用在分隔符及注释作用,不过使用范围只于表名、数据库名、字段名、起别名这些场景,下面具体说下1)表名payload:select\from\users\whereuser\_id1limit0,1;!(https://o
Peter20 Peter20
3年前
mysql中like用法
like的通配符有两种%(百分号):代表零个、一个或者多个字符。\(下划线):代表一个数字或者字符。1\.name以"李"开头wherenamelike'李%'2\.name中包含"云",“云”可以在任何位置wherenamelike'%云%'3\.第二个和第三个字符是0的值wheresalarylike'\00%'4\
Wesley13 Wesley13
2年前
mysql设置时区
mysql设置时区mysql\_query("SETtime\_zone'8:00'")ordie('时区设置失败,请联系管理员!');中国在东8区所以加8方法二:selectcount(user\_id)asdevice,CONVERT\_TZ(FROM\_UNIXTIME(reg\_time),'08:00','0
Stella981 Stella981
2年前
Django中Admin中的一些参数配置
设置在列表中显示的字段,id为django模型默认的主键list_display('id','name','sex','profession','email','qq','phone','status','create_time')设置在列表可编辑字段list_editable
为什么mysql不推荐使用雪花ID作为主键
作者:毛辰飞背景在mysql中设计表的时候,mysql官方推荐不要使用uuid或者不连续不重复的雪花id(long形且唯一),而是推荐连续自增的主键id,官方的推荐是auto_increment,那么为什么不建议采用uuid,使用uuid究
Python进阶者 Python进阶者
8个月前
Excel中这日期老是出来00:00:00,怎么用Pandas把这个去除
大家好,我是皮皮。一、前言前几天在Python白银交流群【上海新年人】问了一个Pandas数据筛选的问题。问题如下:这日期老是出来00:00:00,怎么把这个去除。二、实现过程后来【论草莓如何成为冻干莓】给了一个思路和代码如下:pd.toexcel之前把这