- 每日分享
- 数组的操作
- 浏览器的拖拉事件
- addEventListener
- 利用Css的伪类来解决一些问题
- 鼠标的11个事件
- 关于JavaScript中两个原生API(contains, compareDocumentPosition)
- 浏览器的进度事件
- 使用document.scrollingElement控制窗体滚动高度
- 表单事件的种类
- Css中的scroll-behaviour 和js 中的scrollIntoView让页面滚动平滑
- 浏览器的触摸事件
- 浏览器的异步操作
- 局部作用域和函数提升
- 函数节流
- js中的apply,call区别和用法
- 函数防抖
- js中的定时器
- Promise对象
- vue中如何keep-alive进行组件缓存
- 如何正确使用构造函数
- 如何看待history和hash
- 原生js
- es6的基本知识
- 开发常识
- //题目描 // 移除数组 arr 中的所有值与 item 相等的元素,直接在给定arr 数组上进行操作,并将结果返回
<!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>
</head>
<body>
<script>
//题目描述
// 移除数组 arr 中的所有值与 item 相等的元素,直接在给定的 arr 数组上进行操作,并将结果返回
function remove(arr , item) {
return arr.filter((list) =>{
return list !== item
})
}
var arr = [1,2,3,4,5,6,6,6,6,6]
var result = remove(arr,6)
console.log(result)
</script>
</body>
</html>
(本节完!)
-
拖拉(drag)指的是,用户在某个对象上按下鼠标键不放,拖动它到另一个位置,然后释放鼠标键,将该对象放在那里。 拖拉的对象有好几种,包括元素节点、图片、链接、选中的文字等等。在网页中,除了元素节点默认不可以拖拉,其他(图片、链接、选中的文字)都是可以直接拖拉的。为了让元素节点可拖拉,可以将该节点的draggable属性设为true。
-
当元素节点或选中的文本被拖拉时,就会持续触发拖拉事件,包括以下一些事件。
-
drag:拖拉过程中,在被拖拉的节点上持续触发(相隔几百毫秒)。
-
dragstart:用户开始拖拉时,在被拖拉的节点上触发,该事件的target属性是被拖拉的节点。通常应该在这个事件的监听函数中,指定拖拉的数据。
-
dragend:拖拉结束时(释放鼠标键或按下 ESC 键)在被拖拉的节点上触发,该事件的target属性是被拖拉的节点。它与dragstart事件,在同一个节点上触发。不管拖拉是否跨窗口,或者中途被取消,dragend事件总是会触发的。
-
dragenter:拖拉进入当前节点时,在当前节点上触发一次,该事件的target属性是当前节点。通常应该在这个事件的监听函数中,指定是否允许在当前节点放下(drop)拖拉的数据。如果当前节点没有该事件的监听函数,或者监听函数不执行任何操作,就意味着不允许在当前节点放下数据。在视觉上显示拖进入当前节点,也是在这个事件的监听函数中设置。
-
dragover:拖拉到当前节点上方时,在当前节点上持续触发(相隔几百毫秒),该事件的target属性是当前节点。该事件与dragenter事件的区别是:dragenter事件在进入该节点时触发,然后只要没有离开这个节点,dragover事件会持续触发。
-
dragleave:拖拉操作离开当前节点范围时,在当前节点上触发,该事件的target属性是当前节点。如果要在视觉上显示拖拉离开操作当前节点,就在这个事件的监听函数中设置。
-
drop:被拖拉的节点或选中的文本,释放到目标节点时,在目标节点上触发。注意,如果当前节点不允许drop,即使在该节点上方松开鼠标键,也不会触发该事件。如果用户按下 ESC 键,取消这个操作,也不会触发该事件。该事件的监听函数负责取出拖拉数据,并进行相关处理。
- 事件拖拽一共经历了6 个阶段
- 第一:dragstart ,用户刚拖拽时就立刻在被拖拉的节点上触发
- 第二:dragend, 拖拉结束,该事件也在被拖拉的节点触发
- 第三:dragover, 拖拉到目标节点的上方,当前目标节点事件持续触发。
- 第三:dragenter, 拖拉进入当前目标节点,在当前节点上会触发一次。
- 第四:dragleave, 拖拉离开当前节点范围时,在当前节点⬆️触发。
- 第五:drop, 被拖拉的节点,释放到目标节点时,会触发这个事件。这个节点可以触发添加节点或者删除节点。
(本节完!)
target.addEventListener(type,listener[,userCapture])
该方法接收三个参数 。- type :事件名称,大小写敏感,不含on
- listener : 监听的事件函数,事件发生时,会执行该事件函数
- useCapture: 可选参数(布尔类型的值) ,表示监听的事件函数是否在捕获阶段执行,如果参数是true,则在捕获阶段执行。为false则在冒泡阶段执行! 默认参数为false。 大多情况下可以不写!
- 第二个参数除了可以是一个监听函数之外,还可以是一个handleEvent方法对象。
target.addEventListener('click',{ handleEvent:function(e){console.log(e)}})
// 第三个参数就不写了- 第三个参数不仅可以选,还可以设置。当然也可以是一个配置对象。该对象具有以下属性。
- capture:布尔值,表示该事件是否在捕获阶段执行监听函数
- once:布尔值,表示监听函数是否只触发一次,然后就自动的移除。
- passive:布尔值,表示监听函数不会调用preventDefault方法。如果监听函数调用了,浏览器忽略这个请求,并在监控太台输出一行警告。
btn1.addEventListener('click',function(e){ alert('once') },{once:true})
- 第三个参数不仅可以选,还可以设置。当然也可以是一个配置对象。该对象具有以下属性。
- 这个方法的参数跟绑定的是一摸一样的。
- **但是值得注意的是,第二个参数必须是要移除绑定的事件监听函数。这两个必须一样才能够实现被移除**
- 当然参数基本保持一样就不会出现什么问题了。
(本节完!)
- 在CSS中,伪类是可以级联使用的,于是,如果列表可以匹配:first-child:nth-last-child(2)则表示当前
- 元素即是第1个子元素,又是从后往前第2个子元素,因此,我们就能判断当前总共两个
- 子元素,我们就能精准实现我们想要的布局了,只需要配合相邻兄弟选择符加号+以及兄弟选择符弯弯~即可
摘自张旭鑫 (本节完!)
- click:按下鼠标(通常是按下主按钮)时触发。
- dblclick:在同一个元素上双击鼠标时触发。
- mousedown:按下鼠标键时触发。
- mouseup:释放按下的鼠标键时触发。
- mousemove:当鼠标在一个节点内部移动时触发。当- 鼠标持续移动时,该事件会连续触发。为了避免性能问题,建议对该事件的监听函数做一些限定,比如限定一段时间内只能运行一次。
- mouseenter:鼠标进入一个节点时触发,进入子节点不会触发这个事件(详见后文)。
- mouseover:鼠标进入一个节点时触发,进入子节点会再一次触发这个事件(详见后文)。
- mouseout:鼠标离开一个节点时触发,离开父节点也会触发这个事件(详见后文)。
- mouseleave:鼠标离开一个节点时触发,离开父节点不会触发这个事件(详见后文)。
- contextmenu:按下鼠标右键时(上下文菜单出现前)触发,或者按下“上下文菜单键”时触发。
- wheel:滚动鼠标的滚轮时触发,该事件继承的是WheelEvent接口。
<!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>
<style>
div {
margin: 30px;
width: 100px;
height: 100px;
background-color: red;
}
h4 {
color: #fff;
margin: 0;
padding: 0;
}
.box {
margin: 20px;
width: 30px;
height: 30px;
background-color: pink;
}
</style>
</head>
<button id="btn1">click</button>
<button id="btn2">dblclick</button>
<button id="btn3">mousedown</button>
<button id="btn4">mouseup</button>
<div id="btn6">
<h4>mouseenter</h4>
<h4>mouseover</h4>
<div class="box"></div>
</div>
<div id="btn7">
<h4>mouseout</h4>
<div class="box"></div>
</div>
<div id="btn8">
<h4>mouseleave</h4>
<div class="box"></div>
</div>
<div id="btn9">
<h4>contextmenu</h4>
<div class="box"></div>
</div>
<body>
<script>
// 事件1 click : 按下鼠标即可以触发
btn1.addEventListener('click', function () {
alert('click')
})
// 事件2 dblclick : 双击鼠标可以触发事件 notice: 是dblclick
btn2.addEventListener('dblclick', function () {
alert('dbclick')
}, { once: true }) // 加啦once 只会触发一次。。。
// 事件3 mousedown 按下鼠标键时触发 notice:此时我们监听的是对于按钮按下鼠标键才会触发
btn3.addEventListener('mousedown', function () {
alert('mousedown')
})
// 事件4 mouseup // 点击按钮释放鼠标键时才会触发
btn4.addEventListener('mouseup', function () {
alert('btn-mouseup')
})
// 事件4 mouseup // 释放鼠标键时将会触发 那就绑定为document的全局事件
document.addEventListener('mouseup', function () {
alert('mouseup')
}) // notice 在为看来,不要在全局绑定这样的事件,这样的事件一般在函数内去触发
// 事件5 mousemove 鼠标在移动时触发,但是会持续触发。(test 时,也使用去全局触发)
document.addEventListener('mousemove', function () {
// console.log('moving……')
})
// 事件6 鼠标进进入节点的一瞬间会立刻触发,当进入子节点,或者离开时不会触发
btn6.addEventListener('mouseenter', function () {
// console.log('i am comming ^_^')
})
// 事件7 鼠标进入节点会触发这个事件。进入子节点还会触发这个事件
btn6.addEventListener('mouseover', function () {
console.log('i am mouseover')
}) // 注意 这个与mouseenter 相比。会多触发两次 。离开节点进入自节点会触发,离开子节点,进入父级节点还会触发一次
// 事件8 mouseout 鼠标离开节点时触发。当鼠标进入子接点,相当离开父节点,一样会触发该事件。从子节点离开,进入父级节点一样会触发该事件!
btn7.addEventListener('mouseout', function () {
console.log('i am mouseout')
})
// 事件9 mouseleave // 只有在父级节点出去的时候才会触发。不会在子节点中触发
btn8.addEventListener('mouseleave', function () {
console.log('i am mouseleva')
})
// 事件10 contextmenu // 点击右键触发
btn9.addEventListener('contextmenu',function(){
console.log('i am contextmenu')
})
// 事件11 wheel
document.addEventListener('wheel',function(){
console.log('i am mousewheel')
})
</script>
</body>
</html>
- click 事件是指的是,用户在同一个位置完成mousedown动作,在完成mouseup动作。因此呢,执行的顺序分别为,mousedown 首先触发 -后续执行mouseup,然后执行click
- dbclick 事件会在mousedown - mouseup click - 后执行!
- mouseenter 和 mouseover 都是鼠标进入事件触发,但是两者的区别是。前者只触发一次,后者会在子节点上多次触发。
- mouseout mouseleave 都是鼠标离开时候触发。但两者的区别是,前者会在子节点内多次触发。后者只会离开节点时触发。
键盘事件是由用户敲击键盘时触发的,主要的事件有,keydown/keyup/keypress/三个事件。都继承自 `keyboardEvent` 接口
keydown
:按下键盘时触发keypress
:按下具体的键盘值触发。即Ctrl、Alt、Shfit、Meta这样无值的的键。这个事件不会触发。。。对于有值的事件,按下时首先会触发keydown事件,再触发这事件。keyup
: 松开键盘时触发。 触发事件顺序 如果用户一直在按着键盘不放,首先会触发keydown---keypress---keydown-----keypress--- 直到键盘抬起触发-keyup
document.body.addEventListener('keydown',function(e){
console.log(e.key)
switch(e.key){
case 'Enter':
// console.log('Enter事件')
test()
break
case 'h':
console.log('h事件')
break
}
},false)
(本节完!)
- 第一个是contains - 第二个是compareDocumentPosition() ['地址'](https://developer.mozilla.org/zh-CN/docs/Web/API/Node/compareDocumentPosition) - 具体API介绍,可以上MDN上。 (本节完!) 进度事件用来描述资源加载的进度,主要是由ajax请求。img、audio、video、style、link等外部资源的加载触发。继承了progressEvent接口。主要有以下几个事件 - abort: 外部资源中止加载时(用户取消)触发。但是如果发生错误导致终止,该事件不会触发 - error: 由于错误导致外部资源无法加载时触发。 - load: 外部资源加载成功时触发 - loadstrat: 外部资源开始加载时触发 - loadend: 外部资源停止加载时触发,发生顺序排在error、abort、load等事件的后面。 - progress: 外部资源加载不断的触发 - timeout: 加载超时时触发。 (本节完!)- 如果只是为了获取当前页面的窗体的滚动高度,直接使用window.pageYOffset就可以(IE9+) ,桌面和移动端都是支持的
- 但是windows.pageYOffset是一个只读的属性,我们无法用来设置窗体的滚动高度,此时,就要找到对应的滚动元素,通过设置scrollTop的值来改变窗体的滚动位置。 注意 对于桌面端和手机端窗口的滚动元素都是不一样的
document.documentElement.scrollTop
是可以获取pc端窗口的滚动大小。document.body.scrollTop
是可以用来获取手机端滚动窗体的大小。
通过上面两个例子可以知道获取滚动高度不一致的现象是兼容问题引起的如果需要做适配,则需要写兼容代码。(就是两个代码一起写)
因此document.scrollingElement
这个属性才应运而生。可以直接动态的识别滚动的容器,无需在多写一连串的代码
document.scrollingElement
在桌面端就是documen.documentElement,在移动端就是document.body
当然也可以直接设置其scrollTop
的属性值来改变窗体的位置。
本文参考张旭鑫--- 建议以后使用document.scrollingElement
来获取窗口的高度
(本节完!)
<script>
input.addEventListener('input',(e) =>{
console.log(e)
})
</script>
input事件跟change事件十分相似,都是会监听input输入事件,但是
上,而不是发生在表单成员上。change
事件不同的是会只有当离开焦点时才会触发。 **select事件
**select事件当在input、textarea里选中文本时会触发change事件
该事件特点不会连续触发,只有当用户输入完成之后才会触发。 invalid事件用户提交表单的验证事件,如果表单元素的值不满足校验条件。则就会触发invalid事件。 reset,submit 事件 这两个事件都发生在表单对象
reset
事件当表单重置(所有的表单成员都会变成默认值)时触发submit
事件当数据向服务器提交时触发 。非常需要注意的时submit
事件的发生对象时元素,不是 (本节完!)
html, body { scroll-behavior:smooth; }
- 注意一定的兼容性~ (本节完!)
- 浏览器的触摸事件由三个部分组成,
touch
,touchuList
,touchEvet
,这个三个部分分别是代表一个触摸点,多个触摸点的集合,还有触摸引发的事件案例。 - touch的接口实例对象用来表示触摸点(一根手指或者一根触摸笔),包括了位置,大小,形状,压力,目标元素等属性。
- touchList 表示多个触摸点组成的。
- touchEvent 的接口实例对象代表由触摸引发的事件,(只有触摸屏才会引发这一个事件)
浏览器原生提供了Touch构造函数,用来生成touch实例。
var touch = new Touch(touchOptions);
拥有非常多的参数可选,具体可以点击这里 查看
ctrlKey
: 布尔值,表示Ctrl键是否同时按下,默认值为falseshiftKey
: 布尔值,表示Shift值是否同时按下,默认值为falsealtKey
: 布尔值,表示Alt键是否同时按下,默认值为falsemetaKey
: 布尔值,表示触摸时,是否按下了Meta键(也就是window)。
TouchEvent.changedToches
属性返回一个TouchList实例,成员是由一组Touch实例对象,表示本次触摸的相关节点 。
touchstart
事件:被激活的该触摸点touchmove
事件,发生变化的触摸点。touchend
事件,消失的触摸点。
(本节完)
单线程模型指的是,JavaScript 只在一个线程上执行,也就是说,JavaScript同时只能执行一个任务,其他任务都必须都在后面排队等待。
但是值得注意的是,JavaScript只在一个线程上运行,不代表JavaScript引擎只有一个线程。事实上,JavaScript引擎有多个线程,单个脚本只能在一个线程傻姑娘运行(称为主线程),其他线程都是在后台配合。
JavaScript在设计的时候就是使用的单线程,为什么这么设计呢?JavaScript在开发的时候就是为网页服务的语言,然是对于一个网页脚本来说。如果JavaScript有两个线程,一个线程在网页DOM节点上来添加内容,另一个线程却删除来这个节点,所以为了避免这个问题的出现。JavaScript的单线程特点也成为了这门语言的核心。
单线程的好处是,运行环境单纯,实现起来比较简单。缺点,如果执行一个任务耗时非常长的话,后面的任务必须进行排队等待,就导致整个程序执行的延缓。常见浏览器无响应(假死),通常就是某一段JavaScript代码长时间的执行(死循环造成的) 。JavaScript语言本省执行的速度并不慢,只是执行读取外部数据时,比如执行的ajaxa请求返回结果。这个时候,如果对方服务器长时间没有反应的话,或者网络不通畅,就会导致脚本长时间的停滞。
这么语言这好玩的是,排队如果是因为计算量大,CPU忙不过,倒也是算了。但是很多时候,CUP实际上是空闲的,因为IO操作(输入输出)很慢(比如Ajax操作读取网络数据),不得不等记过出来。在往下执行。JavaScript采用的是“事件循环”机制,什么是事件循环机制呢?(JavaScript语言的设计者意识到,CPU完全可以不管IO操作,挂起处于等待中的任务,先运行排在后面的任务,等到IO操作返回了结果,在回头取,把挂起的任务继续执行下去)
程序里面所有的任务都可以分为两类:同步任务和异步任务。
同步任务是没有被引擎挂起,在主线程上排队执行的任务。只有一个任务执行完了,才能去执行下一个任务。
异步任务是那些被引擎放在一边,不进入任务队列的任务。只有引擎认为某个异步任务可以执行了(比如Ajax的操作从服务器得到了结果—),该任务(采用回调函数的形式)才会进入主线程执行。排在异步任务后面的代码,不用等待异步任务结束会马上运行。也就是说,异步任务不具有堵塞的效应。
JavaScript运行时,除了一个正在运行的主线程,引擎还提供了一个任务队列(task queue),里面是各种需要当前程序处理的异步任务。(实际上,根据异步任务的类型,存在多个任务队列)
首先,主线程会去执行所有的同步任务。等到同步任务全部执行完毕,就会去看任务队列里面的异步任务。如果满足条件的话,那么异步任务就会重新进入到主线程开始执行,这时它就变成了同步任务了,等到执行时,下一个异步任务在进入主线程开始执行。一旦任务队列被清空,程序就会结束执行。
JavaScript引擎怎么知道异步任务有没有结果呢?能不能进入主线程呢?答案就是引擎在不停的检查,一遍又一遍,只要同步任务执行完了,引擎就开始检查哪些挂起的异步任务。是不是可以进入主线程。这种循环的检查机制,叫做事件循环(event loop)。
回调函数是异步操作最基本的方法。
function f1(){}
function f2(){}
// 调用
f1();
f2();
// 这两个函数执行顺序是。f1执行完毕之后,f2才能执行。
// 但是如果f1是异步函数,f2就会先执行。
function f1(callback){
callback();
}
function f2 (){
}
// f1(f2)
// 这样的话,f2会立即执行。不会等到f1执行结束后。
回调函数的优点是,容易理解和实现,缺点是不利于代码的阅读和维护。各部分之间高度耦合,使得程序结构混乱,难以追踪(尤其是多个回调函数嵌套的情况),而且每个任务只能指定一个回调函数。
另外一个思路是采用事件驱动模式,异步任务的执行不取决于代码的顺序,而取决某个事件是否发生。
(未完!待续)
var a = 0;
b = 0;
function A(a) {
// var a = 1;
A = function (b) {
// b = 2
// b = ++b // 3
// b = 3
console.log("ok")
console.log((b+ ++b));
}
console.log((a++),'ok');
}
A(1); // 1 'ok'
A(2); // ok 5
A(3); // ok 7
- 隐式全局变量
- 函数被覆盖成为全局函数 分析 只有当A(1) 被执行时,才会将A 中的函数重新赋值,变成全局函数,所以在第二次调用的时候,直接覆盖掉了之前的函数。
- 函数节流:指定时间间隔只会执行一次任务。(只介绍函数节流)
- 函数防抖:任务频繁的情况下,只有任务触发时间超过指定间隔的时候,任务才会执行。
通过监听window
对象的scroll
事件来分析这个问题,然后在函数体内中写入判断是否滚动底部的逻辑;
// 这里写原生js html,bdoy的高度设置为3000
document.addEventListener('scroll',function(e){
// 判断浏览器是否滚动到地步的逻辑
// console.log(e.target.body.offsetHeight)
let pageHeight = e.target.body.offsetHeight;
// body 总高度3000
let scrollTop = document.scrollingElement.scrollTop;
// 滚动高度。 变量
console.log(scrollTop);
let winHeight = window.innerHeight;
// 获取浏览器窗口的高度
// 浏览器窗口高度821
console.log(window.innerHeight);
//定义一个阀值
let thresold = pageHeight - scrollTop - winHeight;
// 进行判断
if (thresold > -100 && thresold <= 20) {
console.log('到底了');
}
})
可以自行测试这行代码
会发现一个问题, 这个代码在监听scroll
事件的时候,会不停的触发我们设置的阀值计算。不停的进行判断是否到底,这样做,极大的消耗了浏览器的性能,而且在实际的开发场景中,不需要这么去做。实际开发过程中,每隔一段时间去判断这个逻辑效果比较好。所以*函数节流所可以做的就每隔一段时间去执行一次原本需要无时无刻都需要在执行的函数。这个就是函数节流
的最大用处。
// 这个定义的函数需要在上面的那个函数中去被调用
function throttle(fn, interval = 500){
// 设置节流阀
let canRun = true;
return function () {
// 判断这个节流阀是否为true 。
if (!canRun) return ;
// 如果为true,进来之后,设置为false
canRun = false;
// 设置一个300毫秒的定时器
setTimeout(() => {
// 改变this指向
fn.apply(this,arguments);
// 在执行之后,在将节流阀值设置为true
canRun =true;
}, interval);
}
}
加上函数节流之后,当页面滚动的时候,每隔500ms才会去执行逻辑判断。 简单的来说,函数的节流就是通过一个闭包来保存一个标记,在函数的开头判断这个标记是否为true,如果为true的话就继续执行函数,否则就return掉,判断完标记后又立即把这个标记设置为false,然后在把外部传入的函数在一个定时器中去执行,在执行完毕之后,在把这个节流阀的值设置为true。表示可以执行下一次的循环,当定时器还没执行的时候,节流阀的值始终是false,就不会执行我们的判断函数。直接被return掉了。
(本小节完!)
之所以插入这个小节,是对函数节流中使用了apply方法的一个补充知识
// 使用apply方式
var obj = {
name:"zhuyaogeng"
}
function full(firstName, lastName){
console.log(firstName+'==='+'' + this.name + '' +'==='+ lastName);
console.log(this);
}
// 注意这里的参数是一个数组
full.apply(obj,['zhu','yaogeng']); // firstName zhu zhuyaogeng lastName yaogeng
//使用call方式
var obj1 = {
age:19
}
var getInfo = function(A,B){
this.name = 'zhuyaogeng';
console.log('this is a '+ A +' '+ B ,',he age is ' + this.age );
console.log(this);
}
// 注意这里的参数是一个个的对象
getInfo.call(obj1,'good','boy'); // this is a good boy ,he age is 19
总结
- 两种方式都是传递的两个参数,一个是作为函数上下文的对象,一个是作为函数参数所组成的数组,或者是参数列表。
- 两者用法一样,只是参数不一样。
var obj3 = {
name:'zhuyaogeng'
}
function full1(){
this.age = 20;
console.log(this);
console.log(this.name);
// 调用之后相当于
// console.log(obj3.name);
}
full1.call(obj3); //zhuyaogeng
同理 apply
用法一致
var obj4 = {
age:22
}
function full2(){
console.log(this.age);
}
full2.apply(obj4);
call和appl方式的第一参数所作为函数上下文的对象,这里把obj3作为参数传递给函数。传递之后,函数的this指向就变成了obj3的对象。
var Person1 = function (){
this.name = 'zhuyaogeng';
}
var Person2 = function (){
this.getName = function(){
console.log(this);
console.log(this.name);
}
//借用了Person1中的属性
Person1.call(this);
// console.log(this);
}
var person = new Person2();
person.getName();
/**
* Person2实例化出来的对象通过getName方法拿到了Person1中的name..因为
* 在Person2中,Person。call(this)的作用就是使用了Person1代替了this对象
* 那么Person2就用了Person1中的属性和方法了,相当于Person2继承了Person1
* 的属性和方法 。
*/
function foo(){
console.log('hello world');
}
//调用函数
foo.call(); // hello world
- 改变this指向问题
- 调用别的对象的方法
- 调用函数
(本小节完!)
以用户注册时验证为例,验证用户名是否被注册,很多网站为了提高用户体验,不会让输入框失去焦点时在去判断这个用户是否被占用,而是在输入的时候就判断这个用户名是否已经存在。
// 抖动存在的demo
let input = document.getElementById('text');
let span = document.getElementById('span');
input.addEventListener('input', function () {
// 动态获取输入框的值
let text = this.value;
console.log(text);
var xhr = new XMLHttpRequest();
xhr.open('get', './test.json');
xhr.send(null);
xhr.onreadystatechange = function () {
if (xhr.readyState === 4) {
// console.log(xhr.responseText)
var data = xhr.responseText;
// 将字符串变成数组
var arr = JSON.parse(data);
for (var i = 0; i < arr.length; i++) {
// console.log(arr[i].name)
// console.log('----')
if (text != arr[i].name) {
// 用户名不存在可以注册
span.innerText = '恭喜!该用户名可以注册';
span.style.color = 'green';
span.style.fontSize = '10px';
} else {
span.innerText = '用户名已存在!';
span.style.color = 'red';
span.style.fontSize = '10px';
}
}
if (!text.length) {
span.innerText = '';
}
}
}
})
如果是该种方式,极大的消耗了浏览器的运行性能,对用户计算机来说是个考验,这个就是所谓的抖动问题。那么问题既然出现。解决方案如下;
let input = document.getElementById('text');
let span = document.getElementById('span');
input.addEventListener('input', delay(function () {
// 动态获取输入框的值
let text = this.value;
console.log(text);
var xhr = new XMLHttpRequest();
xhr.open('get', './test.json');
xhr.send(null);
xhr.onreadystatechange = function () {
if (xhr.readyState === 4) {
// console.log(xhr.responseText)
var data = xhr.responseText;
// 将字符串变成数组
var arr = JSON.parse(data);
for (var i = 0; i < arr.length; i++) {
// console.log(arr[i].name)
// console.log('----')
if (text != arr[i].name) {
// 用户名不存在可以注册
span.innerText = '恭喜!该用户名可以注册';
span.style.color = 'green';
span.style.fontSize = '10px';
} else {
span.innerText = '用户名已存在!';
span.style.color = 'red';
span.style.fontSize = '10px';
}
}
if (!text.length) {
span.innerText = '';
}
}
}
}))
// 介于bug
function delay(fn , interval = 500) {
// 定义一个定时器
let tiemout = null;
return function(){
// 清空定时器
clearInterval(tiemout);
// 设置定时器
tiemout = setTimeout(() => {
fn.apply(this);
}, interval);
}
}
以上代码尽量用到了原生的js. 当然使用 jquery
可能就几行代码。都可以尝试哦~
函数防抖的原理跟函数节流类似。通过闭包
来保存了一个setTimeout
的返回值,每当用户输入的时候,把前面的setTimeOut
清理掉。然后又重新的创建一个新的setTimeout
,这样能保证输入字符后定时器的间隔时间内还有字符输入的化,就不会执行fn函数。
函数节流和函数防抖都是巧妙的利用了setTimeout来存放将要执行的函数,这样可以很方便的利用清楚定时器在合适的时机来清除执行的函数。
不管使用何种方式,何种目的,最终都是为了解决性能问题
(本节完!)
setTimeout()
函数用来指定某个函数或某段代码,在多少毫米之后执行。它返回的是一个整数,表示定时器的编号,以后可以用这个定时器的编号来取消这个定时器。
可以参考day10文件夹代码
setTimeout()
传递两个参数,第一个参数是一段代码
或者是函数
。- 第二个参数是
delay
也就是延迟的毫秒数。 - 第二个参数可以省略,但是如果省略,则默认为0。
- 值得注意的是,
setTimeout
还可以传递多个参数 ,如:
let timerId = setTimeout(function(a, b) {
console.log(a + b);
},1000, 1, 1);
- 以上代码,传递了四个参数,前面两个参数已经介绍,那么第三个和第四个参数是代表着,将在1000毫秒之后返回回调函数执行时,作为回调函数的参数传入。
- 值得注意的是,如果回调函数是对象的方法,那么在定时器使得方法内部的
this
关键字指向全局环境,而不是定义时所在的那个对象。看下面的代码---
var obj = {
a:1,
sayHello:function(){
console.log(this.a);
}
}
obj.sayHello(); // 1
let timerId = setTimeout(obj.seyHello,1000); //undefined
原因是,obj.sayHello
在一段时间延时后,this
的指向已经不是obj,而是全局环境。
当然为了防止这个发生。第一种解决方案是将obj.sayHello
放入一个函数中。
timerId = setTimeout(function(){
obj.sayHello();
},1000)
// 当然执行的结果又是 1
这样做的好处是,使得obj.sayHello在obj的作用域内执行,而不是在全局作用域内执行,所以能显示正确的值。
- 解决的方式2
timerId = setTimeout(obj.sayHello.bind(obj),1000);
将obj.sayHello绑定到obj上面也是可以的哦
setTimeout
是一次性定时器,返回值是个整数的ID 更多参考,点我查看阮一峰内容
(本节未完,待续~)
两者唯一的区别是,setInterval()
无限次执行,而setTimeout()
是指定某任务每隔一段时间也就执行一次。
setTimeout()
和 setInterval()
函数,都返回一个整数值,表示计算器编号。将该整数传入clearTimeout()
和 clearInterval()
函数,就可以取消对应的定时器。
(本节完!)
Promise
是异步编程的一种解决方案,比传统的解决方案一一回调(是否听说传说中的回调地狱?)�更加合理强大。它最早由社区提出和实现,ES6将它写进来标准的语言语法,统一了用法。并且提供了Promise
对象。
什么是Promise
,简单的来说它是一个容器,里面包裹着某个未来才会结束的事件(通常是一个异步的操作)的结果。从语法来说,Promise
是一个对象,从它可以获取异步操纵的信息。Promise
提供了统一的API,各种异步操作都可以用同样的方式进行处理。
- 对象的状态不受外界影响。
Promise
对象代表的一个异步操作,有三种状态:pending(进行中)
、fulfilled(已成功)
,rejected(已失败)
。只有异步操作的结果,可以决定当前是哪一种状态。任何其他操作都无法改变这个状态。这也是Promise
这个名字的由来,英文意思为承诺,信守诺言
。表示其他手段无法改变。 - 一旦状态改变,就不会在变。任何时候都可以得到这个结果。
Promise
对象的状态改变。只有两种可能:从pending
到fulfilled
,和从pending
到rejected
。只要这两种状况发生。状态就凝固了,不会在改变,会一直保持这个结果。这时就称为resolved(已定型)
。如果改变已经发生,再次对Promise
对象添加回调函数,也会立即得到到这个结果。这个于事件(Event)完全不同,事件的特点是,如果一🥚错过,在去监听,是得不到结果的。(失去了,就失去了,记住承诺Promise
)。
但是Promise
有一定的缺点:1.无法取消Promise
,一🥚新建它就会立即执行,无法中途取消。2.如果不设置回调函数,Promise
内部抛出异常,外部不会收到反馈。3.当pending
状态时,无法得知目前进展到那个状态(是刚开始,还是即将完成。)
const promise = new Promise(function (resolve, reject) {
// let a = '';
// code
if ('') {
// 异步操作成功
resolve(a);
} else {
reject(err);
}
})
Promise
构造函数接受一个函数作为参数,该函数的两个参数分别是resolve
和reject
。它们是两个函数,是js原生提供的。不用自己去部署/..
resolve
函数的作用是。将Promise
对象的状态从未完成
变成成功
。在异步成功调用,并将异步的结果,作为参数传递出去 。
reject
函数的作用,是将Promise
对象从未完成
变为失败
,在异步操作失败时。将异步操作报错的错误,作为参数传递出去。
Promise
实例生成之后,可以用then
方式分别指定resolved
状态和rejected
状态的回调函数。
promise.then(function(value){
// 成功结果
},function(error){
// 失败结果
})
当然错误的函数是可选�
的
// 异步操作ajax
const getJson = function (url) {
const p = new Promise(function (resolve, reject) {
// 处理函数
const hander = function () {
if (this.readyState !== 4) {
return;
}
if (this.status === 200) {
return resolve(this.response);
} else {
reject(new Error(this.statusText));
}
};
const xhr = new XMLHttpRequest();
xhr.open('get', url);
xhr.send(null);
xhr.onreadystatechange = hander;
// 设置响应类型
xhr.responseType = 'json';
});
return p;
}
getJson('./data.json').then(function (data) {
console.log(data);
}, function (err) {
console.log(err);
})
一般来说,调用reslove
和 reject
以后。Promise
的使命已经完成。后续操作都会在then
方法去里面。而不应该直接写在resolve
或者reject
的后面。所以,最好是给他们retun
出去。避免意外情况。
(本小节完~)
在vue中构建单个页面,路由模块一般都是使用vue-router
,vue-router
不会自动的保存被切换的组件状态,当组件进行push
或者replace
时。旧组件会被销毁,而新的组件会被创建,这也就是会体现出vue的生命周期
。
需求 当跳转到详情页时,需要保持列表页的滚动条的深度,等返回时,依旧保持这个高度,极大的提高了用户体验度。这个时候,keep-alive
组件的作用应运而生。
<keep-alive>
<router-view></router-view>
<keep-alvie>
值得注意的时keep-alive
只是一个抽象组件,实际上是不会被渲染在dom树上的,它的作用是在内存中缓存组件(不让组件进行销毁),等到下次渲染的时候,还会保持其中的所有状态。并且会触发activeted
的钩子函数,因为缓存的一般都是出现在页面被切换的时候,故一般都是和rouer-view
一起结合使用。如果将<router-view>都包裹起来的话
,里面渲染的组件都会被缓存。
需求(条件缓存) 只想缓存一部分组件,不想所有的组件都被缓存。可以使用keep-alive
组件中的include/exclude
属性。include
属性表示要缓存的组件名。对于这个问题需要注意,平时书写组件的时候,都需要将组件很清楚的表达当前组件的名字,用法如下。
export default {
name:'组件名字'
}
include
接受的类型可以为String
,RegExp
,String数组
。 exclude
属性跟它有着相反的作用,匹配到的组件不会被缓存。如果页面太多,不想缓存这个组件可以使用该属性。include
的具体用法
<keep-alive :include="['component-1','component-2']">
<router-view />
<keep-alive>
需求 有一个大的页面:首页A,列表页B,详情页C,一般都是可以从A -> B -> C
,B -> C
在返回到B
时,B
需要保持到列表滚动的距离。然而对于B -> A -> B
时,B
不需要保持状态,是全新的需求 。
从这个需求来看。B组件
是条件缓存的,只要C -> B
时,保持缓存。当A -> B
时,放弃缓存。解决方案如下:在vuex中定义一个全局的缓存组件,等待传给 include
export defalut {
namespaced:true,
state:{
keepAliveComponents:[] //缓存数组
},
mutations:{
// 需要缓存的组件
keepAlive(state, component) {
!state.keepAliveComponents.includes(component) && state.keepAliveComponents.push(component)
},
// 不需要缓存的组件
noKeepAlive(state, component) {
const index = state.keepAliveComponents.indexOf(component)
index !== -1 && state.keepAliveComponents.splice(index, 1)
}
}
}
在父页面中定义keep-alive
,并且需要传入全局的缓存中
<div class="app">
<!-- 传入include数组 -->
<keep-alive :include="keepAliveComponents">
<router-view />
<keep-alive>
<div>
export default {
// 计算属性
computed:{
...mapState({
keepAliveComponents:state=> state.global.keepAliveComponentes
})
}
}
缓存机制: 在路由配置页面中,约定meta
属性 keepAlive
,值为true表示组件徐需要缓存。在全局路由钩子beforeEach
中对该属性进行处理。这种做法,对每一次进入该组件,都进行缓存。
const router = new Router({
routes:[
{
path:'/A/B',
name:'B',
component:B,
meta:{
title:'B页面',
keepAlive:true //这里指定B组件的缓存性
}
}
]
})
router.beforeEach((to, form ,next) =>{
//在路由全局钩子函数中,根据keepAive属性,统一设置页面的缓存行
// 作用时每次进入该组件,就将它缓存
if (to.meta.keepALive) {
stroe.commit('global/keepAlive',to.name)
}
})
取消缓存的时机:对缓存组件使用路由的组件钩子beforeRouteLeave
。 B -> A -> B
时不需要缓存 B
,所以可以认为:当B的下一个页面不是C
时就取消B
的缓存,那么下次进入B组件的时候B就是全新的。
export default {
name:'B',
created(){
// 。。。设置滚动条在最顶部
},
beforeRouteLeave(to, from, next){
// 判断页面是否是C ,如果不是则取消B 的缓存
if (to.name !== 'C') {
this.$store.commit('global/noKeepAlive', from.name)
}
// 如果是的,直接
next()
}
}
因为组件B的条件缓存,是B自己职责。所以最好把该业务逻辑写在B的内部,而不是A中,这样不至于让组件之间的跳转变得混乱。
(本小节完!)
// 构造函数的使用方式 构造函数命名必须用大写来区分
var Foo = function(){
this.name
}
// 实例化 必须使用new
var f = new Foo() // 这个括号可以写可以不写 new可以执行函数
- this 表示的是实例化的对象
- 每个实例话对象都必须使用new出来
值得注意的是,如果不实用new 的话,是无法生成实例化对象的,这时构造函数的属性就会变成全局对象window的属性,为了避免这种情况 。最好使用严格模式
var Foo = function(){
'use strict'
this.name
}
如果是实例化不使用new关键字的话就会报错。 但是如果不实用严格模式不会报错,产生全局变量,造成全局污染
- 创建一个空对象,作为要返回的对象实例
- 将这个空对象的原型指向构造函数的
prototype
属性上 - 将这个空对象赋值给函数内部的this关键字
- 开始执行构造函数的内部代码
也可以换个说法
- 在内存中开辟一片空间,存储创建的对象
- this 就是实例化的对象
- 设置对象的属性和方法
- 把创建后的对象返回
let person = {
name:'zs',
age:23,
sayHello:function(){
console.log('hello')
}
}
// 使用Object.create() 实例化一个对象
let person1 = Object.create(person)
console.log(person1.age)
(本节完!)
在我看来路由有两个方面的作用:
- 通过路由变化,实现页面之间的跳转和切换
- 可以通过路由传递参数(query)
路由的种类可以分为以下四种
- hash 主要是基于锚点实现
- browser 使用的是
html5
中的history
这个api来实现的,每次路由发生变化都会重定向 - memory 这种实现是在内中维护一个堆栈管理历史记录,由于比较复杂,现在一般都不用了
- static 是后端管理路由的模式
hash路由的优点
- 兼容性比较好,兼容性达到了ie8
- 绝大数框架的框架都基本支持hash路由方式
- 除了会发送ajax和资源加载之外不会发送其他请求
- 不需要在服务端进行任何设置和开发
hash路由的缺点
- 服务端无法准确捕获路由的信息
- 对于需要锚点功能的需求会与当前路由机制发生冲突
- 对于需要重定向的操作,后段无法获取url全部内容,导致后台无法得到url数据,典型的例子就是微信公众号的oauth验证。
browser路由的优点
- 当发生路由重定向时不会丢失url数据 ,后端也可以拿到这个数据
- 当然,绝大多数框架一样实现了
browser
路由的方式 - 后端可以准确追踪到路由
- 可以使用history.state获取路由的信息
使用browser的缺点
- 兼容性不如hash 。兼容性只到ie10
- 需要后端支持,每次返回html文档
(本节完~)
cookie英文饼干,顾名思义,大小应该非常小,cookie非常小,它的大小限制在4kb左右,是网景公司的前雇员在1993年发明。它的主要用于保存登陆信息,比如你登陆某个网站市场可以看到'记住密码’,这就是通过在cookie中存入一段辨别用户身份的数据来实现的。
localStorage是HTML5标准中新加入的技术,当然早在IE6时代就有一个userData的东西用于本地存储,而当时考虑到浏览器的兼容性,更通用的方案是使用flash。如今localStorage被大多数浏览器所支持。
sessionStorage与localStorage的接口类似,但保存数据的生命周期与localStorage不同,做过后端的同学都知道Session这个词,翻译过来就是会话。而sessionStorage是前端的一个概念。它只是可以将一部分数据在当前会话中保存下来,刷新页面数据依旧存在。但是叶敏啊关闭后,sessionStorage中的数据就会被清空。
Cookie 一般由服务器生成,可设置失效时间,如果在浏览器端生成cookie,默认是关闭后失效。
localStorage 除非被永久清除,否则永久保存。
sessionStorage 仅在当前会话会有效,关闭页面或浏览器后被清除
Cookie 一般为4kb
localStorage 和 sessionStorage 一般为5mb
Cookie 每次都会携带HTTP头中,如果使用cookie保存过多数据会带来性能问题
localStorage 和 sessionStorage 仅在客户端(即浏览器)中保存,不参与和服务器的通信。
Cookie 需要程序员自己来封装,原生的cookie接口不够友好
localStorage 和 sessionStorage 原生接口可以接受,可以封装来对Object和Array有更好的支持。
因为考虑到每个HTTP请求都会带着Cookie的信息,所以Cookie当然能是精简就精简力,比较常用的一个应用场景就是判断用户是否登陆,针对登陆过的用户服务端就在它登陆时往Cookie中祝福一段加密过的唯一识别单一用户的辨识码,下次只要读取这个值就可以判断当前用户是否登陆。曾经还使用Cookie来保存用户在电商网站上的购物车信息,如今有来localStorage,这一切问题变得越来越轻松。
- typeof 运算符
- intenseof 运算符
- Object.prototype.toString 方法
- 原始类型 number,string,boolean
- 函数
- undefined
- object
- typeof window
- typeof {}
- typeof []
- typeof null
- 空数组[]返回的是object,这表示,JavaScript内部,数组本质是一种特殊的对象。
- null的返回值是object,这是由于历史原因造成的,1995年JavaScript语言的第一版,所有值都设计成32位,其中最低的3位用来表述数据类型,object对应的值是000。当时,只设计了五种数据类型(对象、整数、浮点数、字符串和布尔值),完全没考虑null,只把它当作object的一种特殊值,32位全部为0。这是typeof null返回object的根本原因。
当然了,为了兼容以前的代码,后来就没法修改。这并不是说null就属于对象,本质null是一个类似与undefined的特殊值。
instance 是判断变量是否为某个对象的实例,返回值为true或false
- 由于typeof对数组[]和对象{} 的返回值Object,无法区分数组和对象,但是instanceof可以区分。 数组 返回的是true,因为数组是Object的一个子类。