简单介绍

反射提供了描写叙述程序集、模块和类型的对象(Type 类型)。

能够使用反射动态创建类型的实例,将类型绑定到现有对象,或从现有对象获取类型并调用其方法或訪问其字段和属性。

假设代码中使用了特性。能够利用反射来訪问它们。

这里的类型信息包含类型的方法,变量名称。类型等信息。

基于反射机制的工厂模式

例如以下图所看到的,游戏中经常使用的掉落物模型,Item是基类,定义了一些基础属性,也定义了一些abstract方法。

Food和Weapon继承自Item。表示一类Item。再下一层的类就定义了详细的Item。代码例如以下:

Item.cs

using UnityEngine;
using System.Collections; public abstract class Item {
protected string name;
protected int level;
protected int durability;
protected int maxDurability = 10;
protected bool isStackable;
protected string describe; public string Name
{
get { return name; }
set { name = value; }
} public string Level
{
get { return name; }
set { name = value; }
} public int Durability
{
get { return durability; }
set { durability = value; }
} public abstract void Execute(); public void Upgrade()
{
level++;
} public void Fix()
{
durability = maxDurability;
} public virtual bool IsEquipped()
{
return false;
} }

Food.cs

using UnityEngine;
using System.Collections;
using System; public class Food : Item { public Food()
{
isStackable = true;
name = "Food";
} public override void Execute()
{
Debug.Log("Eat " + name);
}
}

Weapon.cs

using UnityEngine;
using System.Collections;
using System; public class Weapon : Item
{
public override void Execute()
{
Debug.Log("Use Weapon " + name);
}
}

FrozenCarpaccio.cs

public class FrozenCarpaccio : Food {

    public FrozenCarpaccio()
{
name = "=FrozenCarpaccio";
}
}

Sword.cs

public class Sword : Weapon {

    public Sword()
{
name = "Sword";
}
}

代码简单意思一下,详细到游戏肯定有很多其它的属性和方法。

如今出现的要求是依据类名来动态创建对象。

常见的方法就是一堆的switchcase....

以下用反射的方式来处理。

工厂类例如以下

using UnityEngine;
using System.Collections;
using System.Collections.Generic;
using System; public class ItemFactory {
public static Dictionary<string, Type> foodClassesDict = new Dictionary<string, Type>();
public static Dictionary<string, Type> weaponClassesict = new Dictionary<string, Type>(); public static void CollectAllEntityClasses()
{
foodClassesDict.Clear();
weaponClassesict.Clear();
System.Reflection.Assembly[] AS = System.AppDomain.CurrentDomain.GetAssemblies(); for (int i = 0; i < AS.Length; i++)
{
Type[] types = AS[i].GetTypes();
for (int j = 0; j < types.Length; j++)
{
string className = types[j].Name;
if (types[j].IsSubclassOf(typeof(Food)))
{
Debug.Log("Food" + className);
foodClassesDict.Add(className, types[j]);
} else if (types[j].IsSubclassOf(typeof(Weapon)))
{
Debug.Log("Weapon" + className);
weaponClassesict.Add(className, types[j]);
}
}
}
} public static Food CreateFoodByClassName(string name)
{
Type foodType = null;
if (foodClassesDict.TryGetValue(name, out foodType))
{
return Activator.CreateInstance(foodType) as Food;
}
else
{
return null;
}
} public static Weapon CreateWeaponByClassName(string name)
{
Type weaponType = null;
if (weaponClassesict.TryGetValue(name, out weaponType))
{
return Activator.CreateInstance(weaponType) as Weapon;
}
else
{
return null;
}
}
}

代码很easy。在使用工厂之前,首先要通过反射。将类型的信息都记录到相应的Dictionary里面。 创建对象的时候仅仅要调用相应的静态方法就能够了。

測试代码

using UnityEngine;
using System.Collections; public class ItemGenerator : MonoBehaviour {
ItemFactory itemFactory;
// Use this for initialization
void Start () {
ItemFactory.CollectAllEntityClasses();
itemFactory = new ItemFactory();
} // Update is called once per frame
void Update () {
if(Input.GetKeyDown(KeyCode.F5))
{
MysteryMeat mysteryMeat = ItemFactory.CreateFoodByClassName("MysteryMeat") as MysteryMeat;
mysteryMeat.Execute();
} if (Input.GetKeyDown(KeyCode.F6))
{
Dagger dagger = ItemFactory.CreateWeaponByClassName("Dagger") as Dagger;
dagger.Execute();
}
}
}

执行结果

添加可配置脚本

如今的需求是,须要用json脚本来配置一些item的属性,这样做的优点是显而易见的 - 灵活!

json的内容例如以下:

[
{
"class": "FrozenCarpaccio",
"name":"Frozen Carpaccio",
"level":"1",
"describe":"It's a piece of frozen raw meat. The only way to eat it is by cutting thin slices of it. And this way it's suprisingly good."
},
{
"class": "MysteryMeat",
"name":"the MysteryMeat",
"level":"1",
"describe":"Eat at your own risk!"
},
]

详细的思路是在工厂中加入一个静态函数。用于载入全部class的配置属性。用json data的方式存起来。在创建相应类的时候用存好的jsondata来给相应的变量赋值。

在Factory中加入

public static Dictionary<string, JsonData> classInfoDict = new Dictionary<string, JsonData>();

相应的方法

 public static void CollectAllItemInfo()
{
classInfoDict.Clear();
TextAsset[] foodTables = Resources.LoadAll<TextAsset>("Data/Food");
foreach (TextAsset table in foodTables)
{
string jsonStr = table.text;
JsonData content = JsonMapper.ToObject(jsonStr);
if (content != null)
{
foreach (JsonData subclass in content)
{
if (LitJsonUtil.JsonDataContainsKey(subclass, "class"))
{
string classname = subclass["class"].ToString();
if (!classInfoDict.ContainsKey(classname))
{
classInfoDict.Add(classname, subclass);
}
}
} }
}
}

在Item类中加入虚方法,用于初始化

    public abstract void InitializeByJsonData(JsonData data);

Food类中加入实现

  public override void InitializeByJsonData(JsonData data)
{
if (LitJsonUtil.JsonDataContainsKey(data, "name"))
{
name = data["name"].ToString();
}
if (LitJsonUtil.JsonDataContainsKey(data, "level"))
{
level = Int32.Parse(data["level"].ToString());
}
if (LitJsonUtil.JsonDataContainsKey(data, "describe"))
{
describe = data["describe"].ToString();
}
}

假设子类中有特殊的属性药初始化,能够通过override这种方法来处理。

这里还能够通过反射获取类型的变量名来自己主动匹配json中的key和类型的成员。然后通过SetValue方法来进行赋值。

工厂里面创建FoodItem相应的方法也要略微改一下

    public static Food CreateFoodByClassName(string className)
{
Type foodType = null;
if (foodClassesDict.TryGetValue(className, out foodType))
{
Food tmp = Activator.CreateInstance(foodType) as Food;
tmp.InitializeByJsonData(classInfoDict[className]);
return tmp;
}
else
{ return null;
}
}

測试代码不变,能够选择把describe打印出来看看。

參考

反射(C# 和 Visual Basic) - https://msdn.microsoft.com/zh-cn/library/ms173183.aspx

动态载入和使用类型 - https://msdn.microsoft.com/zh-cn/library/k3a58006.aspx

C# 反射(Reflection)- http://www.runoob.com/csharp/csharp-reflection.html

详细解释C#编程中的反射机制与方法 - http://developer.51cto.com/art/200904/118971_all.htm

LitJson - https://lbv.github.io/litjson/

基于C#反射机制的工厂模式的更多相关文章

  1. Java反射机制demo(七)—反射机制与工厂模式

    Java反射机制demo(七)—反射机制与工厂模式 工厂模式 简介 工厂模式是最常用的实例化对象模式. 工厂模式的主要作用就是使用工厂方法代替new操作. 为什么要使用工厂模式?直接new不好吗? 直 ...

  2. Java反射机制在工厂模式中的应用

    在本篇文章中就不详细介绍工厂模式,主要介绍一下反射在工厂模式中的使用,让读者对反射机制带来的好处有更深的认识. 首先看一下简单工厂模式 简单工厂模式(simple factory)是类的创建模式,又叫 ...

  3. Java反射机制与工厂模式

    工厂模式属于创建型模式,它提供了一种创建对象的最佳方式. 它的特点是:客户端的程序类不直接牵扯到对象的实例化管理,只与接口发生关联,通过工厂类获取指定接口的实例化对象. 简单工厂模式如下: inter ...

  4. [Java反射机制]用反射改进简单工厂模式设计

    如果做开发的工作,工厂设计模式大概都已经深入人心了,比较常见的例子就是在代码中实现数据库操作类,考虑到后期可能会有数据库类型变换或者迁移,一般都会对一个数据库的操作类抽象出来一个接口,然后用工厂去获取 ...

  5. Java反射机制在代理模式中的使用

    代理模式的核心思路就是一个接口有两个子类,一个子类完成核心的业务操作,另一个子类完成与核心业务有关的辅助性操作. 代理模式分为静态代理模式和动态代理模式.  静态代理模式: //接口类 interfa ...

  6. Java使用反射机制优化工厂方法

    我先举个例子,有一个接口People,这个接口有一个方法: package com.wjy.reflect; public interface People { public abstract voi ...

  7. 反射vs简单工厂模式

    interface Computer { void printpc(); } class lenovo implements Computer { @Override public void prin ...

  8. IOC的实现原理—反射与工厂模式的结合

    反射机制概念   我们考虑一个场景,如果我们在程序运行时,一个对象想要检视自己所拥有的成员属性,该如何操作?再考虑另一个场景,如果我们想要在运行期获得某个类的Class信息如它的属性.构造方法.一般方 ...

  9. Java反射机制及IoC原理

    一. 反射机制概念 主要是指程序可以访问,检测和修改它本身状态或行为的一种能力,并能根据自身行为的状态和结果,调整或修改应用所描述行为的状态和相关的语义.在java中,只要给定类的名字, 那么就可以通 ...

随机推荐

  1. vue .sync 修饰符和自定义v-model的使用

    VUE 是单向数据流 当我们需要对一个 prop 进行"双向绑定"时 vue 修饰符.sync 子组件:this.$emit('update:visible', visible), ...

  2. ANSI转UTF-8中文无乱码解决方案

    近期做的项目需要使用Doxygen生成文档,由于前期代码不是本人完成,他使用的是ANSI格式的文件,后来我用Notepad++写其他文件时,默认保存为UTF-8 无BOM编码格式,因此整个项目文件中既 ...

  3. Android AutoWrapTextView 解决中英文排版问题

    1.概述 最近项目有新需求,UED给了个卡券密码的UI样式,如图: 我一看很简单啊,一个TextView解决问题,然后做好以后在模拟器里一看..... 纳尼,这个时候才想起来,TextView 中英文 ...

  4. 关于python的拷贝

    https://blog.csdn.net/koukehui0292/article/details/82993958 Python的 深度拷贝: import copy d=copy.deepcop ...

  5. Python 极简教程(二)编码工具

    Python 的编码工具很多.目前最流行的是 pycharm,关于 pycharm 的安装使用请参考 PyCharm安装使用教程. 而学习过程中,我觉得最好用的,还是 Python 自带的练习工具 I ...

  6. springmvc hibernate整合

    今天复习一下SpringMVC+Hibernate的搭建,本来想着将Spring-Security权限控制框架也映入其中的,但是发现内容太多 了,Spring-Security的就留在下一篇吧,这篇主 ...

  7. ExtJs4学习(七)MVC中的Store

    Ext.data.Store是extjs中用来进行数据交换和数据交互的标准中间件,不管是Grid还是ComboBox,都是通过它 实现数据读取.类型转换.排序分页和搜索等操作的. Ext.define ...

  8. 51nod Bash游戏(V1,V2,V3,V4(斐波那契博弈))

    Bash游戏V1 有一堆石子共同拥有N个. A B两个人轮流拿.A先拿.每次最少拿1颗.最多拿K颗.拿到最后1颗石子的人获胜.如果A B都很聪明,拿石子的过程中不会出现失误.给出N和K,问最后谁能赢得 ...

  9. 在 AppDelegate 设置屏幕切换

    //禁止横屏显示 - (NSUInteger)application:(UIApplication *)application supportedInterfaceOrientationsForWin ...

  10. [Angular2 Router] Auxiliary Routes bit by bit

    Article Github Auxiliary Routes is is little bit hard to understand. Here is demo, link You can see ...