通过两篇博文,我们了解到,反射是通过非实例化(new)的手段来对对象和对象内的成员访问的,不仅仅如此,反射还可以突破访问修饰符的限制,以上帝视角来窥探对象内部全部成员(字段,属性,方法),包括private成员,这样一来,为我们从另外一个层次去设计程序架构,松散模块耦合,提供了强大而有力的支撑。
本篇博文案例,分离的更彻底,完全上帝视角,说一下前情提要:对于His厂商来说,自己的His只有一份,但His要对接的医保有可能千变万化,His不可能要对接全部医保,所以这里要解耦,那首先要定义一个接口,来完成规范的定义,只要His和医保接口都符合这个规范就行,于是接口出现了:
using System;namespace HisMedical
{/// <summary>/// HIS/// </summary>public interface IHis{/// <summary>/// his登记号/// </summary>string RegisterID { set; }/// <summary>/// 住院登记/// </summary>/// <returns></returns>dynamic Register();/// <summary>/// 缴费/// </summary>/// <returns></returns>dynamic Fee();}
}
这个接口封装成一个单独的dll 叫HisMedical.dll
His中首先要实现这个接口,完成His中所有动作的连动调用,这里做了两个动态,一个是登记住院,一个是住院缴费:
/// <summary>/// his登记住院/// </summary>/// <param name="his"></param>/// <returns></returns>static string Register(IHis his){var registerID = DateTime.Now.ToString("yyyyMMddHHmmss");Console.WriteLine($"*****完成His的登记,登记号:{registerID}");his.RegisterID = registerID;var result = his.Register();return registerID;}/// <summary>/// his缴费/// </summary>/// <param name="his"></param>/// <param name="registerID"></param>/// <returns></returns>static bool Fee(IHis his, string registerID){Console.WriteLine($"*****完成His的结算,登记号:{registerID}");his.RegisterID = registerID;var result = his.Fee();return true;}
接下来实现医保接口的dll就可以了:
先看个东软的:
using HisMedical;
using System;namespace NeusoftMedical
{/// <summary>/// 东软接口/// </summary>public class NeusoftMedical : IHis{/// <summary>/// his登记号/// </summary>public string RegisterID { set; private get; }/// <summary>/// 缴费/// </summary>/// <returns></returns>public dynamic Fee(){Console.WriteLine("-----完成对东软医保的住院缴费");return true;}/// <summary>/// 住院登记/// </summary>/// <returns></returns>public dynamic Register(){Console.WriteLine("-----完成对东软医保的住院登记");return true;}}
}
再看个银海的:
using HisMedical;
using System;namespace YiHaiMedical
{/// <summary>/// 银海接口/// </summary>public class YinHaiMedical : IHis{/// <summary>/// his登记号/// </summary>public string RegisterID { set; private get; }/// <summary>/// 缴费/// </summary>/// <returns></returns>public dynamic Fee(){Console.WriteLine("=====完成对银海医保的住院缴费");return true;}/// <summary>/// 住院登记/// </summary>/// <returns></returns>public dynamic Register(){Console.WriteLine("=====完成对银海医保的住院登记");return true;}}
}
每个接口的实现,依赖RegisterID从His的读库中组织对应的数据就ok,最大限度的与His解耦。
His中究竟是怎么调用不同的接口呢?我们可以把医保的dll的路径配置到一个表里,或配置文件里,当His启动时,自动加载这些dll就ok了,看代码怎么加载。
static void Main(string[] args){while (true){Console.WriteLine("1、东软 2、银海");var no = Console.ReadLine();Console.WriteLine("1、住院登记 2、住院结算");var busNo = Console.ReadLine();var path = "";switch (no){case "1":path = @"C:\MyFile\Source\Repos\Asp.NetCoreExperiment\Asp.NetCoreExperiment\Architecture\NeusoftMedical\bin\Debug\netstandard2.0\NeusoftMedical.dll";break;case "2":path = @"C:\MyFile\Source\Repos\Asp.NetCoreExperiment\Asp.NetCoreExperiment\Architecture\YiHaiMedical\bin\Debug\netstandard2.0\YiHaiMedical.dll";break;}//从文件加载应用程序集并得到具体类型var medicalType = Assembly.LoadFile(path).GetTypes().FirstOrDefault(t => t.GetInterfaces().Where(s => s.Name == "IHis").Count() > 0);IHis his = (IHis)Activator.CreateInstance(medicalType);var registerID = "";switch (busNo){case "1":registerID = Register(his);break;case "2":if (registerID != ""){Fee(his, registerID);}else{Console.WriteLine("请先登记住院");}break;}}}
代码中第一个switch是医保接口选择,看从配置文件中加载那个接口,第二个switch相当于His中执行的某个操作。不难看出通过反射加载dll,最大限度上解耦了His与医保接口,以后有什么新医保接口要对接,对His来说无感(本例中有可能不是无感的,因为这只是一个简单的代码,关于医保异步调用,还有特殊接口对应等,都没有实现)。
通过三篇博文,我们简单看了一下反射的能力,可以让架构人员自由发挥,前两篇,通过集中开发两个转换方法,其他开发都成了体力活(初级程序员就能实现),本篇更让架构师不管具体医保接口的实现,轻松应实现对接,所以说:反射,架构人员法宝