Javascript的一些经验总结
JavaScript作用域
1.作用域
JavaScript的作用域与C、Java等语言不同,它不是以花括号包围的块级作用域,这个特性经常被大多数人忽视。例如下面代码,在大多数类C的语言中会出现变量未定义的错误,但在JavaScript中却完全合法:
if (true) {
var msg = 'msg';
}
console.log(msg); // 输出 msg;
这是因为JavaScript的作用域完全是由函数来决定的,if、for语句中的花括号不是独立的作用域。
2.函数作用域
不同于大多数类C的语言,由一对花括号封闭的代码块就是一个作用域,JavaScript的作用域是通过函数来定义的,在一个函数中定义的变量只对这个函数内部可见,我们称为函数作用域。在函数中引用一个变量时,JavaScript会先搜索当前函数作用域,如果没有找到则搜索其上层作用域,一直到全局作用域。下面是一个简单的例子:
var scope = 'global';
var f1 = function() {
console.log(scope);
}
f1(); // 输出 global
var f2 = function() {
var scope = 'f2';
console.log(scope);
}
f2(); // 输出 f2
以上示例十分明了,JavaScript的函数定义是可以嵌套的,每一层是一个作用域,变量搜索顺序是从内到外,按照作用域搜索顺序,在console.log函数访问scope变量时,JavaScript会先搜索函数f2的作用域,恰巧在f2的作用域里搜索到了scope变量,所以上层作用域中定义的scope就被屏蔽了。但下面这个例子可能就有些让人困惑:
var scope = 'global';
var f3 = function() {
console.log(scope);
var scope = 'f3';
}
f3(); // 输出 undefined
复制代码
上面的代码可能和你的预想不一样,并没有输出global,而是undefined,这是为什么呐?这是JavaScript的一个特性,就是变量声明语句永远在该作用域里最先被执行,所以上面的例子形同如下:
var scope = 'global';
var f4 = function() {
var scope; // 变量声明最先被执行
console.log(scope);
scope = 'f4'; //变量被赋予值
}
f4(); // 输出 undefined
这样就不难理解运行f3()函数为什么没输出global,而是undefined了。
3.函数作用域的嵌套
一个作用域嵌套的例子:
var f5 = function() {
var scope = 'first';
(function() {
var scope = 'second';
(function() {
console.log(scope);
}());
}());
}
f5(); // 输出 second
我们在最内层函数中引用到了scope变量,通过作用域搜索,找到了其父作用域中定义的scope变量。
有一点要注意:函数作用域的嵌套关系是在定义时决定的,而不是在调用时决定的,下面是一个简单的例子:
var scope = 'global';
var f6 = function() {
console.log(scope);
}
var f7 = function() {
var scope = 'f7';
f6();
}
f7(); // 输出 global
这个例子中,通过f7调用的f6在搜索scope时,找到的是f6函数的父作用域中定义的scope变量,而不是f7中定义的scope变量。这说明了函数作用域的嵌套关系是在定义时决定的,而不是在调用时决定的。
Javascript的this用法
this是Javascript语言的一个关键字。
它代表函数运行时,自动生成的一个内部对象,只能在函数内部使用。比如,
function test(){
this.x = 1;
}
随着函数使用场合的不同,this的值会发生变化。但是有一个总的原则,那就是this指的是,调用函数的那个对象。
下面分四种情况,详细讨论this的用法。
情况一:纯粹的函数调用
这是函数的最通常用法,属于全局性调用,因此this就代表全局对象Global。
请看下面这段代码,它的运行结果是1。
function test(){
this.x = 1;
alert(this.x);
}
test(); // 1
为了证明this就是全局对象,我对代码做一些改变:
var x = 1;
function test(){
alert(this.x);
}
test(); // 1
运行结果还是1。再变一下:
var x = 1;
function test(){
this.x = 0;
}
test();
alert(x); //0
情况二:作为对象方法的调用
函数还可以作为某个对象的方法调用,这时this就指这个上级对象。
function test(){
alert(this.x);
}
var o = {};
o.x = 1;
o.m = test;
o.m(); // 1
情况三 作为构造函数调用
所谓构造函数,就是通过这个函数生成一个新对象(object)。这时,this就指这个新对象。
function test(){
this.x = 1;
}
var o = new test();
alert(o.x); // 1
运行结果为1。为了表明这时this不是全局对象,我对代码做一些改变:
var x = 2;
function test(){
this.x = 1;
}
var o = new test();
alert(x); //2
运行结果为2,表明全局变量x的值根本没变。
情况四 apply调用
apply()是函数对象的一个方法,它的作用是改变函数的调用对象,它的第一个参数就表示改变后的调用这个函数的对象。因此,this指的就是这第一个参数。
var x = 0;
function test(){
alert(this.x);
}
var o={};
o.x = 1;
o.m = test;
o.m.apply(); //0
apply()的参数为空时,默认调用全局对象。因此,这时的运行结果为0,证明this指的是全局对象。
如果把最后一行代码修改为
o.m.apply(o); //1
运行结果就变成了1,证明了这时this代表的是对象o。
Javascript的一些经验总结的更多相关文章
- JavaScript 开发人员需要知道的简写技巧
		本文来源于多年的 JavaScript 编码技术经验,适合所有正在使用 JavaScript 编程的开发人员阅读. 本文的目的在于帮助大家更加熟练的运用 JavaScript 语言来进行开发工作. 文 ... 
- 值得认真学习的6 个 JavaScript 框架
		JavaScript JavaScript一种直译式脚本语言,是一种动态类型.弱类型.基于原型的语言,内置支持类型.它的解释器被称为JavaScript引擎,为浏览器的一部分,广泛用于客户端的脚本 ... 
- Javascript显示和隐式类型转换
		1.转换成字符串 多数的JavaScript宿主环境(比如Node.js和Chrome)都提供了全局函数toString: 与此同时Object.prototype也定义了toString方法,使得所 ... 
- 如何学好javascript
		今天逛论坛时看到有朋友问,是否有专门教Javascript的学校,这里想想把自己的一点建议和自己3年来的前端Javascript开发的经验跟大家分享下,也给出几本个人认为不错的书来做为大家学习的参考资 ... 
- JavaScript显式类型转换与隐式类型转换
		隐式类型转换 四则运算 判断语句 toString 在 JavaScript 中声明变量不需指定类型,对变量赋值也没有类型检查,同时还允许隐式类型转换. 这些特征说明 JavaScript 属于弱类型 ... 
- 各大就业网站对web前端的就业要求
		今日与女神有约,在[web前端学习部落22群]有直播公开课,喜欢的朋友赶紧加入吧!随着高等教育的普及,高校毕业生的人数每年都以极快的速度增长,数据显示,2016年,高校毕业生多达765万人,比2015 ... 
- iOS开发60分钟入门
		原文:https://github.com/qinjx/30min_guides/blob/master/ios.md 本文面向已有其它语言(如Java,C,PHP,Javascript)编程经验的i ... 
- 浅谈web前端就业的学习路线
		初级前端 主要学习三个部分:HTML,CSS,JavaScript 一.html + css部分: 这部分特别简单,到网上搜资料,书籍视频非常多.css中盒子模型,流动,block,inline,层叠 ... 
- IT 名企招聘信息
		[搜狗]网页搜索抓取与挖掘组诚聘实习生 工作方向: 抓取架构方向. 职位要求 1.对互联网和搜索引擎技术有浓厚兴趣: 2.熟练使用c .熟悉Linux开发环境.熟悉shell.python.awk等 ... 
随机推荐
- 技术胖Flutter第三季-16Stack层叠布局
			16Stack层叠布局 在上面声明一个变量Stack里面包含两个元素,第一个 是CircleAvater第二个子对象是Container 效果 把文字房子啊中下的位置: 我们需要对齐属性 包含了x轴和 ... 
- Flutter实战视频-移动电商-58.购物车_删除商品功能制作
			58.购物车_删除商品功能制作 主要做购物车后面的删除按钮 删除的方法写在provide里面 provide/cart.dart文件 传入goodsId,循环对比,找到后进行移除 //删除单个购物车商 ... 
- c#类—成员函数和封装及构造函数、析构函数、静态成员
			C# 类(Class) 当您定义一个类时,您定义了一个数据类型的蓝图.这实际上并没有定义任何的数据,但它定义了类的名称意味着什么,也就是说,类的对象由什么组成及在这个对象上可执行什么操作.对象是类的实 ... 
- throw UnsupportedOperationException
			package org.usc.action; import java.util.ArrayList; import java.util.Arrays; import java.util.List; ... 
- E20180518-hm
			priority n. 优先,优先权; (时间,序上的) 先,前; 优先考虑的事; [数] 优先次序; 
- HDU3038【种类并查集】
			题意: 给出m组区间[a,b],以及其区间的和,问有矛盾的有几组: 思路: 种类并查集. 主要是几个关系:同类元素的关系,父亲与儿子的关系,不同类元素的关系: 我们可以类似看作一个前缀和,sum[x] ... 
- Android NDK开发指南(二)Android.mk文件
			http://www.cnblogs.com/yaozhongxiao/archive/2012/03/06/2382225.html 1. 概述 Android.mk文件是用来描述build sy ... 
- iOS app支付宝接口调用的一点总结(补充支付宝SDK&Demo下载地址)
			由于app内需要用到支付功能,选择了当前最流行的支付宝进行支付.在进行内嵌支付宝功能开发时,被它狠狠的耍了一把. 根据支付宝开发文档,参考demo代码.将相关支付功能加到了自己的代码中.一些根据文档来 ... 
- Spring事件机制详解
			一.前言 说来惭愧,对应Spring事件机制之前只知道实现 ApplicationListener 接口,就可以基于Spring自带的事件做一些事情(如ContextRefreshedEvent),但 ... 
- shell学习(1)
			#!/bin/bash #######read从屏幕输入######## echo "what is your name?" read PERSON echo "Hell ... 
