jjzjj

从“把大象装进冰箱”来谈谈面向对象程序设计

buguge - Keep it simple,stupid 2023-03-28 原文

引子

把大象装进冰箱需要3步:打开冰箱门,把大象装入冰箱,关闭冰箱门。

 

扩展一下,我们考虑把动物装进冰箱的场景。比如,把猪?装进冰箱,把狗?装进冰箱,等等。

 

怎么利用面向对象的思想来进行程序设计呢?

 

talk is cheap, show me the code.

 

把大象装进冰箱的程序设计及实现

把大象装进冰箱,这个比较简单,简单到只有某些初级程序员写不出来。

如下,估计都是这个玩法。

 代码就不贴了。

 

把动物装进冰箱的程序设计及实现

推而广之,要把不同的动物装进冰箱,并且假定每种动物装进冰箱的方式不同。那么,该怎么进行程序设计来实现这个需求呢?

程序结构图

类结构图

 

 

 

model-模型类

Animal是抽象类(你可以new一头大象,一条狗,但你总不能new一个动物吧):

package com.animal2icebox.model;

import lombok.Data;

@Data
public abstract class Animal {
    String name;
}

大象模型类-Elephant:

package com.animal2icebox.model;

import lombok.Data;

@Data
public class Elephant extends Animal {
    int weight;

}

 

狗狗模型类-Dog:

package com.animal2icebox.model;

import lombok.Data;

@Data
public class Dog extends Animal {
    /**
     * 狗狗品种
     */
    String breed;

}

 

重点是把动物装进冰箱的服务类

AnimalIntoIceboxService是抽象基类,提供了把动物装进冰箱的门面方法。但是由于每一种动物装入冰箱的方式各不相同,所以,要暴露接口(抽象方法)给派生类来实现。

package com.animal2icebox.service;

import com.animal2icebox.model.Animal;

public abstract class AnimalIntoIceboxService {
    IceboxService iceboxService=new IceboxService();

    public void putIntoIcebox(Animal animal) {
        System.out.println("=============================");
        iceboxService.openTheDoor();
        input(animal);
        iceboxService.closeTheDoor();
    }

    protected abstract void input(Animal animal);
}

 

把大象装进冰箱-ElephantIntoIceboxService:

package com.animal2icebox.service;

import com.animal2icebox.model.Animal;
import com.animal2icebox.model.Elephant;

public class ElephantIntoIceboxService extends AnimalIntoIceboxService {
    @Override
    protected void input(Animal animal) {
        Elephant elephant = (Elephant) animal;
        System.out.println(String.format("大象信息:name=%s,重量=%s吨", elephant.getName(), elephant.getWeight()));
        int threshold = 5;
        int i = 0;
        for (i = threshold; i <= elephant.getWeight(); i += threshold) {
            System.out.println(String.format("这头大象重%s吨,已放入%s吨", elephant.getWeight(), threshold));
        }
        i -= threshold;
        if (elephant.getWeight() > i) {
            System.out.println(String.format("这头大象重%s吨,已放入%s吨", elephant.getWeight(), elephant.getWeight() - i));
        }
        System.out.println("已成功把大象装进冰箱!");
    }

}
View Code

 

把狗狗装进冰箱-DogIntoIceboxService:

package com.animal2icebox.service;

import com.animal2icebox.model.Animal;
import com.animal2icebox.model.Dog;

public class DogIntoIceboxService extends AnimalIntoIceboxService {
    @Override
    protected void input(Animal animal) {
        System.out.println("撒狗粮 here");
        Dog dog=(Dog)animal;
        String words = String.format("%s(品种=%s)", dog.getName(), dog.getBreed());
        System.out.println("Hi " + words + ", welcome into the icebox...");
    }
}
View Code

 

AnimalIntoIceboxFactory-工厂-生产的产品是上面的装进冰箱的动物service类对象

package com.animal2icebox.factory;

import com.animal2icebox.model.Animal;
import com.animal2icebox.model.Dog;
import com.animal2icebox.model.Elephant;
import com.animal2icebox.service.AnimalIntoIceboxService;
import com.animal2icebox.service.DogIntoIceboxService;
import com.animal2icebox.service.ElephantIntoIceboxService;
import lombok.Getter;

public class AnimalIntoIceboxFactory {
    public static AnimalIntoIceboxService getServiceInstance(Animal animal) {
        Class<AnimalIntoIceboxService> serviceClass = AnimalIntoBoxEnum.getServiceClass(animal);
        assert serviceClass != null;

        try {
            return serviceClass.newInstance();
        } catch (InstantiationException | IllegalAccessException e) {
            e.printStackTrace();
        }
        return null;
    }

    @Getter
    enum AnimalIntoBoxEnum {
        Elephants(Elephant.class, ElephantIntoIceboxService.class),
        Dogs(Dog.class, DogIntoIceboxService.class),
        ;

        Class<Animal> animalClass;
        Class<AnimalIntoIceboxService> serviceClass;

        <E extends Animal, S extends AnimalIntoIceboxService> AnimalIntoBoxEnum(Class<E> animalClass, Class<S> serviceType) {
            this.animalClass = (Class<Animal>) animalClass;
            this.serviceClass = (Class<AnimalIntoIceboxService>) serviceType;
        }

        public static Class<AnimalIntoIceboxService> getServiceClass(Animal animal) {
            AnimalIntoBoxEnum[] values = AnimalIntoBoxEnum.values();
            for (AnimalIntoBoxEnum anEnum : values) {
                if (anEnum.animalClass.equals(animal.getClass())) {
                    return anEnum.getServiceClass();
                }
            }
            return null;
        }
    }
}

 

AnimalIntoIceboxContext-上下文容器,供外部调用

package com.animal2icebox.context;

import com.animal2icebox.factory.AnimalIntoIceboxFactory;
import com.animal2icebox.model.Animal;
import com.animal2icebox.model.Dog;
import com.animal2icebox.model.Elephant;
import com.animal2icebox.service.AnimalIntoIceboxService;

import java.util.ArrayList;
import java.util.List;

public class AnimalIntoIceboxContext {

    public void putAnimalIntoIcebox(List<Animal> animals) {
        for (Animal animal : animals) {
            AnimalIntoIceboxService elephantIntoIceboxService = AnimalIntoIceboxFactory.getServiceInstance(animal);
            assert elephantIntoIceboxService != null;
            elephantIntoIceboxService.putIntoIcebox(animal);
        }
    }
}

 

来吧,开启调用之旅

package com.animal2icebox;

import com.animal2icebox.context.AnimalIntoIceboxContext;
import com.animal2icebox.model.Animal;
import com.animal2icebox.model.Dog;
import com.animal2icebox.model.Elephant;

import java.util.ArrayList;
import java.util.List;

public class AnimalIntoIceboxTest {
    public static void main(String[] args) {
        List<Animal> animals = new ArrayList<>();

        Elephant elephant1 = new Elephant();
        elephant1.setName("Emily");
        elephant1.setWeight(13);
        animals.add(elephant1);

        Dog dog = new Dog();
        dog.setName("Lucky");
        dog.setBreed("dhole");
        animals.add(dog);

        Elephant elephant2 = new Elephant();
        elephant2.setName("包包大人");
        elephant2.setWeight(2);
        animals.add(elephant2);

        AnimalIntoIceboxContext context=new AnimalIntoIceboxContext();
        context.putAnimalIntoIcebox(animals);
    }
}

 

程序运行结果:

=============================
the door is open.
大象信息:name=Emily,重量=13吨
这头大象重13吨,已放入5吨
这头大象重13吨,已放入5吨
这头大象重13吨,已放入3吨
已成功把大象装进冰箱!
the door is closed.
=============================
the door is open.
撒狗粮 here
Hi Lucky(品种=dhole), welcome into the icebox...
the door is closed.
=============================
the door is open.
大象信息:name=包包大人,重量=2吨
这头大象重2吨,已放入2吨
已成功把大象装进冰箱!
the door is closed.

 

so,本文在“把动物装进冰箱”这个需求场景里,使用了哪些设计模式或设计思想呢?

 

装进谈谈spancolorstyle软件设计

有关从“把大象装进冰箱”来谈谈面向对象程序设计的更多相关文章

  1. ruby - 如何从 ruby​​ 中的字符串运行任意对象方法? - 2

    总的来说,我对ruby​​还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用

  2. ruby - 在 Ruby 程序执行时阻止 Windows 7 PC 进入休眠状态 - 2

    我需要在客户计算机上运行Ruby应用程序。通常需要几天才能完成(复制大备份文件)。问题是如果启用sleep,它会中断应用程序。否则,计算机将持续运行数周,直到我下次访问为止。有什么方法可以防止执行期间休眠并让Windows在执行后休眠吗?欢迎任何疯狂的想法;-) 最佳答案 Here建议使用SetThreadExecutionStateWinAPI函数,使应用程序能够通知系统它正在使用中,从而防止系统在应用程序运行时进入休眠状态或关闭显示。像这样的东西:require'Win32API'ES_AWAYMODE_REQUIRED=0x0

  3. ruby-on-rails - Rails - 子类化模型的设计模式是什么? - 2

    我有一个模型:classItem项目有一个属性“商店”基于存储的值,我希望Item对象对特定方法具有不同的行为。Rails中是否有针对此的通用设计模式?如果方法中没有大的if-else语句,这是如何干净利落地完成的? 最佳答案 通常通过Single-TableInheritance. 关于ruby-on-rails-Rails-子类化模型的设计模式是什么?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.co

  4. ruby-on-rails - 按天对 Mongoid 对象进行分组 - 2

    在控制台中反复尝试之后,我想到了这种方法,可以按发生日期对类似activerecord的(Mongoid)对象进行分组。我不确定这是完成此任务的最佳方法,但它确实有效。有没有人有更好的建议,或者这是一个很好的方法?#eventsisanarrayofactiverecord-likeobjectsthatincludeatimeattributeevents.map{|event|#converteventsarrayintoanarrayofhasheswiththedayofthemonthandtheevent{:number=>event.time.day,:event=>ev

  5. ruby - 如何指定 Rack 处理程序 - 2

    Rackup通过Rack的默认处理程序成功运行任何Rack应用程序。例如:classRackAppdefcall(environment)['200',{'Content-Type'=>'text/html'},["Helloworld"]]endendrunRackApp.new但是当最后一行更改为使用Rack的内置CGI处理程序时,rackup给出“NoMethodErrorat/undefinedmethod`call'fornil:NilClass”:Rack::Handler::CGI.runRackApp.newRack的其他内置处理程序也提出了同样的反对意见。例如Rack

  6. ruby - 在 Ruby 中编写命令行实用程序 - 2

    我想用ruby​​编写一个小的命令行实用程序并将其作为gem分发。我知道安装后,Guard、Sass和Thor等某些gem可以从命令行自行运行。为了让gem像二进制文件一样可用,我需要在我的gemspec中指定什么。 最佳答案 Gem::Specification.newdo|s|...s.executable='name_of_executable'...endhttp://docs.rubygems.org/read/chapter/20 关于ruby-在Ruby中编写命令行实用程序

  7. ruby-on-rails - Rails 应用程序之间的通信 - 2

    我构建了两个需要相互通信和发送文件的Rails应用程序。例如,一个Rails应用程序会发送请求以查看其他应用程序数据库中的表。然后另一个应用程序将呈现该表的json并将其发回。我还希望一个应用程序将存储在其公共(public)目录中的文本文件发送到另一个应用程序的公共(public)目录。我从来没有做过这样的事情,所以我什至不知道从哪里开始。任何帮助,将不胜感激。谢谢! 最佳答案 无论Rails是什么,几乎所有Web应用程序都有您的要求,大多数现代Web应用程序都需要相互通信。但是有一个小小的理解需要你坚持下去,网站不应直接访问彼此

  8. ruby - 无法运行 Rails 2.x 应用程序 - 2

    我尝试运行2.x应用程序。我使用rvm并为此应用程序设置其他版本的ruby​​:$rvmuseree-1.8.7-head我尝试运行服务器,然后出现很多错误:$script/serverNOTE:Gem.source_indexisdeprecated,useSpecification.Itwillberemovedonorafter2011-11-01.Gem.source_indexcalledfrom/Users/serg/rails_projects_terminal/work_proj/spohelp/config/../vendor/rails/railties/lib/r

  9. ruby-on-rails - Rails 应用程序中的 Rails : How are you using application_controller. rb 是新手吗? - 2

    刚入门rails,开始慢慢理解。有人可以解释或给我一些关于在application_controller中编码的好处或时间和原因的想法吗?有哪些用例。您如何为Rails应用程序使用应用程序Controller?我不想在那里放太多代码,因为据我了解,每个请求都会调用此Controller。这是真的? 最佳答案 ApplicationController实际上是您应用程序中的每个其他Controller都将从中继承的类(尽管这不是强制性的)。我同意不要用太多代码弄乱它并保持干净整洁的态度,尽管在某些情况下ApplicationContr

  10. ruby-on-rails - 如何验证非模型(甚至非对象)字段 - 2

    我有一个表单,其中有很多字段取自数组(而不是模型或对象)。我如何验证这些字段的存在?solve_problem_pathdo|f|%>... 最佳答案 创建一个简单的类来包装请求参数并使用ActiveModel::Validations。#definedsomewhere,atthesimplest:require'ostruct'classSolvetrue#youcouldevencheckthesolutionwithavalidatorvalidatedoerrors.add(:base,"WRONG!!!")unlesss

随机推荐