先来描述一下问题吧,游戏中的物品原来只有一个属性加成:攻击,防御,获得经验加成,金币加成,等等。现在要增加一个属性,这个属性可以为之前的属性之一。

  这个属性加成涉及到类里的三个属性,value,type,grow_value,在战斗,角色属性预览,还有物品的详细信息里面都有使用到,type表示的是这个加成的类型,value表示这个加成的值,grow_value表示这个加成的成长值。现在加了一个属性,之前的计算方法就都不能用了,而且变量名字也得改,在添加后应该变成,value1,type1,grow_value1,value2,type2,grow_value2,之前所有计算的地方都需要修改,UI显示部分也需要修改。而且这些改动很多,很无脑,很繁杂。(也许有好的修改方法,知道的请在评论里面提示一下。)

  上头就把这个任务给了我,然后告诉我应该这么改,就没了。我改了几个地方发现异常蛋疼,于是就想着有什么好的解决办法。学了1年多的函数式编程,现在终于被我用上了。

  可以把新的属性加成看成是一个新的物品,这个物品只有3个属性value,type,grow_value,我们现在需要的是一个原来的物品附加一个新的物品,这样原来计算的方法都可以使用了,我们要做的只是再计算一次。并不是所有的物品都有两个属性,只是新加的一些物品才会有两种属性,所以再计算的时候判断一下这个物品是否有附加的物品就好了。下面用代码简单地描述一下各个类的修改,我使用的是AS3,类似JS,可以把函数作为一个对象来使用,支持闭包和匿名函数(lambda)。

  第一步在原来的类里面添加三个附加的属性,以及一个attachItem

  

private var _attachType:int;

private var _attachValue:int;

private var _attachGrowValue:int;

private var _attachEquipment:EquipmentData = null;

//三个属性设置为只写
public function set attachType(value:int):void
{
_attachType = value;
} public function set attachValue(value:int):void
{
_attachValue = value;
} public function set attachGrowValue(value:int):void
{
_attachGrowValue = value;
} //附加对象设置为只读
public function get attachEquipment():EquipmentData
{
if(_attachType && _attachValue && _attachGrowValue)
{
_attachEquipment = new EquipmentData();
_attachEquipment.value = _attachValue;
_attachEquipment.valueType = _attachType;
_attachEquipment.growValue = _attachGrowValue;
} return _attachEquipment;
}

  新加的三个属性只在读表的时候用到。另外在使用这个类作为数据,并将这个类的数据进行加工后提供给外部的类中添加两个方法和一个属性。

public class CEquipment
{
private var _preCardXmlData:EquipmentData;
public function CEquipment()
{ } /**
* 当需要计算本装备的属性加成时将需要的计算设置为函数传入
* @param func
*
*/
public function handleProp(func:Function):void
{
func();
handleAttachEquip(func);
} /**
* 仅针对附加属性执行的操作
* @param func
*
*/
public function handleAttachEquip(func:Function):void
{
if((_cCardXmlData as EquipmentXmlData).attachEquipment)
{
_preCardXmlData = _cCardXmlData;
_cCardXmlData = (_cCardXmlData as EquipmentData).attachEquipment;
func();
_cCardXmlData = _preCardXmlData;
}
func = null;
} public function get valueType():int
{
//使用_cCardXmlData 的type各种计算
} public function get value():int
{
//使用_cCardXmlData 的value各种计算
} public function get valueShow():String
{
//使用_cCardXmlData 的value和grow_value各种计算
}

说明一下第一个函数handleProp按照注释说的,可以这样修改原来调用那3个属性计算的方法。比如原来是:

      switch(cEquipment.valueType)
{
case CardConst.ATTACK:
_cEquipmentAttackMin += cEquipment.value;
_cEquipmentAttackMax += cEquipment.value;
break;
case CardConst.DEFENSE:
_cEquipmentDefenseMin += cEquipment.value;
_cEquipmentDefenseMax += cEquipment.value;
break;
case CardConst.MINATTACKMILLI:
_equipmentAttackMinMilli += cEquipment.value / 1000;
break;
case CardConst.MAXATTACKMILLI:
_equipmentAttackMaxMilli += cEquipment.value / 1000;
break;
case CardConst.MINDEFENSEMILLI:
_equipmentDefenseMinMilli += cEquipment.value / 1000;
break;
case CardConst.MAXDEFENSEMILLI:
_equipmentDefenseMaxMilli += cEquipment.value / 1000;
break;
}

只要这样处理就好:

        cEquipment.handleProp(function():void{
switch(cEquipment.valueType)
{
case CardConst.ATTACK:
_cEquipmentAttackMin += cEquipment.value;
_cEquipmentAttackMax += cEquipment.value;
break;
case CardConst.DEFENSE:
_cEquipmentDefenseMin += cEquipment.value;
_cEquipmentDefenseMax += cEquipment.value;
break;
case CardConst.MINATTACKMILLI:
_equipmentAttackMinMilli += cEquipment.value / 1000;
break;
case CardConst.MAXATTACKMILLI:
_equipmentAttackMaxMilli += cEquipment.value / 1000;
break;
case CardConst.MINDEFENSEMILLI:
_equipmentDefenseMinMilli += cEquipment.value / 1000;
break;
case CardConst.MAXDEFENSEMILLI:
_equipmentDefenseMaxMilli += cEquipment.value / 1000;
break;
}
});

下面说明一下原理,将原来的计算过程作为一个函数传入CEquipment类中,在handleProp中立刻进行调用,这样就计算完成了第一个属性。然后handleProp将函数传给了handlerAttachEquip函数,这个函数判断物品是否有附加的物品,如果有,则保存原来的物品数据,暂时将物品的数据替换为附加的物品数据,然后再调用传入的计算函数,这时候计算所使用的值就是附加物品的值。当计算完毕后,又将物品的数据还原。感觉就像变魔术一样。

 在UI使用上,只需要将UI的代码作为函数传到handlerAttachEquip函数中就好了,如果有第二个属性则显示第二个属性,否则就不显示。

可惜我这个做法得不到上面的认同,写完后,上面说看不懂,而且说我的代码看了他想死。。

还好我们经过漫长的讨论后,他说这次就算了,下次不能再使用这个方法。

今天发布在这里感慨一下,希望各位使用新技术的不要气馁,美好的时代会来临的。

使用闭包和lambda解决问题与常规方式解决问题的对比。的更多相关文章

  1. 如何设计一门语言(七)——闭包、lambda和interface

    人们都很喜欢讨论闭包这个概念.其实这个概念对于写代码来讲一点用都没有,写代码只需要掌握好lambda表达式和class+interface的语义就行了.基本上只有在写编译器和虚拟机的时候才需要管什么是 ...

  2. 闭包、lambda和interface

    闭包.lambda和interface 人们都很喜欢讨论闭包这个概念.其实这个概念对于写代码来讲一点用都没有,写代码只需要掌握好lambda表达式和class+interface的语义就行了.基本上只 ...

  3. C++:几种callable实现方式的性能对比

    C++中几种callable实现方式的性能对比 前言 C++中想实现一个callable的对象,通常有四种方式: std::function:最common的方式,一般会配合std::bind使用. ...

  4. ArrayList和LinkedList的几种循环遍历方式及性能对比分析(转)

    主要介绍ArrayList和LinkedList这两种list的五种循环遍历方式,各种方式的性能测试对比,根据ArrayList和LinkedList的源码实现分析性能结果,总结结论. 通过本文你可以 ...

  5. ArrayList和LinkedList的几种循环遍历方式及性能对比分析

    最新最准确内容建议直接访问原文:ArrayList和LinkedList的几种循环遍历方式及性能对比分析 主要介绍ArrayList和LinkedList这两种list的五种循环遍历方式,各种方式的性 ...

  6. ArrayList和LinkedList的几种循环遍历方式及性能对比分析(转载)

    原文地址: http://www.trinea.cn/android/arraylist-linkedlist-loop-performance/ 原文地址: http://www.trinea.cn ...

  7. HashMap循环遍历方式及其性能对比(zhuan)

    http://www.trinea.cn/android/hashmap-loop-performance/ ********************************************* ...

  8. ArrayList和LinkedList遍历方式及性能对比分析

    ArrayList和LinkedList的几种循环遍历方式及性能对比分析 主要介绍ArrayList和LinkedList这两种list的五种循环遍历方式,各种方式的性能测试对比,根据ArrayLis ...

  9. HashMap循环遍历方式及其性能对比

    主要介绍HashMap的四种循环遍历方式,各种方式的性能测试对比,根据HashMap的源码实现分析性能结果,总结结论.   1. Map的四种遍历方式 下面只是简单介绍各种遍历示例(以HashMap为 ...

随机推荐

  1. Html : 将submit变成像文字一样的按钮

    直接上代码: <html> <head> <title>像文字一样的按钮</title> <style> body{ background- ...

  2. failed: No module named 'catkin_pkg' Make sure that you have installed "catkin_pkg"

    https://stackoverflow.com/questions/43024337/why-this-error-when-i-try-to-create-workspaces-in-ros# ...

  3. Selenium入门系列5 默认不显示的下拉列表元素操作

    本节课程的下拉框是那种默认隐藏,当鼠标移到菜单上下拉框才显示的.如果直接getelement会报错,提示元素不可见: so,得先让下拉列表显示出来再获取元素 用到的新知识: is_display()  ...

  4. DOM(十四):代理检测和事件处理(跨浏览器)

    一.检测 用于用户代理检测,检测范围包括浏览器引擎.平台.Windows.移动设备和游戏系统等 /* *用户代理检测脚本,检测范围包括浏览器引擎.平台.Windows.移动设备和游戏系统 */ var ...

  5. Compass Card Sales(模拟)

    Compass Card Sales 时间限制: 3 Sec  内存限制: 128 MB提交: 35  解决: 13[提交] [状态] [讨论版] [命题人:admin] 题目描述 Katla has ...

  6. 在idea中配置jetty

    第一步:在pom.xml中配置jetty插件 <plugins> <!-- jetty插件 --> <plugin> <groupId>org.mort ...

  7. 读取当前路径,列出xls文件

    import java.io.File; public class GetCurrentDirectory { public String GetDirectory() { File director ...

  8. WebStrome react-native代码智能提示

    1.clone到本地 git clone https://github.com/virtoolswebplayer/ReactNative-LiveTemplate  2,添加ReactNative. ...

  9. Javascript和HTML5的关系

    HTML5是一种新的技术,就目前而言,我们所知的HTML5都是一些标签,但是有了JS之后,这些标签深层的扩展功能才得以实现.       比如video标签,我们对其理解为一个简单的标签,但实际上,v ...

  10. 1060: [ZJOI2007]时态同步

    Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 3610  Solved: 1521[Submit][Status][Discuss] Descript ...