js реализует 23 шаблона проектирования

Шаблоны проектирования

js реализует 23 шаблона проектирования

Недавно я изучаю 23 шаблона объектно-ориентированного проектирования и реализовал их с помощью java и javascript.Однако, поскольку я сейчас джуниор, у меня больше нет формального опыта в крупных проектах, поэтому я думаю, что процесс обучения сложен без определенный проектный опыт. Чтобы полностью контролировать, поэтому я только примерно изучил характеристики каждого шаблона проектирования в процессе обучения. Я думаю, что в процессе обучения я должен найти сцену для его реализации, чтобы эффект обучения был очень хорошо Следующий код, конечно, только для справки. Вот код:

творческий режим

одноэлементный шаблон

Три элемента:

  • тот же экземпляр
  • Класс создает свой собственный экземпляр объекта
  • Этот экземпляр можно экспортировать во всю систему

Классификация:

  • Голодный китайский стиль (создать объект экземпляра, когда класс загружается в экземпляр)

Пример кода: ленивая формула

// 手机 用来打电话,玩游戏,看电影,且都是同一个手机
// 懒汉式

var Phone = (function () {
    // 规定只能使用 Phone.getInstance 获取实例
    var res = function () {
        throw new Error("Please use Phone.getInstance() to get the object.");
    };
    var has = false;
    var phone = null;
    Object.defineProperty(res, 'getInstance', {
        value: function () { 
            if (has) {
                return phone;
            } else {
                has = true;
                // 调用时才创建实例
                phone = {
                    call () {
                        console.log("打电话");
                    },
                    playGame () {
                        console.log("玩游戏");
                    },
                    watchMovie () {
                        console.log("看电影");
                    }
                }
                return phone;
            }
        },
        writable: false,
        configurable: false,
        enumerable: false
    });
    return res;
}());

var p1 = Phone.getInstance();
var p2 = Phone.getInstance();
var p3 = Phone.getInstance();
p1.call();
p2.playGame();
p3.watchMovie();
console.log(p1 === p2 && p2 === p3);

// 手机 用来打电话,玩游戏,看电影,且都是同一个手机
// 饿汉式
// 规定只能使用 Phone.getInstance 获取实例
var Phone = (function () {
    // 在此创建实例,装入内存时 实例已创建
    var phone = {
        call () {
            console.log("打电话");
        },
        playGame () {
            console.log("玩游戏");
        },
        watchMovie () {
            console.log("看电影");
        }
    };
    var res = function () {
        throw new Error("Please use Phone.getInstance() to get the object.");
    };
    Object.defineProperty(res, 'getInstance', {
        value: function () {
            return phone;
        },
        writable: false,
        configurable: false,
        enumerable: false
    });
    return res;
}());

var p1 = Phone.getInstance();
var p2 = Phone.getInstance();
var p3 = Phone.getInstance();
p1.call();
p2.playGame();
p3.watchMovie();
console.log(p1 === p2 && p2 === p3);

Phone.getInstance = function () {
    console.log("I am Groot!");
}
var p4 = Phone.getInstance();
p4.call();

определение:

Определите фабрику, и фабрика определит метод, который может возвращать экземпляр класса в соответствии с переданными параметрами.

Пример кода:

// 根据参数决定去实例汽车还是摩托车

// 定义接口
const Vehicle = {
    run () {
        console.log(this.name + '跑...');
    }
}

//汽车类
function Car () {
    this.name = "汽车";
}
Car.prototype = Object.assign(Vehicle);

//摩托车类
function Moto () {
    this.name = "摩托车";
}
Moto.prototype = Object.assign(Vehicle);

// 车库
const Garage = {
    chooseVehicle (constructor) {
        return new constructor();
    }
};
Object.freeze(Garage);

var car = Garage.chooseVehicle(Car);
var moto = Garage.chooseVehicle(Moto);
car.run();
moto.run();

заводской метод

краткий:
Определение заводского интерфейса, категория интерфейсного продукта Есть метод для создания, создает такой класс завод для каждого класса.

Пример кода:

// 工厂方法,创建工厂接口,为每个类创建一个工厂

// 定义接口
const Vehicle = {
    run () {
        console.log(this.name + '跑...');
    }
}

//汽车类
function Car () {
    this.name = "汽车";
}
Car.prototype = Object.assign(Vehicle);

//摩托车类
function Moto () {
    this.name = "摩托车";
}
Moto.prototype = Object.assign(Vehicle);

// 汽车车库
function CarGarage () {
    this.createVehicle = function () {
        return new Car();
    }
    Object.freeze(this);
}

// 摩托车车库
function MotoGarage () {
    this.createVehicle = function () {
        return new Moto();
    }
    Object.freeze(this);
}

// 测试
var carGarage = new CarGarage();
var motoGarage = new MotoGarage();

var car = carGarage.createVehicle();
var moto = motoGarage.createVehicle();
car.run();
moto.run();

абстрактная фабрика

краткий:
Можно сказать, что по сравнению с фабричным методом абстрактная фабрика представляет собой множество продуктов, и каждый продукт должен быть реализован фабричным методом.

Пример кода:

// 生产枪的工厂
// createGun 生产枪

// ak47 枪工厂,生产 ak47 枪
function Ak47GunCompany () {
    this.createGun = function () {
        console.log("生产 ak47 枪...");
    }
}
// 巴雷特 枪工厂,生产 巴雷特 枪
function BarrettGunCompany () {
    this.createGun = function () {
        console.log("生产 巴雷特 枪...");
    }
}

// 生产子弹的工厂
// createBubble 生产子弹

// ak47 子弹工厂,生产 ak47 子弹
function Ak47BubbleCompany () {
    this.createBubble = function () {
        console.log("生产 ak47 子弹...");
    }
}
// 巴雷特 子弹工厂,生产 巴雷特 子弹
function BarrettBubbleCompany () {
    this.createBubble = function () {
        console.log("生产 巴雷特 子弹...");
    }
}

// 武器工厂,生产枪和子弹
// createGun 生产枪
// createBubble 生产子弹

// ak47 武器工厂,生产 ak47枪 和 ak47 子弹
function Ak47Company () {
    
    var ak47GunCompany = new Ak47GunCompany();
    var ak47BubbleCompany = new Ak47BubbleCompany();
    this.createGun = function () {
        ak47GunCompany.createGun();
    }
    this.createBubble = function () {
        ak47BubbleCompany.createBubble();
    }    
}

// 巴雷特 武器工厂,生产 巴雷特枪 和 巴雷特子弹
function BarrettCompany () {
    var barrettGunCompany = new BarrettGunCompany();
    var barrettBubbleCompany = new BarrettBubbleCompany();
    this.createGun = function () {
        barrettGunCompany.createGun();
    }
    this.createBubble = function () {
        barrettBubbleCompany.createBubble();
    }
}

var ak47Company = new Ak47Company();
var barrettCompany = new BarrettCompany();
ak47Company.createGun();
ak47Company.createBubble();
barrettCompany.createGun();
barrettCompany.createBubble();

режим строителя

краткий:

Пример кода:

// 适用于 构造一个实例对象需要多个步骤,且每个步骤可能不同会导致不同的实例类

/** 果汁制作步骤
 *  StirFruit()  将水果打碎
 *  addWater() 加水
 */

// 西瓜汁制作步骤
function WatermelonJuiceStep() {
    
    this.StirFruit = function () {
        console.log("将西瓜打碎...");
    };
    this.addWater = function () {
        console.log("加水...");
    }
}

// 橙汁制作步骤
function OrangeJuiceStep() {

    this.StirFruit = function () {
        console.log("将橙子打碎...");
    };
    this.addWater = function () {
        console.log("加水...");
    }
}

/** 果汁生成器
 *  make() 生成果汁
 *  getJuice() 获取制作的果汁
 */

// 西瓜汁生成器
function WatermelonJuiceMaker () {
    
    var maker = new WatermelonJuiceStep();
    this.make = function () {
        maker.StirFruit();
        maker.addWater();
    }
    this.getJuice = function () {
        return maker;
    }
}

// 橙汁生成器
function OrangeJuiceMaker () {

    var maker = new OrangeJuiceStep();
    this.make = function () {
        maker.StirFruit();
        maker.addWater();
    }
    this.getJuice = function () {
        return maker;
    }
}

// 果汁生成器
function JuiceMaker() {
    
    var orangeJuiceMaker = new OrangeJuiceMaker();
    var watermelonJuiceMaker = new WatermelonJuiceMaker();
    this.makeOrangeJuice = function () {
        orangeJuiceMaker.make();
        return orangeJuiceMaker.getJuice();
    }
    this.makeWatermelonJuice = function () {
        watermelonJuiceMaker.make();
        return watermelonJuiceMaker.getJuice();
    }
}

// 使用果汁生成器
var juiceMaker = new JuiceMaker();

var watermelonJuice = juiceMaker.makeWatermelonJuice();
var orangeJuice = juiceMaker.makeOrangeJuice();

Структурный образец

прокси-режим

краткий:

Пример кода:

// 代理模式
// 共同继承同一接口或抽象类,代理类包含被代理类(has-a)

// 场景:帮别人考试

/** 接口(考试人)
 *  test() 参加考试
 */

/** 被代理人考试
 * 
 */
function testSelf() {
    this.test = function() {
        console.log("参加考试,可是我不会.");
    }
}

/** 代理人考试
 * 
 */
function testOther(self) {
    this.test = function() {
        console.log("参加考试,我会.");
    }
}

// 测试

var self = new testSelf();
var other = new testOther(self);
other.test();

режим адаптера

краткий:

Пример кода:

// 适配器模式
// 将两个不能一块工作的接口或者类,通过新建一个类继承两者,从而使得可以一起工作

// 比如小米8的方形耳机插口与圆形耳机接头需要耳机适配器才能工作

/** 手机接口
 *  access() 提供的接口类型
 */

function Mi8() {
    this.access = function () {
        console.log("小米8提供方形插口.");
    }
}

/** 耳机接头
 *  insert() 提供的接头类型
 */

function MiHeadset() {
    this.insert = function () {
        console.log("小米耳机提供圆形插头.");
    }
}

// 适配器 需要实现 手机接口与耳机接头
function HeadsetAdapter() {
    this.access = function () {
        console.log("耳机适配器提供圆形插口.");
    }
    this.insert = function () {
        console.log("耳机适配器提供方形插头.");
    }
}

// 测试
var mi8 = new Mi8();
var miHeadset = new MiHeadset();
var headsetAdapter = new HeadsetAdapter();
mi8.access();
headsetAdapter.insert();
headsetAdapter.access();
miHeadset.insert();

режим моста

краткий:
Основная причина в том, что существует множество видов двух разных классов, а объединение нескольких классов осуществляется посредством метода has-acombination.
Образец кода:

// 桥接模式,主要是两个不同的类有多种种类,通过 has-a 组合方式去进行多种类的结合

/** 场景:
 *  鞋子有跑鞋,篮球鞋,鞋子的品牌有李宁,耐克
 */

/** 接口:鞋柜 ShoesBar
 *  saleShoes()  出售鞋子
 */

/** 继承鞋柜接口:跑鞋鞋柜 
 * 
 */
function RunShoesBar() {
    this.saleShoes = function () {
        console.log("出售跑鞋.");
    }
}

/** 继承鞋柜接口:篮球鞋鞋柜
 *  
 */
function BasketballShoesBar() {
    this.saleShoes = function () {
        console.log("出售篮球鞋.");
    }
}

/** 抽象类:品牌鞋柜
 *  shoesBar 继承 ShoesBar 的鞋柜
 *  saleShoes 鞋柜
 */

/** 品牌鞋柜继承类:李宁鞋柜
 *  
 */
function LiNingShoesBar(shoesBar) {
    var shoesBar = shoesBar;
    this.saleShoes = function () {
        console.log("李宁鞋柜:");
        shoesBar.saleShoes();
    }
}

/** 品牌鞋柜继承类:耐克鞋柜
 * 
 */
function NickShoesBar(shoesBar) {
    var shoesBar = shoesBar;
    this.saleShoes = function () {
        console.log("耐克鞋柜:");
        shoesBar.saleShoes();
    }
}

// 测试
// 定义一个跑鞋柜
var runShoesBar = new RunShoesBar();
// 定义一个李宁的跑鞋柜
var liningRunShoesBar = new LiNingShoesBar(runShoesBar);
liningRunShoesBar.saleShoes();
// 定义一个耐克的跑鞋柜
var nickShoesBar = new NickShoesBar(runShoesBar);
nickShoesBar.saleShoes();

Комбинированный режим

краткий:
Два класса имеют разные уровни очень похожей конфигурации, поэтому для представления этих двух классов может быть создан только один класс.

// 组合模式:就是不同层级的两个类具有极其相似的结构,可以只构造一个类来表示这两个类

// 场景:表示爷爷,爸爸,儿子三代关系

// 接口:属于人,都有名字
const Person = {
    getName() {
        return this.name;
    },
    setName(name) {
        this.name = name;
    },
    display () {

    }
    
}


// 爷爷和爸爸都是父亲,都有儿子,所以...
function Father(name) {
    this.setName(name);
    var sons = [];
    this.add = function (person) {
        sons.push(person);
    }
    this.display = function () {
        console.log("作为父亲:" + this.getName());
        sons.forEach((son) => {
            son.display();
        });
    }
}
// 继承一下
Father.prototype = Person;

// 作为儿子,只能是儿子
function Son(name) {
    this.setName(name);
    this.display = function () {
        console.log("作为儿子:" + this.getName());
    }
}
// 继承以下
Son.prototype = Person;

// 测试
var grandfather = new Father("张爷爷");
var father1 = new Father("张伯伯");
var father2 = new Father("张大爷");
var son1 = new Son("张娃子");
var son2 = new Son("张嘎子");
grandfather.add(father1);
grandfather.add(father2);
father1.add(son1);
father2.add(son2);
grandfather.display();

орнамент

краткий:
В режиме оформления базовый класс и класс оформления совместно наследуют интерфейс или абстрактный класс, Украсьте базовый класс, включив базовый класс в класс декоратора и добавив метод декоратора в класс декоратора.
Пример кода:

// 场景:lol 英雄buff, 普通英雄,露露buff,努努buff

/** 公共接口:Hero
 *  getBuff() 获取英雄的 buff
 */
function NormalHero() {
    this.getBuff = function () {
        console.log("初始的英雄,无 buff.");
    }
}

// 加露露 buff 
function LuLuBuffHero(hero) {
    this.getBuff = function() {
        hero.getBuff();
        console.log("加露露 buff.");
    }
}

// 加努努 buff
function NuNuBuffHero(hero) {
    this.getBuff = function () {
        hero.getBuff();
        console.log("加努努 buff.");
    }
}

// 测试
var noBuffHero = new NormalHero();
var luluBuffHero = new LuLuBuffHero(noBuffHero);
var nunuBuffHero = new NuNuBuffHero(luluBuffHero);
nunuBuffHero.getBuff();

Внешний вид Режим

краткий:
На основе внутреннего управления, при наличии возможности получать сообщения извне через интерфейс, унифицированный класс управления
Пример кода:

/** 外观模式
 *  通过统一的管理类对内部类管理,同时暴露接口接收来自外部类的消息
 */

// 场景描述: 需求人员提出需求,开发人员进行开发,测试人员进行测试
// 需求人员不需要通知开发人员去开发,测试人员去测试
// 只需要告诉小组组长这个需求就可以了

// 开发人员,负责开发需求
function Developter() {
    this.develop = function(demand_name) {
        console.log("开发人员开发需求:" + demand_name);
    }
}

// 测试人员,负责测试需求
function Tester() {
    this.test = function (demand_name) {
        console.log("测试人员测试需求:" + demand_name);
    }
}

// 技术部组长,负责安排开发人员开发和测试人员测试
function Leader() {
    var developer = new Developter();
    var tester = new Tester();
    this.processDemand = function (demand_name) {
        developer.develop(demand_name);
        tester.test(demand_name);
    }
}

// 需求人员,提出需求
function Demander() {
    var leader = new Leader();
    this.demand = function (demand_name) {
        console.log("提出需求:" + demand_name);
        leader.processDemand(demand_name);
    }
}

// 测试
var demand_name = "开发一款MOBA游戏.";
var demander = new Demander();
demander.demand(demand_name);

наилегчайший образец

краткий:
Для некоторых объектов, используемых в системе, которые можно разделить и использовать, то каждый раз, когда вы их используете, вы сначала судите, есть ли они, если они используются напрямую, а не создаете их снова, экономя место в памяти.
Пример кода:

/** 享元模式
 *  对于系统中使用的一些对象可以共享使用,那么每次使用时先判断有没有
 *  有直接使用,没有再去创建,节省内存空间
 */

// 场景:
// 土豪打英雄联盟,想用哪个皮肤,就用哪个
// 有皮肤直接使用,没有就买买买...

/** 英雄皮肤类
 *  name 皮肤名字
 *  show() 皮肤展示
 */
function HeroSkin(name) {
    console.log("玩家购买了" + name + "皮肤");
    this.show = function () {
        console.log("玩家使用了" + name + "皮肤");
    }
}

/** 玩家以及拥有的皮肤
 *  useSkin(skinName) 使用皮肤
 */
function Player() {
    var mySkins = {};
    this.useSkin = function (skinName) {
        if (!(skinName in mySkins)) {
            mySkins[skinName] = new HeroSkin(skinName);
        }
        mySkins[skinName].show();
    }
}

// 测试
var player = new Player();
player.useSkin("伊泽瑞尔-未来战士");
player.useSkin("锐雯-光明使者");
player.useSkin("锐雯-光明使者");

поведенческая модель

краткий:

package actionModel.templateModel;

// 模板方法
// 场景:召唤师选择英雄,皮肤和召唤师技能
// 步骤:选择英雄 -> 选择皮肤 -> 选择召唤师技能1 -> 选择召唤师技能2
// 角色:召唤师(就是玩家)
abstract class Player {
    private String name;
    public Player(String name) {
        this.name = name;
    }
    public abstract void chooseHero();
    public abstract void chooseSkin();
    public abstract void chooseSummonerSkillFirst();
    public abstract void chooseSummonerSkillSecond();
    public void show() {
        // 显示玩家信息
        System.out.println(this.name + "的选择:");
        // 显示选择的英雄
        chooseHero();
        // 显示选择的皮肤
        chooseSkin();
        // 显示选择的召唤师技能一
        chooseSummonerSkillFirst();
        // 显示选择的召唤师技能二
        chooseSummonerSkillSecond();
    }
}

// 玩家小明
class XiaoMing extends Player{
    public XiaoMing(){
        super("小明");
    }
    @Override
    public void chooseHero() {
        System.out.println("英雄:奥拉夫");
    }

    @Override
    public void chooseSkin() {
        System.out.println("皮肤:铁哥们");
    }

    @Override
    public void chooseSummonerSkillFirst() {
        System.out.println("召唤师技能一:疾走");
    }

    @Override
    public void chooseSummonerSkillSecond() {
        System.out.println("召唤师技能二:闪现");
    }
}

// 玩家小张
class XiaoZhang extends Player {
    public XiaoZhang() {
        super("小张");
    }
    @Override
    public void chooseHero() {
        System.out.println("英雄:锐雯");
    }

    @Override
    public void chooseSkin() {
        System.out.println("皮肤:光明使者");
    }

    @Override
    public void chooseSummonerSkillFirst() {
        System.out.println("召唤师技能一:传送");
    }

    @Override
    public void chooseSummonerSkillSecond() {
        System.out.println("召唤师技能二:闪现");
    }
}

public class Test {
    public static void main(String[] args) {
        // 测试
        Player xiaoming = new XiaoMing();
        Player xiaozhang = new XiaoZhang();
        xiaoming.show();
        xiaozhang.show();
    }
}

модель посредника

краткий:

Пример кода:

// 中介者模式
// 简单来说就是通过中介者进行数据传递
// 一方提供数据,一方订阅数据

// 场景:使用第三方买二手手机
// 购买者去预定手机,当出售者卖该型号的手机时候通知购买者

/** 购买者构造函数
 * 
 * @param phoneName 购买人需要的手机
 */
function Buyer(phoneName) {
    this.getPhoneName = function() {
        return phoneName;
    }
    this.callSellerPhone = function(phone) {
        console.log(`联系卖家:${phone}买 ${phoneName}`);
    }
}

/** 出售者构造函数
 *  @param phoneName 卖的的手机
 *  @param phone 卖主联系方式
 */
function Seller(phoneName, phone) {
    this.getPhoneName = function() {
        return phoneName;
    }
    this.getPhone = function() {
        return phone;
    }
}

/** 中介者构造函数
 *  
 */
function Intermediary() {
    var buyerList = [];
    var sellerList = [];
    this.addBuyer = function(buyer) {
        // 若存在一个合适的卖家,直接通知买主,不添加到列表
        for (let i of sellerList) {
            if (i.getPhoneName() === buyer.getPhoneName()) {
                buyer.callSellerPhone(i.getPhone());
                break;
            }
        }

        buyerList.push(buyer);
    }
    this.addSeller = function(seller) {
        // 若存在一个合适的买家,直接通知买主,不添加到列表
        for (let i of buyerList) {
            if (i.getPhoneName() === seller.getPhoneName()) {
                i.callSellerPhone(seller.getPhone());
                break;
            }
        }

        sellerList.push(seller);
    }
}


var intermediary = new Intermediary();
var buyer1 = new Buyer("小米3");
intermediary.addBuyer(buyer1);
var buyer2 = new Buyer("小米8");
intermediary.addBuyer(buyer2);
var seller1 = new Seller("小米8", "15684175538");
intermediary.addSeller(seller1);

командный режим

краткий:
Использование командного режима не может изменить код диспетчерского центра при расширении диспетчерского центра
Пример кода:

// 命令模式
// 使用命令模式可以在扩展调度中心的时候不修改调度中心的代码

// 场景:玩具遥控汽车

/** 汽车构造函数
 * 
 */
function ToyCar() {
    this.goOn = function() {
        console.log("小车前进");
    }
    this.stop = function() {
        console.log("小车停止");
    }
    this.speedUp = function() {
        console.log("小车加速");
    }
}

/** 命令接口 CarCommand
 *  car 遥控汽车的实例
 *  execute() 执行命令
 */

/** 前进的命令 extends CarCommand
 *  @param car 汽车实例 
 */
function GoOnCommand(car) {
    this.execute = function() {
        car.goOn();
    }
}

/** 停止的命令 extends CarCommand
 *  @param car 汽车实例
 */
function StopCommand(car) {
    this.execute = function() {
        car.stop();
    }
}

/** 加速的命令 extends CarCommand
 *  @param car 汽车实例
 */
function SpeedUpCommand(car) {
    this.execute = function() {
        car.speedUp();
    }
}


/** 汽车遥控器
 *  setCarCommand() 设置命令
 *  trigger() 触发命令
 */
function CarControlHandle() {
    var carCommand = null;
    this.setCommand = function(newCarCommand) {
        carCommand = newCarCommand;
    }
    this.trigger = function() {
        carCommand.execute();
    }
}

// 测试

var car = new ToyCar();
var controlHandle = new CarControlHandle();

var goOn = new GoOnCommand(car);
controlHandle.setCommand(goOn);
controlHandle.trigger();

var stop = new StopCommand(car);
controlHandle.setCommand(stop);
controlHandle.trigger();

var speedUp = new SpeedUpCommand(car);
controlHandle.setCommand(speedUp);
controlHandle.trigger();

Режим ответственной цепи

краткий:
Запрос передается в цепочку обработки, и в цепочке обработки есть несколько процессоров. Когда процессор в цепочке обработки обрабатывает запрос, результат обработки возвращается. Преимущество заключается в том, что внутренний код не требует изменяться при добавлении или удалении процессоров. Его необходимо добавлять или удалять в соответствии с принципом открытия-закрытия

Пример кода:

// 责任链模式
// 将请求交给一条处理链,处理链上的有多个处理器处理,
// 当处理链上某个处理器处理了该请求,返回处理的结果
// 优点是添加删除处理器时不需要修改内部代码,只需要添加或者删除即可
// 符合开闭原则

// 场景:dnf 玩家刷图打怪,怪物有普通怪,精英怪,boss

/** 怪物抽象类
 *  nextMonster 下一个怪物
 *  setNextMonster 设置下一个怪物
 *  battle() 和玩家战斗
 *  battleSuccess() 战斗胜利
 *  battalFail() 战斗失败
 */


/** 普通怪
 * 
 */
function NomalMonster() {
    var nextMonster = null;
    this.setNextMonster = function(Monster) {
        nextMonster = Monster;
    }
    this.battle = function(player) {
        var res = Math.round(Math.random() * 10) % 2 === 0;
        if (res) {
            this.battleSuccess();
            nextMonster.battle(player);
        } else {
            this.battleFail();
        }
    }
    this.battleSuccess = function() {
        console.log("打败了普通怪.");
    }
    this.battleFail = function() {
        console.log("被普通怪打死, 请使用复活币");
    }
}

/** 精英怪
 * 
 */
function CreamMonster() {
    var nextMonster = null;
    this.setNextMonster = function(Monster) {
        nextMonster = Monster;
    }
    this.battle = function(player) {
        var res = Math.round(Math.random() * 10) % 2 === 0;
        if (res) {
            this.battleSuccess();
            nextMonster.battle(player);
        } else {
            this.battleFail();
        }
    }
    this.battleSuccess = function() {
        console.log("打败了精英怪.");
    }
    this.battleFail = function() {
        console.log("被精英怪打死, 请使用复活币");
    }
} 


/** Boss怪
 * 
 */
function BossMonster() {
    var nextMonster = null;
    this.setNextMonster = function(Monster) {
        nextMonster = Monster;
    }
    this.battle = function(player) {
        var res = Math.round(Math.random() * 10) % 2 === 0;
        if (res) {
            this.battleSuccess();
        } else {
            this.battleFail();
        }
    }
    this.battleSuccess = function() {
        console.log("打败了boss怪,通关成功!");
    }
    this.battleFail = function() {
        console.log("被boss怪打死, 请使用复活币");
    }
}


/** 玩家类
 * 
 */
function Player() {

}

// 测试
var player = new Player();

var nomalMonster = new NomalMonster();

var creamMonster = new CreamMonster();
nomalMonster.setNextMonster(creamMonster);

var bossMonster = new BossMonster();
creamMonster.setNextMonster(bossMonster);

nomalMonster.battle(player);

режим стратегии

краткий:
Определяет набор алгоритмов, каждый алгоритм упакован вместе и взаимозаменяемы между ними.

Пример кода:

// 策略模式
// 定义一组算法,将每个算法都封装起来,并且使他们之间可以互换

// 场景: 五个人租房子,要么找五室一厅,要么三室一厅 + 二室一厅
// 角色:找房人

/** 方案接口
 *  sayMethod()  输出方案
 */

/** 方案一  五室一厅
 *  sayMethod()  输出方案
 */
function Method1() {
    this.sayMethod = function() {
        console.log("找一个五室一厅.");
    }
}

/** 方案二 三室一厅 + 二室一厅
 *  sayMethod()  输出方案
 */
function Method2() {
    this.sayMethod = function() {
        console.log("找一个三室一厅和一个二室一厅");
    }
}


/** 找房人
 *  method 方案
 *  findHouse() 找房子
 */
function findHousePeople(method) {
    
    this.findHouse = function() {
        method.sayMethod();    
    }
    this.setMethod = function(newMethod) {
        method = newMethod;
    }
}

// 测试
var method = new Method1();
var people = new findHousePeople(method);
people.findHouse();

method = new Method2();
people.setMethod(method);
people.findHouse();

шаблон итератора

краткий:
Учитывая правило обход, независимо от ее структур данных для достижения

Пример кода:

// 迭代器模式
// 给定一个遍历规则,不管其数据结构实现

// 场景:排队拿快递

/** 学生构造函数
 *  
 */
function Student(name, phone, expressId) {
    this.getName = function() {
        return name;
    }
    this.getPhone = function() {
        return phone;
    }
    this.getExpressId = function() {
        return expressId;
    }
}

/** 快递点构造函数
 * 
 */
function ExpressStation() {
    var index = -1;
    var students = [];
    var iterator = null;
    iterator = {
        hasNext() {
            return index < students.length - 1;
        },
        next() {
            index ++;
            return students[index];
        }
    }
    this.getIterator = function() {
        return iterator;
    }
    this.addStudent = function(student) {
        students.push(student);
    }
}

// 测试
var s1 = new Student("张三", "15684175538", "5-68-9");
var s2 = new Student("李四", "15806378470", "5-98-6");

var expressStation = new ExpressStation();
expressStation.addStudent(s1);
expressStation.addStudent(s2);

var iterator = expressStation.getIterator();
while (iterator.hasNext()) {
    var student = iterator.next();
    console.log(`快递员:"下一位"`);
    console.log(`学生:"${student.getExpressId()}"`);
    console.log(`快递员:"姓名,电话"`);
    console.log(`学生: ${student.getName()}, ${student.getPhone()}`);
    console.log();
}

Шаблон наблюдателя

краткий:

Пример кода:

// 观察者模式
// 定义对象间一种一对多的依赖关系,使得每当一个对象改变状态,则所有依赖于它的对象都会得到通知并被自动更新。

// 场景:订阅公众号

/** 观察者接口 Observer
 *  update(String barName, String message) 收到消息
 */

/** 被观察者接口 Observable
 *  addObserver(Observer ob) 添加观察者
 *  removeObserver(Observe ob) 删除观察者
 *  notifyObservers(String message) 通知所有观察者 
 */

/** 微信用户构造函数
 *  
 */
function WxUser() {
    this.update = function(barName, message) {
        console.log(`公众号${barName}发来消息:${message}`);
    }
}

/** 微信公众号构造函数
 *  
 */
function WxBar(name) {
    var obs = new Set();
    this.addObserver = function(ob) {
        obs.add(ob);
    }
    this.removeObserver = function(ob) {
        obs.delete(ob);
    }
    this.notifyObservers = function(message) {
        for (let ob of obs) {
            ob.update(name, message);
        }
    }

}

// 测试
var user1 = new WxUser();
var user2 = new WxUser();

var wxBar = new WxBar("党尼玛的公众号");
wxBar.addObserver(user1);
wxBar.addObserver(user2);
wxBar.notifyObservers("这波超级帅!");

режим состояния

краткий:
Разрешая объекту изменять поведение при изменении его внутреннего состояния, создается впечатление, что объект изменил свой класс. Проще говоря, у объекта есть несколько состояний, и у объекта есть несколько вариантов поведения, и поведение каждого состояния различно.

Пример кода:

// 状态模式
// 当一个对象内在状态改变时允许其改变行为,这个对象看起来像改变了其类。
// 简单讲就是一个对象有多个状态,且这个对象有几个行为,
// 每个状态的这些行为不同

// 场景:文件权限(普通用户只可读,一般管理员可读可写,超级管理员可读可写可删除)

/** 状态接口 State
 *  read()  是否可读
 *  write() 是否可写
 *  delete() 是否可删除
 */

/** 普通用户状态构造函数
 * 
 */
function NomalUser() {
    this.read = function() {
        console.log("可读");
    }
    this.write = function() {
        console.log("不可写");
    }
    this.delete = function() {
        console.log("不可删除");
    }
}

/** 一般管理员状态构造函数
 *  
 */
function Admin() {
    this.read = function() {
        console.log("可读");
    }
    this.write = function() {
        console.log("可写");
    }
    this.delete = function() {
        console.log("不可删除");
    }
}

/** 超级管理员构造函数
 * 
 */
function SuperAdmin() {
    this.read = function() {
        console.log("可读");
    }
    this.write = function() {
        console.log("可写");
    }
    this.delete = function() {
        console.log("可删除");
    }
}

/** 用户构造函数
 * 
 */
function User(state) {
    this.setState = function(newState) {
        state = newState;
    }
    this.readFile = function() {
        state.read();
    }
    this.writeFile = function() {
        state.write();
    }
    this.deleteFile = function() {
        state.delete();
    }
}

// 测试
var user = new User(new NomalUser());
user.readFile();
user.writeFile();

user.setState(new SuperAdmin());
user.readFile();
user.writeFile();
user.deleteFile();

режим заметки

краткий:
Три персонажа, составитель, памятка, хранитель памятки. В оригинаторе выставлены два интерфейса, один используется для обертывания собственного состояния в меморандум, а другой используется для восстановления собственного состояния через меморандум-хранитель.В меморандуме сохраняется только состояние оригинатора, а меморандум хранится в меморандуме опекуна. , доступен для чтения и записи.

Пример кода:

// 备忘录模式
// 设置另外的对象作为备忘录对象,保存对象的状态

// 场景:英雄联盟购买装备撤回

/** 装备备忘录构造函数
 * 
 */
function EquipmentsMemento(equipments) {
    this.setEquipments = function(newEquipments) {
        equipments = newEquipments;
    }
    this.getEquipments = function() {
        return equipments;
    }
}

/** 装备栏构造函数 
 *  
 */
function EquipmentBar() {
    var equipments = [];
    this.buyEquipment = function(equipment) {
        console.log(`购买了装备:${equipment}`);
        equipments.push(equipment);
    }
    this.showEquipments = function() {
        console.log("已购买装备: ", equipments.join(" "));
    }
    this.getEquipmentsMemento = function() {
        return new EquipmentsMemento([...equipments]);
    }
    this.recoverEquipments = function(equipmentCaretaker) {
        equipments = equipmentCaretaker.getEquipmentsMemento().getEquipments();
    }
}

/** 装备状态管理者构造函数
 * 
 */
function EquipmentCaretaker() {
    var equipmentsMemento = null;
    this.setEquipmentsMemento = function(newEquipmentsMemento) {
        equipmentsMemento = newEquipmentsMemento;
    }
    this.getEquipmentsMemento = function() {
        return equipmentsMemento;
    }
}


// 测试

// 初始化状态看守者
var equipmentCaretaker = new EquipmentCaretaker();

// 初始化装备栏
var equipmentBar = new EquipmentBar();
// 购买装备
equipmentBar.buyEquipment("无尽之刃");
equipmentBar.buyEquipment("狂战士胫甲");

// 保存当前
equipmentCaretaker.setEquipmentsMemento(equipmentBar.getEquipmentsMemento());

// 购买了一件不想要的装备
equipmentBar.buyEquipment("无用大棒");

equipmentBar.showEquipments();

// 撤回
console.log("玩家买错了,撤回...");

equipmentBar.recoverEquipments(equipmentCaretaker);
equipmentBar.showEquipments();

режим парсера

краткий:
Для данного языка определите представление его грамматики и определите интерпретатор, который использует это представление для интерпретации предложений на языке.

Пример кода:

// 解释器模式
// 给定一门语言,定义它的文法的一种表示,并定义一个解释器,该解释器使用该表示来解释语言中的句子。

// 场景:sql 解释器  

/** sql 内容构造函数
 *  
 */
function Context() {
    var tableName = null/** string */;
    var params = null/** object */;
    var wheres = null/** object */;
    var fields = null/** set */;

    this.setTableName = function(newTableName) {
        tableName = newTableName;
    }
    this.getTableName = function() {
        return tableName;
    }

    this.setParams = function(newParams) {
        params = newParams;
    }
    this.getParams = function() {
        return params;
    }

    this.setWheres = function(newWheres) {
        wheres = newWheres;
    }
    this.getWheres = function() {
        return wheres;
    }

    this.setFields = function(newFields) {
        fields = newFields;
    }
    this.getFields = function() {
        return fields;
    }
}

/** 解释器接口 SQLExpression
 *  string interpret(Context context)
 */

/** insert sql 解释器
 * 
 */
function InsertSQLExpression() {
    this.interpret = function(context) {

        var params = context.getParams();

        // 拼接 key
        var keys = "(";
        var allKey = Object.getOwnPropertyNames(params);
        allKey.forEach((key) => {
            keys += key + ",";
        });
        keys = keys.substring(0, keys.length - 1);
        keys += ")";

        // 拼接 value
        var values = "(";
        allKey.forEach((key) => {
            values += (typeof params[key] === 'string' ? `'${params[key]}'` : params[key]) + ",";
        });
        values = values.substring(0, values.length - 1);
        values += ")";
        
        return `insert into ${context.getTableName()} ${keys} values ${values}`;
        
    }
}

// 测试
var insertSQLExpression = new InsertSQLExpression();

var context = new Context();
context.setTableName("student");
context.setParams({
    name: 'dcw',
    age: 22
});

var sql = insertSQLExpression.interpret(context);
console.log(sql)

шаблон посетителя

краткий:
видеть, как люди разговаривают с людьми, видеть, как призраки разговаривают с призраками

Пример кода:

// 访问者模式
// 见人说人话,见鬼说鬼话

// 场景:买衣服时服务员的引导,男生引导到男生区,女生引导到女生区


/** 服务员抽象类 Waiter
 *  accept(Customer customer)
 */

/** 以纯商场服务员 extends Waiter
 *  
 */
function YiChunWaiter() {
    this.accept = function(customer) {
        customer.visit(this);
    }
    // 服务女士
    this.serverWoman = function() {
        console.log("带领女士到女士服装区.");
    }
    // 服务男士
    this.serveMan = function() {
        console.log("带领男士到男士服装区.");
    }
}

/** 顾客接口(访问者)
 *  visit(MarketWaiter waiter)
 */

/** 女士顾客
 * 
 */
function WomanCustomer() {
    this.visit = function(waiter) {
        waiter.serverWoman();
    }
}

/** 男士顾客
 * 
 */
function ManCustomer() {
    this.visit = function(waiter) {
        waiter.serveMan();
    }
}

// 测试
var yichunWaiter = new YiChunWaiter();

var womanCustomer = new WomanCustomer();
var manCustomer = new ManCustomer();

yichunWaiter.accept(womanCustomer);
yichunWaiter.accept(manCustomer);