Electron+HTML做界面,C#做后台(四)反射实现执行方
2019-01-14 本文已影响0人
黄二狗家的黄二庄
Electron+HTML做界面,C#做后台(三)中简单说明了,如何使用socket获取前后台数据。现在就说说,如何使用反射来获取C#端的业务数据
反射创建类,并调用方法
1.创建 Controller类
using System;
using System.Collections.Generic;
using System.Linq;
using System.Net;
using System.Reflection;
namespace ElectronHTMlCSharp
{
public class Controller
{
#region 反射 执行方法
/// <summary>
/// 反射调用方法入口处理函数
/// action!method.do?a=1&b=2
/// </summary>
/// <param name="str"></param>
/// <returns></returns>
public object EventHander(string str, IPEndPoint client)
{
var Info = str.Split('?');
var classInfo = Info[0].Split('!');
var value = Info.Length > 1 && Info[1] != "undefined" && !string.IsNullOrEmpty(Info[1]) ? Info[1] : null;
var result = ExeMethod(classInfo, value, client);
return result;
}
/// <summary>
/// 执行方法 并返回数据
/// </summary>
public object ExeMethod(string[] classInfo, string value, IPEndPoint client)
{
try
{
var mytypes = GetAllTypes();
foreach (var my in mytypes.Where(my => my.Name.ToLower().Equals(classInfo[0].ToLower() + "controller")))
{
return ExecuteMethod(my, classInfo[1], value, client);
}
throw new Exception("没有找到相关类或方法:" + classInfo[0]);
}
catch (Exception em)
{
return null;
}
}
public static Type[] typeInfo;
/// <summary>
/// 反射得到所有类
/// </summary>
/// <returns></returns>
public Type[] GetAllTypes()
{
//获取程序集 所有类
return typeInfo = typeInfo ?? ReGetAllTypes();
}
/// <summary>
/// 反射得到所有类
/// </summary>
/// <returns></returns>
public static Type[] ReGetAllTypes()
{
//获取程序集 所有类型
string codeBase = Assembly.GetExecutingAssembly().CodeBase;
UriBuilder uri = new UriBuilder(codeBase);
string path = Uri.UnescapeDataString(uri.Path);
var ass = Assembly.LoadFrom(path);
return ass.GetTypes();
}
/// <summary>
/// 反射执行方法
/// </summary>
/// <param name="my"></param>
/// <param name="functionName"></param>
/// <param name="value"></param>
/// <returns></returns>
public static object ExecuteMethod(Type my, string functionName, string value, IPEndPoint client)
{
//实例化对象 实例参数
var objName = Activator.CreateInstance(my);
value += client != null ? (!string.IsNullOrEmpty(value) ? "&" : "") + "Address=" + client.Address + "&Port=" + client.Port : "";
//为对象属性赋值
SetProperty(my, objName, value);
//执行方法
var objMethod = my.GetMethod(functionName);
return objMethod.Invoke(objName, null);
}
//反射 为对象属性赋值
public static void SetProperty(Type type, object objName, string list)
{
var dic = new Dictionary<String, String>();
//拆解参数
if (!string.IsNullOrEmpty(list))
{
var paraList = list.Split('&');
for (var d = 0; d < paraList.Length; d++)
{
var _d = paraList[d].Split('=');
dic[_d[0]] = _d[1];
}
}
//获取当前类的所有属性,并为属性赋值
var propertyInfo = type.GetProperties();
for (var i = 0; i < propertyInfo.Length; i++)
{
var property = propertyInfo[i];
var str = "&" + list;
var p = "&" + property.Name + "=";
if (!("&" + list).Contains("&" + property.Name + "=")) { continue; }
var value = dic[property.Name];
value = value.ToLower() == "true" || value.ToLower() == "false" ? value.ToLower() : value;
if (!property.PropertyType.IsGenericType)
{
//非泛型
property.SetValue(objName, string.IsNullOrEmpty(value) ? null : Convert.ChangeType(value, property.PropertyType), null);
}
else
{
//泛型Nullable<>
var genericTypeDefinition = property.PropertyType.GetGenericTypeDefinition();
if (genericTypeDefinition == typeof(Nullable<>))
{
property.SetValue(objName, string.IsNullOrEmpty(value) ? null : Convert.ChangeType(value, Nullable.GetUnderlyingType(property.PropertyType)), null);
}
}
}
}
#endregion
}
}
2.在前面创建的Services.cs类中,对接受消息MessageReceived
方法进行改造
2.1 申明全局私有静态 变量 private static Controller con = new Controller();
var msg = Encoding.UTF8.GetString(bMsg); //消息使用UTF-8编码
if (msg.IndexOf("!") > 0)
{
//判断是请求方法,使用反射执行结果
var responseText = con.EventHander(msg, msg.IndexOf("isLong=true") > 0?client:null);
if (responseText != null)
{
Console.WriteLine("得到数据" + responseText);
SendMsgToClient(responseText.ToString(), client);
}
}
else if (msg.IndexOf("quit") >= 0)
{
Application.ExitThread();
Application.Exit();
Process.GetCurrentProcess().Kill();
}
else
{
SendMsgToClient("经过服务端转播的消息" + msg, client);
}
Console.WriteLine("接收到客户端消息:" + msg);
如此反射调用类方法就创建好了,那么怎么验证呢,我们需要添加一个类,TestController
,后缀是Controller
,因为在反射执行方法中我们只执行后缀为controller
的里
在这里插入图片描述
2.2创建测试类 TestController.cs
namespace ElectronHTMlCSharp
{
public class TestController
{
public string name { get; set; }
public int age { get; set; }
public string sex { get; set; }
public string info{ get; set; }
public string Test()
{
return string.Format("传递过来的参数值:name:{0} , age:{1} , sex:{2} , info :{3}",name,age,sex,info);
}
}
}
2.3在前面的index.html
页面中修改Hander
参数,这里的参数后期可以根据业务进行更改,动态获取
var msg = "Test!Test?name=张三&age=20&sex=男&info=1427953302";
在这里插入图片描述
2.4 启动程序执行结果
在这里插入图片描述如此就大功告成,可以使用 socket 前后台通讯,基本结构已经具备,只需要实现业务即可
但是
,这里边的传递的数据长度有一定限制,并不适合于,前端请求,后端查询表格,返回表格数据。为此 后面将会建立【 winform的http
】服务来解决这个问题
如果你已经迫不及待,那么请下载整个项目的源代码,代码已放置 GITHUB
源码下载,上面有最新的效果图
每次都是满满的干货