使用闭包和lambda解决问题与常规方式解决问题的对比。
先来描述一下问题吧,游戏中的物品原来只有一个属性加成:攻击,防御,获得经验加成,金币加成,等等。现在要增加一个属性,这个属性可以为之前的属性之一。
这个属性加成涉及到类里的三个属性,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解决问题与常规方式解决问题的对比。的更多相关文章
- 如何设计一门语言(七)——闭包、lambda和interface
人们都很喜欢讨论闭包这个概念.其实这个概念对于写代码来讲一点用都没有,写代码只需要掌握好lambda表达式和class+interface的语义就行了.基本上只有在写编译器和虚拟机的时候才需要管什么是 ...
- 闭包、lambda和interface
闭包.lambda和interface 人们都很喜欢讨论闭包这个概念.其实这个概念对于写代码来讲一点用都没有,写代码只需要掌握好lambda表达式和class+interface的语义就行了.基本上只 ...
- C++:几种callable实现方式的性能对比
C++中几种callable实现方式的性能对比 前言 C++中想实现一个callable的对象,通常有四种方式: std::function:最common的方式,一般会配合std::bind使用. ...
- ArrayList和LinkedList的几种循环遍历方式及性能对比分析(转)
主要介绍ArrayList和LinkedList这两种list的五种循环遍历方式,各种方式的性能测试对比,根据ArrayList和LinkedList的源码实现分析性能结果,总结结论. 通过本文你可以 ...
- ArrayList和LinkedList的几种循环遍历方式及性能对比分析
最新最准确内容建议直接访问原文:ArrayList和LinkedList的几种循环遍历方式及性能对比分析 主要介绍ArrayList和LinkedList这两种list的五种循环遍历方式,各种方式的性 ...
- ArrayList和LinkedList的几种循环遍历方式及性能对比分析(转载)
原文地址: http://www.trinea.cn/android/arraylist-linkedlist-loop-performance/ 原文地址: http://www.trinea.cn ...
- HashMap循环遍历方式及其性能对比(zhuan)
http://www.trinea.cn/android/hashmap-loop-performance/ ********************************************* ...
- ArrayList和LinkedList遍历方式及性能对比分析
ArrayList和LinkedList的几种循环遍历方式及性能对比分析 主要介绍ArrayList和LinkedList这两种list的五种循环遍历方式,各种方式的性能测试对比,根据ArrayLis ...
- HashMap循环遍历方式及其性能对比
主要介绍HashMap的四种循环遍历方式,各种方式的性能测试对比,根据HashMap的源码实现分析性能结果,总结结论. 1. Map的四种遍历方式 下面只是简单介绍各种遍历示例(以HashMap为 ...
随机推荐
- Element-ui安装与使用(网站快速成型工具)
我之所以将Element归类为Vue.js,其主要原因是Element是(饿了么团队)基于MVVM框架Vue开源出来的一套前端ui组件.我最爱的就是它的布局容器!!! 下面进入正题: 1.Elemen ...
- do..while(false)的用法总结
首先要注意: do..while(0) 代表do里面的东西至少被执行一次,在这里仅仅执行一次. 此种用法有三个用处: 代替{}代码块,实现局部作用域.在某些宏定义时非常有用: #define f(x) ...
- 使用ABAP代码创建S/4HANA里的Sales Order
下图是使用ABAP代码创建的S/4HANA的Sales Order的截图: 其中红色区域的值是我代码里硬编码的,而蓝色是函数SD_SALESDOCUMENT_CREATE自己创建的. 来看下代码: D ...
- Leetcode back(215) to be continue
solution discussion https://leetcode.com/problems/kth-largest-element-in-an-array/description/ -- 21 ...
- Xapian简明教程(未完成)
第一章 简介 1.1 简介 Xapian是一个开源的搜索引擎库,它可以让开发者自定义的开发一些高级的的索引和查找因素应用在他们的应用中. 通过阅读这篇文档,希望可以帮助你创建第一个你的索引数据库和了解 ...
- 引用类型(二):Array类型
一.js中的数组与其它语言中的数组的区别1.ECMAScript数组的每一项可以保存任何类型的数据2.ECMAScript数组的大小是可以动态调整的 二.创建数组的基本方式1.使用Array构造函数 ...
- 6.Netbackup-Oracle数据库恢复演练报告(下)
1.1 Oracle数据库恢复 目录 1 安装新的客户端 2 新建异机恢复文件 3 新建数据库实例 4 建立spfile文件 5 建立数据文件夹 6 启动数据库到nomount状态 7 列出备份信息 ...
- iterm2配置项
1. 启动终端Terminal2. 进入当前用户的home目录 输入cd ~3. 创建.bash_profile 输入touch .bash_profile 在导入并应用完颜色方案之后,通 ...
- Vuex基础-Getter
官方地址:https://vuex.vuejs.org/zh/guide/getters.html Vuex 允许我们在 store 中定义“getter”(可以认为是 store 的计算属性).就像 ...
- jQuery deferred 使用心得
因为项目的原因,我接触到了jQuery deferred 的这个神奇的工具,下面我用几个例子,与大家分享我的感悟. 我们有5个很耗时的函数 分别为fA.fB.fC.fD.fE 我们的需求是fA和fB ...