行为型设计模式(二)

  • 命令模式(Command)
    • 核心
    • 示例一:自由化创建视图
    • 示例二:绘图命令
  • 访问者模式(Visitor)
    • 核心
    • 示例一:IE 兼容
    • 示例二:操作类数组对象
  • 中介者(Mediator)
    • 核心
    • 示例一:导航设置功能
  • 备忘录模式(Memento)
    • 核心
    • 示例一:网络请求数据的缓存
  • 迭代器模式(Iterator)
    • 核心
    • 示例一:简化循环遍历
    • 示例二:数组/对象迭代器
    • 示例三:同步变量迭代器
    • 示例四:分支循环嵌套问题
  • 解释器模式(Iterpreter)
    • 核心
    • 示例一:统计元素路径
  • 用于不同 对象间 职责划分或算法抽象
  • 代码:https://github.com/baixc1/csdn/tree/master/DesignPatterns/Behavior

命令模式(Command)

核心

  • 定义:将请求与实现解耦,封装成独立对象。使不同的请求对客户端的实现 参数化。
  • 重点:简化使用方法,以参数配置的形式,实现复杂业务。重点是封装底层实现部分。
  • 应用:组件,库,框架,编程中大量使用。调用和实现的解耦,上层调用者不用关心具体实现,加快业务开发
  • 举例:以参数配置的形式,动态生成UI
viewCommand({cmd: "display",params: ["titleId", titleData, "title"], // display 参数
})({cmd: "create",params: [productData, "product"], // create参数,创建3个列表项
})({cmd: "display",params: [// 创建1个列表项,并展示"productId",{src: "4.jpg",text: "text4",},"product",],
});

示例一:自由化创建视图

  • 需求:自由创建视图(标题,单张图片,多张图片。。。)
  • js
// 模块实现模块
var viewCommand = (function () {// 模板(变量:{#var#})var tpl = {product: `<div><img src="{#src#}" /><p>{#text#}</p></div>`,title: `<div class="title"><div class="main"><h2>{#title#}</h2><p>{#tips#}</p></div></div>`,};// 方法集合(通过key调用)var Action = {create(data, view) {// 数组if (data.length) {for (let v of data) {html += formatString(tpl[view], v);}} else {html += formatString(tpl[view], data);}return this; // 链式调用},display(container, data, view) {if (data) {this.create(data, view);}document.getElementById(container).innerHTML = html;html = "";return this; // 链式调用},};// 当前的 格式化字符串(可用作列表缓存数据拼接)var html = "";/*** 命令接口* cmd: Action 命令* params: 模版及其变量值,[data, view]*/return function excute({ cmd, params }) {// 调用时,this绑定为Action(apply第二个参数为数组)Action[cmd].apply(Action, Array.isArray(params) ? params : [params]);return excute;};/*** 生成模版字符串* @function formatString* @param {string} str - 模版字符串* @param {object} obj - 变量对象* @return {string}  - 替换变量后的模版字符串*/function formatString(str, obj) {return str.replace(/\{#(\w+)#\}/g, function (match, key) {return obj[key];});}
})();var productData = [{src: "1.jpg",text: "text1",},{src: "2.jpg",text: "text2",},{src: "3.jpg",text: "text3",},
];var titleData = {title: "夏日里的一片温馨",tips: "暖暖的温情带给人们家的感受",
};viewCommand({cmd: "display",params: ["titleId", titleData, "title"], // display 参数
})({cmd: "create",params: [productData, "product"], // create参数,创建3个列表项
})({cmd: "display",params: [// 创建1个列表项,并展示"productId",{src: "4.jpg",text: "text4",},"product",],
});
  • 引用
<!DOCTYPE html>
<html lang="en"><head><meta charset="UTF-8" /><meta http-equiv="X-UA-Compatible" content="IE=edge" /><meta name="viewport" content="width=device-width, initial-scale=1.0" /><title>Document</title><style>#productId {display: flex;}#productId > div {margin-right: 20px;}</style></head><body><div id="titleId"></div>------------------------<div id="productId"></div><script src="./index.js"></script></body>
</html>
  • 效果

示例二:绘图命令

  • 需求:使用canvas绘图时,将上下文对象封装在命令对象内部(可统一解决兼容性问题)
  • js
// index2.js
// 绘图命令
// 功能:用于解耦,封装上下文对象function getType(v) {return Object.prototype.toString.call(v);
}
// 绘图对象
var CanvasCmd = function (id) {var canvas = document.getElementById(id);var ctx = canvas.getContext("2d");return {excute(msg) {if (!msg) return;// 处理命令数组if (msg.length) {for (let v of msg) {arguments.callee(v);}}// 处理每条命令else {let { param, cmd } = msg;// canvas绘图 没有该 apiif (ctx[cmd] === undefined) return;// 命令是函数if (typeof ctx[cmd] === "function") {ctx[cmd](...param);} else {ctx[cmd] = param;}}},};
};CanvasCmd("canvas").excute([{cmd: "fillStyle",param: "red",},{cmd: "fillRect",param: [20, 20, 100, 100],},
]);
  • 调用
<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><meta http-equiv="X-UA-Compatible" content="IE=edge"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>Document</title>
</head>
<body><canvas id="canvas" width="300" height="300">抱歉,您的浏览器不支持canvas元素(这些内容将会在不支持<canvas>元素的浏览器或是禁用了JavaScript的浏览器内渲染并展现)</canvas><script src="./index2.js"></script>
</body>
</html>
  • 效果

访问者模式(Visitor)

核心

  • 定义:为对象提供统一的、兼容的新的访问方法
  • 重点:二次封装对象的访问方法
  • 应用:IE 相关 api 的兼容,操作类数组对象

示例一:IE 兼容

// 问题:兼容访问api
// 在IE 9之前,必须使用 attachEvent
document.getElementById("id").attachEvent("onclick", function () {// 使用 attachEvent 方法有个缺点,this 的值会变成 window 对象的引用而不是触发事件的元素。console.log(this);
});// 显示:使用call绑定this,并传入自定义数据
function bindIEEvent(dom, type, fn, data = {}) {dom.attachEvent("on" + type, function (e) {fn.call(dom, e, data);});
}

示例二:操作类数组对象

  • 对象原型添加数组方法
// index3.js
// 扩展对象原型:新增数组方法(类数组对象)
// arguments除了length属性和索引元素之外没有任何Array属性
Object.prototype = Object.assign(Object.prototype, {splice() {// const params = Array.prototype.slice.call(arguments)// const params = [...arguments]// const params = Array.from(arguments);// this指向对象实例,不需转化arguments为数组,自动添加length属性return Array.prototype.splice.apply(this, arguments);},push() {// this.length += (this.length || 0) + arguments.length;return Array.prototype.push.apply(this, arguments);},pop() {return Array.prototype.pop.apply(this);},
});const obj = {};
console.log(obj); // {}
console.log(obj.push(1, 2, 3, 4)); // 4
console.log(obj); // { '0': 1, '1': 2, '2': 3, '3': 4, length: 4 }
console.log(obj.splice(1, 1, 6, 7, 8)); // [ 2 ]
console.log(obj); // { '0': 1, '1': 6, '2': 7, '3': 8, '4': 3, '5': 4, length: 6 }
console.log(obj.pop()); // 4
console.log(obj); // { '0': 1, '1': 6, '2': 7, '3': 8, '4': 3, length: 5 }
console.log(Object.prototype);

*其他测试

// index2.js
// 原生对象构造器:访问者(内部访问了this)
const toString = Object.prototype.toString;
console.log(toString.apply({})); // [object Object]
console.log(toString.call(1)); // [object Number]
console.log("------------");
// 其他测试
console.log({}.__proto__.toString === toString); // true
console.log("toString" in {}); // true
console.log({}.hasOwnProperty("toString")); // false

中介者(Mediator)

核心

  • 定义:通过中介者对象(消息容器),封装对象间的交互,减少对象间的耦合
  • 重点:多个模块的统一设置功能
  • 应用:页面设置功能(供用户操作页面元素的显示/隐藏。。。)(解决模块间通信问题)

示例一:导航设置功能

  • 需求:通过导航设置层,统一控制页面中多个导航的元素显示方法
  • 导航组件封装
// index2.js
// 格式化字符串
function formatString(str, data) {// 全局匹配 {#xxx#} -> xxx(数字,字母,下划线)return str.replace(/\{#(\w+)#\}/g, function (match, key) {return (data && data[key]) || "";});
}// 基础导航
var Nav = function (data) {// 模板this.item = '<a href="{#href#}" title="{#title#}">{#name#}</a>';this.html = "";for (const v of data) {this.html += formatString(this.item, v);}return this.html;
};// 信息导航
var NumNav = function (data) {// 模板var tpl = "<b>{#num#}</b>";for (let i = 0; i < data.length; i++) {data[i].name += formatString(tpl, data[i]);}return Nav.call(this, data);
};// 网址导航
var LinkNav = function (data) {var tpl = "<span>{#link#}</span>";for (let i = 0; i < data.length; i++) {data[i].name += formatString(tpl, data[i]);}return Nav.call(this, data);
};// 网址/消息导航
var LinkNumNav = function (data) {var tpl = "<span>{#link#}</span>";for (let i = 0; i < data.length; i++) {data[i].name += formatString(tpl, data[i]);}return NumNav.call(this, data);
};
  • 中介者对象(消息容器)
// 中介者对象(消息系统)
var Mediator = (function () {// 消息对象var msg = {};return {/*** 注册消息方法* @param {string} type* @param {function} action*/register(type, action) {if (!msg[type]) {msg[type] = [];}msg[type].push(action);return this;},// 发布消息emit(type) {if (!msg[type]) return;for (let fn of msg[type]) {fn && fn();}},};
})();/*** 显隐导航组件* @param {dom} ele* @param {string} tag 标签* @param {boolean} isShow*/
var showHideNav = function (ele, tag, isShow) {var subEle = ele.getElementsByTagName(tag);var display = isShow ? "initial" : "none";for (let v of subEle) {v.style.display = display;}
};// Mediator.register("demo", () => console.log("first"));
// Mediator.register("demo", () => console.log("second"));// Mediator.emit("demo");

*页面调用

<!DOCTYPE html>
<html lang="en"><head><meta charset="UTF-8" /><meta http-equiv="X-UA-Compatible" content="IE=edge" /><meta name="viewport" content="width=device-width, initial-scale=1.0" /><title>Document</title><style>a {margin-right: 6px;display: block;}.mu {margin-bottom: 50px;}</style>
</head><body><div>用户收藏导航模块(网址+消息)</div><div class="mu"></div><div>推荐用户导航模块(消息)</div><div class="mu"></div><div>最近常用导航模块(网址)</div><div class="mu"></div><div>用户收藏导航模块(网址+消息)</div><div class="mu"></div><input type="checkbox" id="hide_num">隐藏消息</input><input type="checkbox" id="hide_url">隐藏链接</input><script src="./index.js"></script><script src="./index2.js"></script><script>// 导航数据const list = [{href: "http://www.baidu.com",title: "面板",name: "tab1",type: 'LinkNumNav'},{href: "http://www.baidu.com",title: "面板",name: "tab2",type: 'NumNav'},{href: "http://www.baidu.com",title: "面板",name: "tab3",type: 'LinkNav'},{href: "http://www.baidu.com",title: "面板",name: "tab4",type: 'LinkNumNav'},];// 导航模块(主体列表)(() => {[...document.getElementsByClassName('mu')].forEach((item, index) => {const { type } = list[index]const hasNum = ['LinkNumNav', 'NumNav'].includes(type)const hasLink = ['LinkNumNav', 'LinkNav'].includes(type)// 导航元素显示item.innerHTML = window[type](list.map((item, index) => {const obj = { ...item }// 5-10随机数const r1 = Math.floor(Math.random() * 5) + 5// 97 - 97+26(ascii字母)let path = ''for (let i = 0; i < r1; i++) {path += String.fromCharCode(Math.floor(Math.random() * 26) + 97)console.log(path)}if (hasNum) {obj.num = `(消息:${r1})`}if (hasLink) {obj.link = `(网址:http://xiaoxi.www/${path}.com`}return obj}))// 注册消息事件if (hasNum) {Mediator.register('hideAllNavNum', function () {showHideNav(item, 'b', false)}).register('showAllNavNum', function () {showHideNav(item, 'b', true)})}// 注册链接事件if (hasLink) {Mediator.register('hideAllNavUrl', function () {showHideNav(item, 'span', false)}).register('showAllNavUrl', function () {showHideNav(item, 'span', true)})}});})();// 设置层模块(底部多选框)(() => {var hideNum = document.getElementById("hide_num");var hideUrl = document.getElementById("hide_url");[hideNum, hideUrl].forEach((item, index) => {item.onchange = function () {const str = index === 0 ? "Num" : "Url";if (item.checked) {Mediator.emit("hideAllNav" + str);} else {Mediator.emit("showAllNav" + str);}};});})();</script>
</body></html>
  • 效果

备忘录模式(Memento)

核心

  • 定义:在不破坏对象封装性的前提下,在对象外缓存对象的状态,以便重复使用
  • 应用:网络请求数据设置有效期缓存(不同场景的缓存模式),MVC中 Model 层部分

示例一:网络请求数据的缓存

// 缓存数据(备忘录模式)
var Page = (function () {// 缓存对象var cache = {};// 获取数据(缓存或网络请求)return function (page) {if (cache[page]) {return Promise.resolve(cache[page]);} else {return new Promise((resolve, reject) => {// 模拟ajax请求setTimeout(() => {const res = { list: [], total: 0, page };cache[page] = res;resolve(res);}, 1000);});}};
})();// 测试代码
(async () => {let data = await Page(1);console.log(data); // 1s后获取数据(模拟网络请求)data = await Page(1);console.log(data); // 立即获取数据(缓存)data = await Page(2);console.log(data); // 1s后获取数据(模拟网络请求)
})();

迭代器模式(Iterator)

核心

  • 定义:在不暴露对象内部结构的同时,提供顺序访问(聚合对象内部)元素的方法
  • 重点:优化大量重复循环导致的代码臃肿问题
  • 应用:重复循环逻辑的封装提取,定制化的对象/数组迭代器

示例一:简化循环遍历

  • 需求:不同种类的焦点图(轮播图),封装其访问/操作元素的部分(通过迭代器控制元素及其访问)
  • js
// Iterator.js
/*** 迭代器(简化遍历/访问操作)* @param {string} containerId - 容器元素id* @param {string} subTag - 容器子元素 tag* @returns*/
var Iterator = function (containerId, subTag) {const container = document.getElementById(containerId);const items = container.getElementsByTagName(subTag);const len = items.length;let index = 0; // 当前访问的元素的索引return {// 获取第一个元素first() {index = 0;return items[index];},// 获取最后一个元素last() {index = len - 1;return items[index];},// 上一个(负数取第一个)pre() {if (--index > 0) {return items[index];} else {index = 0;return null;}},// 下一个next() {if (++index < length) {return items[index];} else {index = length - 1;return null;}},// 获取第 n 个元素(负数和超过len时,转化为0 -> len-1)get(num) {index = num > 0 ? num % length : (num % length) + length;},// 处理每个元素, 回调函数 + 参数dealEach(fn, ...params) {[...items].forEach((item) => {fn.apply(item, params);});},// 处理某个元素,元素下标 + 回调函数 + 参数dealItem(num, fn, ...params) {fn.apply(items[num], params);},// 排他方式处理元素,元素下标(number/array) + 处理全部元素的回调 + 处理num元素的回调exclusive(num, allFn, numFn) {this.dealEach(allFn);if (Array.isArray(num)) {for (let v of num) {this.dealItem(v, numFn);}} else {this.dealItem(num, numFn);}},};
};

*html

<!--Iterator.html-->
<!DOCTYPE html>
<html lang="en"><head><meta charset="UTF-8"><meta http-equiv="X-UA-Compatible" content="IE=edge"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>Document</title><script src="./Iterator.js"></script>
</head><body><ul id="container"><li>1</li><li>2</li><li>3</li><li>4</li><li>5</li></ul><script>var demo = new Iterator('container', 'li')console.log(demo.first())console.log(demo.last())console.log(demo.pre())console.log(demo.get(-10))console.log(demo.next(-10))// 处理每个元素demo.dealEach(function (text, color) {this.innerText += textthis.style.color = color}, '-test', 'pink')// 排他方式处理元素demo.exclusive([1, 2], function () {this.innerText += '-被排除的'}, function () {this.innerText += '-被选中的'})</script>
</body></html>
  • 效果

示例二:数组/对象迭代器

  • 需求:数组相关原型方法(forEach, some)的实现
  • js
// ArrayAndObjectIterator.js
// 数组原型方法 forEach
// arr.forEach(callback(currentValue [, index [, array]])[, thisArg])Array.prototype.forEach = function (callback, thisArg) {var len = this.length;var i = 0;for (; i < len; i++) {callback.call(thisArg, this[i], i, this);}
};// arr.some(callback(element[, index[, array]])[, thisArg])
// 数组中有至少一个元素通过回调函数的测试就会返回true;所有元素都没有通过回调函数的测试返回值才会为false。
Array.prototype.some = function (callback, thisArg) {var len = this.length;var i = 0;for (; i < len; i++) {if (callback.call(thisArg, this[i], i, this)) {return true;}}return false;
};Object.prototype.forEach = function (callback, thisArg) {for (var key in this) {if (!this.hasOwnProperty(key)) continue; // 排除forEachcallback.call(thisArg, this[key], key, this);}
};console.log("forEach",[1, 2, 3].forEach((item, index, arr) => {console.log("forEachItem", item, index, arr);if (index === 1) return true;})
);
console.log("some",[1, 2, 3].some((item, index, arr) => {console.log("someItem", item, index, arr);if (index === 1) return true;})
);
console.log("obj-forEach",{ a: 1, b: 2 }.forEach((item, key, obj) => {console.log("obj-forEachItem", item, key, obj);})
);

示例三:同步变量迭代器

  • 需求:设置/访问对象属性,处理 undefined 的情况(对象的链式访问 - 可选链)
  • js
ViaIterator = {/*** 变量链式迭代取值器* @param {string} key - 链式可以, 例: 'a.b.c'* @param {object} obj* @returns*/get(obj, key) {let ret = obj; // 对象指针,指向访问的对象层级var keys = key.split("."); // key 的数组for (let v of keys) {if (!ret) return;if (ret[v] !== undefined) {ret = ret[v];}}return ret;},// 变量链式迭代赋值器set(obj, key, val) {let ret = obj; // 对象指针,指向访问的对象层级var keys = key.split("."); // key 的数组for (var i = 0, len = keys.length; i < len - 1; i++) {const v = keys[i];if (ret[v] === undefined) {ret[v] = {};}if (!(ret[v] instanceof Object)) {throw new Error(`obj.${keys.slice(0, i + 1).join(".")}不是对象`);}ret = ret[v];}return (ret[keys[i]] = val);},
};console.log(ViaIterator.get({ a: null }, "a.b.c")); // undefined
console.log(ViaIterator.get({ a: { b: { c: null } } }, "a.b.c")); // undefinedvar obj = {};
console.log(ViaIterator.set(obj, "a.b", { c: 1 })); // { c: 1 }
console.log(obj); // { a: { b: { c: 1 } } }
try {ViaIterator.set(obj, "a.b.c.val", "d");
} catch (e) {console.dir(e.message); // 'obj.a.b.c不是对象'
}

示例四:分支循环嵌套问题

  • 需求:canvas 处理图片像素数据,数据量大时,分支会影响性能
  • js
// canvas.js
// 分支循环嵌套问题// canvas 处理图片像素window.onload = function () {var canvas = document.getElementsByTagName("canvas")[0];var ctx = canvas.getContext("2d");var img = document.images[0];var width = (canvas.width = img.width * 2) / 2;var height = (canvas.height = img.height);// canvas左侧图片ctx.drawImage(img, 0, 0);// canvas置灰(蒙层1)dealImageYh("gray", 0, 0, width, height, 0);// canvas红色矩形(蒙层2)dealImageYh("red", 100, 100, 200, 200, 100);// canvas蓝色矩形(蒙层3)dealImageYh("blue", 120, 120, 160, 160, 255);/*** 绘制特效图片(未简化)* @param {string} t 类型* @param {*} x x坐标* @param {*} y* @param {*} w 宽* @param {*} h 高* @param {*} a 透明度*/function dealImage(t, x, y, w, h, a) {// 画布数据var canvasData = ctx.getImageData(x, y, w, h);// 像素数据var data = canvasData.data;// 遍历(rgba)for (var i = 0, len = data.length; i < len; i += 4) {data[i + 3] = a;switch (t) {case "red":data[i + 1] = data[i + 2] = 0;break;case "green":data[i] = data[i + 2] = 0;break;case "blue":data[i] = data[i + 1] = 0;break;case "gray":var num = parseInt((data[i] + data[i + 1] + data[i + 2]) / 3);data[i] = data[i + 1] = data[i + 2] = data[i + 3] = num;break;// ...}}ctx.putImageData(canvasData, width + x, y);}// 绘制特效图片(简化分支逻辑,优化性能)function dealImageYh(t, x, y, w, h, a) {var canvasData = ctx.getImageData(x, y, w, h);var data = canvasData.data;// 状态模式,简化分支var Deal = (function () {var methods = {default(i) {return methods.gray(i);},red(i) {data[i + 3] = a;data[i + 1] = data[i + 2] = 0;},green(i) {data[i + 3] = a;data[i] = data[i + 2] = 0;},blue(i) {data[i + 3] = a;data[i] = data[i + 1] = 0;},gray(i) {data[i + 3] = a;var num = parseInt((data[i] + data[i + 1] + data[i + 2]) / 3);data[i] = data[i + 1] = data[i + 2] = data[i + 3] = num;},};return function (type = "default") {return methods[type];};})();for (var i = 0, len = data.length; i < len; i += 4) {Deal(t)(i);}ctx.putImageData(canvasData, width + x, y);}
};

*html

<!--canvas.html-->
<!DOCTYPE html>
<html lang="en"><head><meta charset="UTF-8"><meta http-equiv="X-UA-Compatible" content="IE=edge"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>Document</title><script src="./canvas.js"></script>
</head><body><canvas></canvas><!--base64图片,解决getImageData跨域问题--><img style="visibility: hidden;"src="" />
</body></html>
  • 效果

解释器模式(Iterpreter)

核心

  • 定义:对于一种语言,给出其文法表示形式,并定义一种解释器,解释定义的语句
  • 应用:dom 元素路径统计

示例一:统计元素路径

  • 需求:点击某个元素,计算其相对某个父元素的路径
  • js
// index.js
// 获取兄弟元素名称
function getSublingName(node) {if (node.previousSibling) {var name = "",count = 1,nodeName = node.nodeName, // 节点名称sibling = node.previousSibling; // 返回当前节点的前一个兄弟节点while (sibling) {// 元素节点,类型相同,名称存在if (sibling.nodeType === 1 &&sibling.nodeType === node.nodeType &&sibling.nodeName) {if (nodeName === sibling.nodeName) {// 名称相同,后缀数字+1name = String(++count);} else {// 名称不同,后缀加 '|' 和 名称count = 1;name = "|" + sibling.nodeName.toUpperCase();}}sibling = sibling.previousSibling;}return name;} else {return "";}
}// xPath解释器(冒泡遍历节点树)
var Iterpreter = (function () {// 递归函数return function fn(node, wrap = document) {var path = [];// 终止条件一(目标节点等于容器节点)if (node === wrap) {if (wrap.nodeType === 1) {path.push(wrap.nodeName.toUpperCase());}return path;} else {// 当前节点的父节点 不等于 容器节点(递归操作)if (node.parentNode !== wrap) {path = fn(node.parentNode, wrap); // 递归,返回 path 数组}// 终止条件二(目标节点父节点等于容器节点)else {if (wrap.nodeType === 1) {path.push(wrap.nodeName.toUpperCase());}}// 统计当前节点元素信息if (node.nodeType === 1) {// 获取兄弟元素的统计var sublingsNames = getSublingName(node);path.push(node.nodeName.toUpperCase() + sublingsNames);}return path;}};
})();

*html

<!DOCTYPE html>
<html lang="en"><head><meta charset="UTF-8"><meta http-equiv="X-UA-Compatible" content="IE=edge"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>Document</title>
</head><body><div></div><div></div><div><div><ul><li><span id="s1">1</span></li><li><span id="s2">2</span></li><li><span id="s3">3</span></li></ul></div><div><ul id="u2"><li><span id="s11">111<b>bbbbb</b></span></li><li><span id="s22">22</span></li><li id="l2"><span id="s33">33</span></li></ul></div></div><script src="./index.js"></script><script>const s33 = document.getElementById('s33')const l2 = document.getElementById('l2')const u2 = document.getElementById('u2')console.log(Iterpreter(s33).join('->')) //HTML->BODY|HEAD->DIV3->DIV2->UL->LI3->SPANconsole.log(Iterpreter(s33, l2).join('->')) //LI->SPANconsole.log(Iterpreter(l2).join('->')) // HTML->BODY|HEAD->DIV3->DIV2->UL->LI3u2.onclick = function (e) {console.log(Iterpreter(e.target, this).join('->')) // 打印点击元素相对u2的路径}</script>
</body></html>

行为型设计模式(二)相关推荐

  1. 中介者模式 调停者 Mediator 行为型 设计模式(二十一)

    中介者模式(Mediator) 调度.调停 意图 用一个中介对象(中介者)来封装一系列的对象交互,中介者使各对象不需要显式地相互引用,从而使其耦合松散 而且可以独立地改变它们之间的交互. 中介者模式又 ...

  2. 模板方法模式 Template method 行为型 设计模式(二十六)

    模板方法模式 Template method 上图为网上百度的一份简历模板截图 相信大家都有求职的经历,那么必然需要简历,写简历的时候,很可能你会网上检索一份简历模板,使用此模板的格式,然后替换为你的 ...

  3. 观察者模式 Observer 发布订阅模式 源 监听 行为型 设计模式(二十三)

    观察者模式 Observer 意图 定义对象一种一对多的依赖关系,当一个对象的状态发生改变时,所有依赖他的对象都得到通知并自动更新. 别名:依赖(Dependents),发布订阅(Publish-Su ...

  4. Java设计模式(二)创建型设计模式

    文章目录 三 创建型设计模式 3.1 单例设计模式 3.1.1 饿汉式(线程安全) 3.1.2 懒汉式(线程不安全) 3.1.3 优缺点 3.1.4 补充 3.1.5 框架中的使用 3.1.4.1 S ...

  5. 二、java设计模式之工厂方法+抽象工厂模式(创建型设计模式)

    创建型设计模式-工厂模式和应用 工厂模式介绍: 它提供了一种创建对象的最佳方式,我们在创建对象时不会对客户端暴露创建逻辑,并且是通过使用一个共同的接口来指向新创建的对象 例子: 需要购买一辆车,不用管 ...

  6. 备忘录模式 Memento 快照模式 标记Token模式 行为型 设计模式(二十二)

    备忘录模式 Memento 沿着脚印,走过你来时的路,回到原点. 苦海翻起爱恨 在世间难逃避命运 相亲竟不可接近 或我应该相信是缘份 一首<一生所爱>触动了多少人的心弦,一段五百年都没有结 ...

  7. 设计模式二:创建型-工厂模式

    创建型模式:工厂模式 文章目录 创建型模式:工厂模式 工厂模式 1.工厂模式:介绍 2.工厂模式:模拟场景 3.工厂模式:代码实现 4.工厂模式:总结 工厂模式 1.工厂模式:介绍 工厂模式 简单工厂 ...

  8. 《深入设计模式》笔记 -创建型模式二、工厂方法模式

    抽象工厂模式 亦称: Abstract Factory 意图 抽象工厂模式是一种创建型设计模式, 它能创建一系列相关的对象, 而无需指定其具体类. 问题 假设你正在开发一款家具商店模拟器. 你的代码中 ...

  9. 技术图文:02 创建型设计模式(下)

    创建型设计模式(下) 知识结构: 图1 知识结构 单例模式 – 确保对象的唯一性 Sunny 软件公司承接了一个服务器负载均衡软件的开发工作,该软件运行在一台负载均衡服务器上,可以将并发访问和数据流量 ...

  10. 技术图文:02 创建型设计模式(上)

    创建型设计模式(上) 知识结构: 图1 知识结构 简单工厂模式 Sunny 软件公司欲基于 C# 语言开发一套图表库,该图表库可以为应用系统提供各种不同外观的图表,如: 柱状图(histogram) ...

最新文章

  1. struct 类型指针技巧
  2. md3600i存储服务器连接 iscsi+multipath配置
  3. 前端学习(3157):react-hello-react之一个简单的helloworld
  4. 温昱:架构实践全景图
  5. 【debug】mount: unknown filesystem type ‘nfs’
  6. Functional Interface JDK1.8
  7. xUtils项目框架
  8. 【嵌入式Linux】嵌入式Linux应用开发基础知识之输入系统应用编程
  9. Qt5.4生成安装包过程
  10. 综合评价模型的缺点_【必备】目标检测中的评价指标有哪些?
  11. python读写将excel转换为xml_Python实现将Excel转换成xml的方法示例
  12. 被圈粉的微信小程序纯UI组件colorUi
  13. python英文词频统计代码_python词频统计_英文
  14. oracle job定时报错,Oracle定时任务Job笔记
  15. [转]如何在NIOS II中读写EPCS剩余空间
  16. 京东咚咚架构演讲读后感
  17. 上门洗车APP --- Android客户端开发 之 网络框架封装介绍(二)
  18. 190502 Expressing Belief
  19. NCCL+Ubuntu20.04安装
  20. iPhone4/4s 5.1.1版本越狱后无法连接iTunes,出现0xE8000012错误的解决方法

热门文章

  1. 数字音频信号--Dither
  2. Kinect2.0在win10平台上时断时续问题的解决办法
  3. 一元高次方程c语言实现,c语言实现一元二次方程求解
  4. sonar代码审查问题分析
  5. 统一身份管理项目最佳实践
  6. 留美学子安全手册,这个可以有
  7. 华为云SSL证书申请流程
  8. 加一 — Python
  9. 【大数据实战】苏宁大数据离线任务开发调度平台实践:设计与开发过程中的要点
  10. python笔记三之面向对象(继承,封装,多态)