• 15.IGame游戏公司的故事[20220624]


    1 讨论会

    话说有一个叫IGame的游戏公司,正在开发一款ARPG游戏(动作&角色扮演类游戏,如魔兽世界、梦幻西游这一类的游戏)。一般这类游戏都有一个基本的功能,就是打怪(玩家攻击怪物,借此获得经验、虚拟货币和虚拟装备),并且根据玩家角色所装备的武器不同,攻击效果也不同。这天,IGame公司的开发小组正在开会对打怪功能中的某一个功能点如何实现进行讨论,他们面前的大屏幕上是这样一份需求描述的ppt:

    各个开发人员,面对这份需求,展开了热烈的讨论,下面我们看看讨论会上都发生了什么。 

    2 实习生小李的实现方式

    在经过一番讨论后,项目组长Peter觉得有必要整理一下各方的意见,他首先询问小李的看法。小李是某学校计算机系大三学生,对游戏开发特别感兴趣,目前是IGame公司的一名实习生。经过短暂的思考,小李阐述了自己的意见:

    “我认为,这个需求可以这么实现。HP当然是怪物的一个属性成员,而武器是角色的一个属性成员,类型可以使字符串,用于描述目前角色所装备的武器。角色类有一个攻击方法,以被攻击怪物为参数,当实施一次攻击时,攻击方法被调用,而这个方法首先判断当前角色装备了什么武器,然后据此对被攻击怪物的HP进行操作,以产生不同效果。”而在阐述完后,小李也飞快的在自己的电脑上写了一个Demo,来演示他的想法,Demo代码如下。

    1. package com.wangxing.game.test1;
    2. /**
    3. * 怪物类
    4. * @author Administrator
    5. *
    6. */
    7. public class GuaiWu {
    8. //怪物名称
    9. private String guaiwu_name;
    10. //怪物生命值
    11. private int guaiwu_hp;
    12. public String getGuaiwu_name() {
    13. return guaiwu_name;
    14. }
    15. public void setGuaiwu_name(String guaiwu_name) {
    16. this.guaiwu_name = guaiwu_name;
    17. }
    18. public int getGuaiwu_hp() {
    19. return guaiwu_hp;
    20. }
    21. public void setGuaiwu_hp(int guaiwu_hp) {
    22. this.guaiwu_hp = guaiwu_hp;
    23. }
    24. }
    25. package com.wangxing.game.test1;
    26. /**
    27. * 角色类
    28. * @author Administrator
    29. *
    30. */
    31. public class JueSe {
    32. //武器变量
    33. private String wuqi;
    34. //通过构造方法设置武器数据值
    35. public JueSe(String wq){
    36. this.wuqi=wq;
    37. }
    38. /**
    39. * 角色的攻击方法
    40. * @param gw
    41. */
    42. public void gongji(GuaiWu gw){
    43. //判断怪物是否死亡,如果已经死亡,则无序继续攻击
    44. if(gw.getGuaiwu_hp()<=0){
    45. System.out.println("怪物-"+gw.getGuaiwu_name()+"-已经死亡!");
    46. return;
    47. }
    48. //根据角色装配的武器不同,对怪物产生不同的效果
    49. //木剑攻击效果
    50. if(this.wuqi.equals("木剑")){
    51. //得到当前怪物的生命值
    52. int dangqianhp=gw.getGuaiwu_hp();
    53. //对当前生命值减20
    54. int lessafterhp=dangqianhp-20;
    55. //重新将损失以后的生命值赋值给怪物
    56. gw.setGuaiwu_hp(lessafterhp);
    57. //继续判断怪物是否死亡
    58. if(gw.getGuaiwu_hp()<=0){
    59. System.out.println("怪物-"+gw.getGuaiwu_name()+"-已经死亡!");
    60. return;
    61. }else{
    62. System.out.println("怪物-"+gw.getGuaiwu_name()+"-损失20hp,还剩"+gw.getGuaiwu_hp()+"hp!");
    63. }
    64. }
    65. //铁剑攻击效果
    66. if(this.wuqi.equals("铁剑")){
    67. //得到当前怪物的生命值
    68. int dangqianhp=gw.getGuaiwu_hp();
    69. //对当前生命值减50
    70. int lessafterhp=dangqianhp-50;
    71. //重新将损失以后的生命值赋值给怪物
    72. gw.setGuaiwu_hp(lessafterhp);
    73. //继续判断怪物是否死亡
    74. if(gw.getGuaiwu_hp()<=0){
    75. System.out.println("怪物-"+gw.getGuaiwu_name()+"-已经死亡!");
    76. return;
    77. }else{
    78. System.out.println("怪物-"+gw.getGuaiwu_name()+"-损失50hp,还剩"+gw.getGuaiwu_hp()+"hp!");
    79. }
    80. }
    81. //魔剑攻击效果
    82. if(this.wuqi.equals("魔剑")){
    83. //得到当前怪物的生命值
    84. int dangqianhp=gw.getGuaiwu_hp();
    85. //对当前生命值减100/暴击200
    86. int random=((int)(Math.random()*10))+1;
    87. int lesshp=0;
    88. if(random>5){
    89. System.out.println("产生暴击!!!");
    90. lesshp=200;
    91. }else{
    92. lesshp=100;
    93. }
    94. int lessafterhp=dangqianhp-lesshp;
    95. //重新将损失以后的生命值赋值给怪物
    96. gw.setGuaiwu_hp(lessafterhp);
    97. //继续判断怪物是否死亡
    98. if(gw.getGuaiwu_hp()<=0){
    99. System.out.println("怪物-"+gw.getGuaiwu_name()+"-已经死亡!");
    100. return;
    101. }else{
    102. System.out.println("怪物-"+gw.getGuaiwu_name()+"-损失"+lesshp+"hp,还剩"+gw.getGuaiwu_hp()+"hp!");
    103. }
    104. }
    105. }
    106. }
    107. //测试类
    108. package com.wangxing.game.test1;
    109. public class TestMain {
    110. public static void main(String[] args) {
    111. //创建怪物对象
    112. GuaiWu gw1=new GuaiWu();
    113. gw1.setGuaiwu_name("巡山小妖");
    114. gw1.setGuaiwu_hp(1000);
    115. //创建角色对象
    116. JueSe js1=new JueSe("木剑");
    117. JueSe js2=new JueSe("铁剑");
    118. JueSe js3=new JueSe("魔剑");
    119. while(gw1.getGuaiwu_hp()>0){
    120. //角色攻击怪物
    121. js3.gongji(gw1);
    122. }
    123. }
    124. }

    3 架构师的建议

    小李阐述完自己的想法并演示了Demo后,项目组长Peter首先肯定了小李的思考能力、编程能力以及初步的面向对象分析与设计的思想,并承认小李的程序正确完成了需求中的功能。但同时,Peter也指出小李的设计存在一些问题,他请小于讲一下自己的看法。

    小于是一名有五年软件架构经验的架构师,对软件架构、设计模式和面向对象思想有较深入的认识。他向Peter点了点头,发表了自己的看法:

    “小李的思考能力是不错的,有着基本的面向对象分析设计能力,并且程序正确完成了所需要的功能。不过,这里我想从架构角度,简要说一下我认为这个设计中存在的问题。

    首先,小李设计的角色类的攻击方法很长,并且方法中有一个冗长的if…else结构,且每个分支的代码的业务逻辑很相似,只是很少的地方不同。

    再者,我认为这个设计比较大的一个问题是,违反了O【Open】C【Close】P原则。在这个设计中,如果以后我们增加一个新的武器,如倚天剑,每次攻击损失500HP,那么,我们就要打开角色类,修改攻击方法。而我们的代码应该是对修改关闭的,当有新武器加入的时候,应该使用扩展完成,避免修改已有代码。

    一般来说,当一个方法里面出现冗长的if…else或switch…case结构,且每个分支代码业务相似时,往往预示这里应该引入多态性来解决问题。而这里,如果把不同武器攻击看成一个策略,那么引入策略模式(Strategy Pattern)是明智的选择。

    最后说一个小的问题,被攻击后,减HP、死亡判断等都是怪物的职责【高内聚】,这里放在角色中有些不当。”

    小于边说,边画了一幅UML类图,用于直观表示他的思想

    Peter让小李按照小于的设计重构Demo,小李看了看小于的设计图,很快完成。相关代码如下: 

    1. package com.wangxing.game.test2;
    2. /**
    3. * 怪物类
    4. * @author Administrator
    5. *
    6. */
    7. public class GuaiWu {
    8. //怪物名称
    9. private String guaiwu_name;
    10. //怪物生命值
    11. private int guaiwu_hp;
    12. public String getGuaiwu_name() {
    13. return guaiwu_name;
    14. }
    15. public void setGuaiwu_name(String guaiwu_name) {
    16. this.guaiwu_name = guaiwu_name;
    17. }
    18. public int getGuaiwu_hp() {
    19. return guaiwu_hp;
    20. }
    21. public void setGuaiwu_hp(int guaiwu_hp) {
    22. this.guaiwu_hp = guaiwu_hp;
    23. }
    24. /**
    25. * 怪物掉血
    26. * @param lesshp
    27. */
    28. public void notify(int lesshp){
    29. //判断怪物是否死亡
    30. if(this.guaiwu_hp<=0){
    31. System.out.println("怪物-"+this.guaiwu_name+"-已经死亡!");
    32. return;
    33. }
    34. //设置怪物掉血
    35. this.guaiwu_hp=this.guaiwu_hp-lesshp;
    36. //判断怪物是否死亡
    37. if(this.guaiwu_hp<=0){
    38. System.out.println("怪物-"+this.guaiwu_name+"-已经死亡!");
    39. return;
    40. }else{
    41. System.out.println("怪物-"+this.guaiwu_name+"-损失"+lesshp+"hp,还剩"+this.guaiwu_hp+"hp!!!");
    42. }
    43. }
    44. }
    45. package com.wangxing.game.test2;
    46. /**
    47. * 武器接口
    48. * @author Administrator
    49. *
    50. */
    51. public interface AttackStrategyInterface {
    52. //攻击怪物方法
    53. void AttackTarget(GuaiWu gw);
    54. }
    55. package com.wangxing.game.test2;
    56. /**
    57. * 角色类
    58. * @author Administrator
    59. *
    60. */
    61. public class JueSe {
    62. //定义角色装配的武器
    63. public AttackStrategyInterface weapon;
    64. /**
    65. * 角色的攻击方法
    66. * @param gw
    67. */
    68. public void Attack(GuaiWu gw){
    69. weapon.AttackTarget(gw);
    70. }
    71. }
    72. package com.wangxing.game.test2;
    73. /**
    74. * 木剑
    75. * @author Administrator
    76. *
    77. */
    78. public class IronSword implements AttackStrategyInterface{
    79. @Override
    80. public void AttackTarget(GuaiWu gw) {
    81. gw.notify(20);
    82. }
    83. }
    84. package com.wangxing.game.test2;
    85. /**
    86. * 铁剑
    87. * @author Administrator
    88. *
    89. */
    90. public class WoodSword implements AttackStrategyInterface{
    91. @Override
    92. public void AttackTarget(GuaiWu gw) {
    93. gw.notify(50);
    94. }
    95. }
    96. package com.wangxing.game.test2;
    97. /**
    98. * 魔剑
    99. * @author Administrator
    100. *
    101. */
    102. public class MagicSword implements AttackStrategyInterface{
    103. @Override
    104. public void AttackTarget(GuaiWu gw) {
    105. int random=((int)(Math.random()*10))+1;
    106. int lesshp=(random>5)?200:100;
    107. if(lesshp==200){
    108. System.out.println("产生暴击!!!");
    109. }
    110. gw.notify(lesshp);
    111. }
    112. }
    113. //测试
    114. package com.wangxing.game.test2;
    115. public class TestMain {
    116. public static void main(String[] args) {
    117. GuaiWu gw=new GuaiWu();
    118. gw.setGuaiwu_name("银角大王");
    119. gw.setGuaiwu_hp(10000);
    120. JueSe js1=new JueSe();
    121. //js1.weapon=new IronSword();
    122. //js1.weapon=new WoodSword();
    123. //js1.weapon=new MagicSword();
    124. js1.weapon=new YiTianJian();
    125. while(gw.getGuaiwu_hp()>0){
    126. js1.Attack(gw);
    127. }
    128. }
    129. }

    Tip:OCP原则,即开放关闭原则,指设计应该对扩展开放,对修改关闭。

    Tip:策略模式,英文名Strategy Pattern,指定义算法族,分别封装起来,让他们之间可以相互替换,此模式使得算法的变化独立于客户。

  • 相关阅读:
    Rust 错误处理
    科尔伯格道德发展阶段论:重点识记,比皮亚杰考频更高。
    学习MySQL-第四章
    APP不存在,AK有误请检查再重试。详情查看: http://lbsyun.baidu.com/apiconsole/key
    3.1依赖注入
    使用vue脚手架快速搭建vue 2项目
    邮件营销:怎么正确地收集邮件地址?
    Sphinx Docstring
    [ffmpeg] 解码
    哈希相关知识
  • 原文地址:https://blog.csdn.net/guizhaiteng/article/details/125448313