[js高手之路]设计模式系列课程-委托模式实战微博发布功能
在实际开发中,经常需要为Dom元素绑定事件,如果页面上有4个li元素,点击对应的li,弹出对应的li内容,怎么做呢?是不是很简单?
大多数人的做法都是:获取元素,绑定事件
<ul>
<li>跟着ghostwu学习javascript设计模式的应用1</li>
<li>跟着ghostwu学习javascript设计模式的应用2</li>
<li>跟着ghostwu学习javascript设计模式的应用3</li>
<li>跟着ghostwu学习javascript设计模式的应用4</li>
</ul>
<script>
var aLi = document.querySelectorAll( "li" );
aLi.forEach(( ele, ind ) => {
ele.addEventListener( "click", ()=> {
alert( ele.innerHTML );
} );
});
</script>
如果页面上有1w个元素, 甚至10w个元素呢?
继续使用上述方式,会有很大的性能问题,这个时候,有人可能要问,实际中的项目 哪有1w个,10w个元素的。一般的项目可能没有,但是社交类的网站,如微博,其他的如大批量文件上传等能功能,都是需要动态创建dom元素,而且数量巨大,并且创建出来的dom元素,一般都需要绑定事件和相应的特效。一般情况下,用普通的绑定事件方式是不能给动态创建的dom元素绑定到事件的,所以这里就产生了两个问题:
1,当页面元素很多的时候,如果给这些元素绑定上事件?
2,当动态创建dom时,如果给动态创建的dom绑定上事件和相应的特效?
这就是本文需要讨论的模式-委托模式
采用事件委托可以顺利解决上面2个问题
那么,什么是事件委托呢?
给元素的父元素绑定事件,利用冒泡原理,当子元素触发事件的时候,会去触发父元素的事件,然后把相应的业务逻辑放在父元素的事件中去处理。通俗点讲就是,子元素不做事件绑定,把绑定事件的操作委托给父元素,这就叫做事件委托, 委托有一个特性,他能够在事件触发中,识别到具体是由哪个子元素触发的,这个就是事件对象的target属性
<ul>
<li>跟着ghostwu学习javascript设计模式的应用1</li>
<li>跟着ghostwu学习javascript设计模式的应用2</li>
<li>跟着ghostwu学习javascript设计模式的应用3</li>
<li>跟着ghostwu学习javascript设计模式的应用4</li>
</ul>
<script>
var oUl = document.querySelector("ul");
oUl.addEventListener( "click", ( ev )=>{
var oEvent = ev || event;
target = oEvent.target || oEvent.srcElement;
alert( target.innerHTML );
});
</script>
当我们点击li的时候,就能通过事件对象oEvent.target识别到触发事件的li元素, srcElement是兼容ie的写法。
在没有事件委托之间,我们通过javascript做一个hover的功能,一般这么做.
<ul>
<li>跟着ghostwu学习设计模式</li>
<li>跟着ghostwu学习设计模式</li>
<li>跟着ghostwu学习设计模式</li>
<li>跟着ghostwu学习设计模式</li>
<li>跟着ghostwu学习设计模式</li>
</ul>
<script>
var aLi = document.getElementsByTagName( "li" );
for( var i = 0, len = aLi.length; i < len; i++ ){
aLi[i].onmouseover = function(){
this.style.backgroundColor = 'red';
}
aLi[i].onmouseout = function(){
this.style.backgroundColor = '';
}
}
</script>
如果li元素很多,就会产生性能问题,而采用委托模式,我们可以这么做
<ul>
<li>跟着ghostwu学习设计模式1</li>
<li>跟着ghostwu学习设计模式2</li>
<li>跟着ghostwu学习设计模式3</li>
</ul>
<script>
var aLi = document.getElementsByTagName("li");
var oUl = document.getElementsByTagName( "ul" )[0];
oUl.onmouseover = function( ev ){
var oEvent = ev || event;
var target = oEvent.target || oEvent.srcElement;
if ( target.tagName.toLowerCase() == 'li' ) {
target.style.backgroundColor = 'red';
}
}
oUl.onmouseout = function( ev ){
var oEvent = ev || event;
var target = oEvent.target || oEvent.srcElement;
if ( target.tagName.toLowerCase() == 'li' ) {
target.style.backgroundColor = '';
}
}
</script>
通过事件委托,把元素绑定到父元素,大大提高性能
至此,我们解决了第一个问题:当页面元素很多的时候,如果给这些元素绑定上事件
对于新创建的dom元素,普通绑定事件的方式,是不能绑定到这些dom元素的
<input type="button" value="创建">
<ul>
<li>ghostwu1</li>
<li>ghostwu2</li>
</ul>
<script>
var oBtn = document.getElementsByTagName( "input" )[0];
var oUl = document.getElementsByTagName( "ul" )[0];
var aLi = document.getElementsByTagName( "li" );
oBtn.onclick = function(){
var oLi = document.createElement( "li" );
oLi.innerHTML = 'ghostwu';
oUl.appendChild( oLi );
}
for( var i = 0, len = aLi.length; i < len; i++ ){
aLi[i].onmouseover = function(){
this.style.backgroundColor = 'red';
}
aLi[i].onmouseout = function(){
this.style.backgroundColor = '';
}
} </script>
新创建的li元素,是不能绑定到onmouseover和onmouseout事件的,我们采用委托模式之后,可以这么做
<input type="button" value="创建">
<ul>
<li>ghostwu1</li>
<li>ghostwu2</li>
</ul>
<script>
var oBtn = document.getElementsByTagName("input")[0];
var oUl = document.getElementsByTagName("ul")[0];
var aLi = document.getElementsByTagName("li");
oBtn.onclick = function () {
var oLi = document.createElement("li");
oLi.innerHTML = 'ghostwu';
oUl.appendChild(oLi);
}
oUl.onmouseover = function( ev ){
var oEvent = ev || event;
var target = oEvent.target || oEvent.srcElement;
if ( target.tagName.toLowerCase() == 'li' ) {
target.style.backgroundColor = 'red';
}
}
oUl.onmouseout = function( ev ){
var oEvent = ev || event;
var target = oEvent.target || oEvent.srcElement;
if ( target.tagName.toLowerCase() == 'li' ) {
target.style.backgroundColor = '';
}
}
</script>
至此,我们解决了第二个问题:当动态创建dom时,如果给动态创建的dom绑定上事件和相应的特效
最后,我们结合委托模式,来实战一个微博发布的功能,微博发布之后,给动态创建的dom元素添加手风琴折叠功能
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<meta http-equiv="X-UA-Compatible" content="ie=edge">
<title>Document</title>
<link href="https://cdn.bootcss.com/bootstrap/3.3.4/css/bootstrap.min.css" rel="stylesheet">
</head>
<body>
<div class="container">
<div class="row">
<div class="col-md-offset-3 col-md-6">
<form class="form-horizontal">
<div class="form-group">
<input id="title" type="text" class="form-control" placeholder="请输入标题">
</div>
<div class="form-group">
<textarea name="" id="txt" cols="30" rows="5" class="form-control"></textarea>
</div>
<div class="form-group">
<input type="button" value="发布" class="btn btn-primary" id="btn-publish">
</div>
</form>
</div>
<div class="col-md-offset-3 col-md-6" id="content">
</div>
</div>
</div>
</body>
<script>
var oBtnPublish = document.getElementById("btn-publish");
var aTpl = [
'<div class="panel panel-success">',
'<div class="panel-heading">',
'<h4 class="panel-title">',
'<a href="javascript:;">',
'[title]',
'</a>',
'</h4>',
'</div>',
'<div class="panel-body">',
'[content]',
'</div>',
'</div>'
];
var oContent = document.getElementById( "content" ),
str = aTpl.join( "" ),
title = '', content = '', panelParent = null;
oBtnPublish.onclick = function(){
str = aTpl.join( "" );
title = document.getElementById( "title" ).value;
txt = document.getElementById( "txt" ).value;
str = str.replace( /\[title\]/, title );
str = str.replace( /\[content\]/, txt );
oContent.innerHTML += str;
}
oContent.onclick = function( ev ){
var oEvent = ev || event;
var target = oEvent.target || oEvent.srcElement;
if ( target.tagName.toLowerCase() == 'a' ) {
panelParent = target.parentNode.parentNode.parentNode;
if ( panelParent.children[1].style.display == "block" || panelParent.children[1].style.display == '' ) {
panelParent.children[1].style.display = 'none';
} else {
panelParent.children[1].style.display = 'block';
}
}
}
</script>
</html>
[js高手之路]设计模式系列课程-委托模式实战微博发布功能的更多相关文章
- [js高手之路]设计模式系列课程-组合模式+寄生组合继承实战新闻列表
所谓组合模式,就是把一堆结构分解出来,组成在一起,现实中很多这样的例子,如: 1.肯德基套餐就是一种组合模式, 比如鸡腿堡套餐,一般是是由一个鸡腿堡,一包薯条,一杯可乐等组成的 2.组装的台式机同理, ...
- [js高手之路] 设计模式系列课程 - jQuery的extend插件机制
这里在之前的文章[js高手之路] 设计模式系列课程 - jQuery的链式调用与灵活的构造函数基础上增加一个extend浅拷贝,可以为对象方便的扩展属性和方法, jquery的插件扩展机制,大致就是这 ...
- [js高手之路]设计模式系列课程-发布者,订阅者重构购物车
发布者订阅者模式,是一种很常见的模式,比如: 一.买卖房子 生活中的买房,卖房,中介就构成了一个发布订阅者模式,买房的人,一般需要的是房源,价格,使用面积等信息,他充当了订阅者的角色 中介拿到卖主的房 ...
- [js高手之路] 设计模式系列课程 - jQuery的链式调用与灵活的构造函数
一.我们从一个简单的构造函数+原型程序开始 var G = function(){}; G.prototype = { length : 5, size : function(){ return th ...
- [js高手之路]设计模式系列课程-单例模式实现模态框
什么是单例呢? 单,就是一个的意思.例:就是实例化出来的对象,那合在一起就是保证一个构造函数只能new出一个实例,为什么要学习单例模式呢?或者说单例模式有哪些常见的应用场景.它的使用还是很广泛,比如: ...
- [js高手之路]设计模式系列课程-设计一个模块化扩展功能(define)和使用(use)库
模块化的诞生标志着javascript开发进入工业时代,近几年随着es6, require js( sea js ), node js崛起,特别是es6和node js自带模块加载功能,给大型程序开发 ...
- [js高手之路] 设计模式系列课程 - DOM迭代器(2)
如果你对jquery比较熟悉的话,应该用过 eq, first, last, get, prev, next, siblings等过滤器和方法.本文,我们就用迭代设计模式来封装实现,类似的功能 < ...
- [js高手之路] 设计模式系列课程 - 迭代器(1)
迭代器是指通过一种形式依次遍历数组,对象,或者类数组结构中的每个元素. 常见的有jquery中的each方法, ES5自带的forEach方法. 下面我们就来自定义一个类似jquery或者ES5的迭代 ...
- [js高手之路] es6系列教程 - 迭代器与生成器详解
什么是迭代器? 迭代器是一种特殊对象,这种对象具有以下特点: 1,所有对象都有一个next方法 2,每次调用next方法,都会返回一个对象,该对象包含两个属性,一个是value, 表示下一个将要返回的 ...
随机推荐
- 文本三剑客---gawk基础
gawk程序是Unix中原始awk程序的GNU版本.gawk程序让流编辑器迈上了一个新的台阶,它提供了一种编程语言而不只是编辑器命令.在gawk编程语言中,可以完成下面的事情: (1)定义变量来保存数 ...
- Nginx文档-初学者指南
原文档: http://nginx.org/en/docs/beginners_guide.html 译者:Oopsguy 本指南旨在介绍nginx基本内容和一些在Nginx上可以完成的简单任务.这里 ...
- API和schema开发过程问题汇总
场景:在日常工作中很多都是重复性的劳动,有的坑踩过一次下次很可能还回踩到,所以很有必要将这些问题记录下来,提升工作效率! 1 API 2 schema 问题:提示schame中有元素没有定义 这个时候 ...
- (转)mysql水平分表和垂直分表和数据库分区
坚信数据库的物理设计在对高级数据库的性能影响上远比其他因素重要.给大家说一下经过专家对Oracle的研究,他们解释了为什么拙劣的物理设计是数据库停机(无论是有计划的还是没计划的)背后的主要原因.但在这 ...
- js 带遮罩层的弹出层
最近有个小伙伴问我关于mui的带遮罩层的弹出层,我给他发了个相关demo,但是因为他是新手的原因没有看懂,所以我写了一个单纯的原生的带遮罩层的弹出层dome.写这篇微博希望可以帮到需要的小伙伴. &l ...
- vue使用之剑走偏锋——菜单组件篇
背景 习惯了这样的写法了,上来先来一个背景,交待一下事情的起因.事情的起因很简单,用惯了mvc框架,想来一发前后端完全分离的框架试试.选用的人员和技术是这样子的,让原本做java的web开发的人员使用 ...
- Android学习笔记-ImageView(图像视图)
本节引言: 本节介绍的UI基础控件是:ImageView(图像视图),见名知意,就是用来显示图像的一个View或者说控件! 官方API:ImageView;本节讲解的内容如下: ImageView的s ...
- Java工程师书单(初级、中级、高级)
简介 怎样学习才能从一名Java初级程序员成长为一名合格的架构师,或者说一名合格的架构师应该有怎样的技术知识体系,这是不仅一个刚刚踏入职场的初级程序员也是工作一两年之后开始迷茫的程序员经常会问到的问题 ...
- bzoj 3594: [Scoi2014]方伯伯的玉米田
3594: [Scoi2014]方伯伯的玉米田 Time Limit: 60 Sec Memory Limit: 128 MB Submit: 1399 Solved: 627 [Submit][ ...
- ABP+AdminLTE+Bootstrap Table权限管理系统第五节--WBEAPI及SwaggerUI
一,Web API ABP的动态WebApi实现了直接对服务层的调用(其实病没有跨过ApiController,只是将ApiController公共化,对于这一点的处理类似于MVC,对服务端的 调用没 ...