C#提高知识-001:反射的应用和原理(一)

发布时间:2020-07-12 17:21:05 作者:yangyoushan
来源:网络 阅读:254

在项目中,程序集间的相互引用是经常遇到的。比如,主程序引用各分模块,各分模块引用公用程序集,以及平行的程序集间为了某些功能的实现也需要相互引用。这样的引用一方面是迫不得已的选择,而另一方面也反映出系统设计的水平。下面,简单介绍一下C#中的一种机制——反射。反射可以在避免某些情况下的程序集引用问题,比如主程序引用各功能模块的问题,当然其它模块间也是可以用反射的,只是使用是否方便这些问题需要在使用前根据实际情况进行考虑。本文以主程序加载分模块为例,介绍一下反射的使用。
所谓反射,就是对程序集或模块利用基础类型进行解析,然后还原出一个对象模型,在调用者工作域里运行的一个过程。其核心部分就是解析。工作原理是这样的。
无论你创建的多么结构复杂的类,归根结底都是由元数据构成的。如下,

public class Person
 {
     private string name;
     private int  age;
     private string content;
 }

在程序编译时,编译器会创建类型表,字段表,方法表或其它表。再利用System.Reflection命名空间中的包含的类型进行解析,也可以看成对比的过程,将要被反射的程序集中的表读出,根据System.Reflection的基本类型,进行重组,从而还原出原来程序集的结构。
例如,序列化的过程就是使用了反射,序列化格式器将被序列化的对象中的字段的值获取出来,然后写入一个字节流,进行传输;因为字节流传输不容易出错或信息丢失。接收到字节流后,根据基本类型再还原出原对象的模型。
反射中,System.Type类型很重要,它遍历被反射的表中的类型和反射中的基本类型进行比较,然后判断出当前是什么类型。
简单了解了原理,那么再看如何使用的。
建一个工程,包含主程序和子程序集,如图

C#提高知识-001:反射的应用和原理(一)

主程序生成在SetupApp文件夹中,子程序生成在\SetupApp\Library\中。
子程序的程序入口需要遵循一些约定,比如入口类名字需要都一样,这样才可统一加载。

namespace ReflecLibrary2
{
    public class MainWindow
    {
        public MainWindow()
        {
            Welcome();
        }
        private void Welcome()
        {
            Console.Write(@"当前程序为:ReflecLibrary2 ");
            Console.WriteLine(@"开始执行ReflecLibrary2!");
        }
    }
}
namespace ReflectLibrary1
{
    public class MainWindow
    {
        public MainWindow()
        {
            Welcome();
        }
        private void Welcome()
        {
            Console.Write(@"当前程序为:ReflectLibrary1 ");
            Console.WriteLine(@"开始执行ReflecLibrary1!");
        }
    }
}

然后看调用的部分,

class Program
    {
        static void Main(string[] args)
        {
            /////////////////////设置约定的规则,比如需要加载的程序的目录,程序集程序入口的类///////////////////
            string startPath = AppDomain.CurrentDomain.BaseDirectory + @"Library\";
            string suffix=@".dll";
            string commonMainClass = @"MainWindow";
            DirectoryInfo directory = new DirectoryInfo(startPath);
            /////////////////////将程序集文件名读入,这里其实只需要string类型的路径即可,
            //////为了后面处理字符串方便所以才读取文件信息
            var libraries = directory.GetFiles().OrderBy(o=>o.FullName);
            List<FileInfo> loadDlls = new List<FileInfo>();
            if (libraries != null)
            {
                foreach (FileInfo item in libraries)
                {
                    if (item.FullName.ToLower().EndsWith(suffix))
                    {
                        loadDlls.Add(item);
                    }
                }
            }
            /////////////////////执行程序集///////////////////
            //程序集1
            Assembly assembly1 = Assembly.LoadFile(loadDlls[0].FullName.Replace(@"/", @"\"));
            string typeName1 = loadDlls[0].Name.Replace(loadDlls[0].Extension,string.Empty) + @"." + commonMainClass;
            assembly1.CreateInstance(typeName1);
            //程序集2
            Assembly assembly2 = Assembly.LoadFile(loadDlls[1].FullName.Replace(@"/", @"\"));
            string typeName2 = loadDlls[1].Name.Replace(loadDlls[0].Extension, string.Empty) + @"." + commonMainClass;
            assembly2.CreateInstance(typeName2);
            Console.ReadLine();
        }

各个部分的作用都写在了注释中。
运行结果就是,程序集1和程序集2中的方法都执行了。当然这里只是为了方便说明只写了一个方法,实际上
public MainWindow()
{
Welcome();
}
就是子程序的入口。
结果如下,

C#提高知识-001:反射的应用和原理(一)

那么反射我们就有了一个直观的理解,里面的详细原理,下一篇继续介绍

代码下载

推荐阅读:
  1. 反射的应用
  2. 如何提高使用Java反射的效率

免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。

反射加载 多个程序集

上一篇:浅谈共享软件如何不被暴力蹂躏

下一篇:RHEL6基础之六RHEL6基本命令&bash常用快捷键&变量&环境变量&命令的返回值

相关阅读

您好,登录后才能下订单哦!

密码登录
登录注册
其他方式登录
点击 登录注册 即表示同意《亿速云用户服务条款》