JS面向对象之作用域
作用域
词法作用域
作用域
域表示的就是范围,即作用范围
- 就是一个名字在什么地方能使用,在什么地方不能使用
块级作用域
块级别的作用范围
// 在 c , java 等编程语言中,下面的语法报错
{
var num = 123; // 应该用int , 这里是伪代码
{
console.log(num); // 123
}
console.log(num); // 报错
}
在 js 中采取词法作用域
词法(代码)作用域,就是代码在编写过程中体现出来的作用范围,代码一旦写好,不用执行,他的作用范围就已经确定好了,这个就是所谓的词法作用域
在 js 中的词法作用域规则
1. 函数允许方位函数外的数据
2. 整个代码结构中只有函数可以限定作用域
3. 作用规则首先是提升规则分析
4. 就近原则如果当前作用规则有了名字,就不考虑外面的名字
在 js 中作用域分析方法
1. 先进行预解析,分析预解析过程
* 程序在执行过程, 会先将代码读取到内存中检查. 会将所有的声明在此时进行标记. 所谓的标记就是
让 js 解释器知道有这个名字, 后面在使用名字的时候, 不会出现未定义的错误. 这个标记过程就是提升.
* 声明
1. 名字的声明, 标识符的声明( 变量名声明 )
* 名字的声明就是让我们的解释器知道有这个名字
* 名字没有任何数据与之对应
2. 函数的声明
* 函数声明包含两部分
* 函数声明与函数表达式有区别, 函数声明是单独写在一个结构中, 不存在任何语句, 逻辑判断等结构中
* 首先函数声明告诉解释器有这个名字存在. 该阶段与名字声明一样
* 告诉解释器, 这个名字对应的函数体是什么**(函数名和函数体绑定链接)**
2. 再进行代码执行过程
常见的简单作用域问题
例子 1:
var num = 123;
function foo(){
console.log(num);
}
foo(); // 输出 123
分析
- 预解析
变量num变量名提升
函数foo函数名提升
- 代码执行
num赋值123,函数内区域为独立区域,可以使用外部数据,但是现在变量名相同,覆盖外面的值
函数foo执行
函数区域中变量num变量名提升
输出num 为undefined
num赋值456
输出num 为456
例子 2:
if(false){
var num = 123;
}
console.log(num); // 输出undefined
分析
- 预解析
变量名num提升
- 执行代码
if判断为false,不进入
输出num,num定义未赋值,为undefined
例子 3:
var num = 123;
function foo(){
var num = 456;
function fn(){
console.log(num); // 输出456
};
fn();
}
foo();
分析
- 预解析
变量名num和函数名foo声明提升,函数名foo和函数体链接
- 执行代码
num = 123, 赋值
执行函数, 函数内部声明提升,变量名num和函数名fn
变量num=456, 执行函数fn
输出num, fn函数内部没有num,向上级寻找
num=456, 输出num
例子 4:
var num = 123;
function foo1(){
var num = 456;
function foo2(){
num = 789;
function foo3(){
console.log(num); // 输出789
}
foo3();
}
foo2();
}
foo1(); // 输出456
console.log(num); // 输出123
分析
- 预解析
变量名num和函数名foo1声明提升,函数名foo1和函数体绑定
- 执行代码
num = 123, 赋值
执行函数foo1, 变量名num和函数名foo2声明提升, 函数名foo2和函数体绑定
num = 456, 赋值
执行函数foo2, num在foo2函数中没有, 访问上级foo1得到num, num = 789, 赋值函数名foo3声明提升并且绑定函数体
执行函数foo3, 输出num, foo3中没有num,访问上级,得到num = 789
输出num为789
跳出函数, 输出num为123
例子 5:
if ( ! 'a' in window ) {
var a = 123;
}
console.log( a ); // undefined
分析
- 预解析
变量a声明提升
- 执行代码
判断window中是否存在a,a存在
判断为false,不执行if中的代码
输出a,为undefined
复杂的作用域问题
例子 1
if ( true ) {
function f1 () {
console.log( 'true' );
}
} else {
function f1 () {
console.log( 'false' );
}
}
f1();
分析
新版浏览器
- 预解析
无,函数f1被浏览器认为是函数表达式,不进行变量名提升
- 执行代码
if判断进入true
执行函数表达式f1
输出结果为true
旧版浏览器
- 预解析
函数f1变量声明提升,f1函数名和最后一个函数体连接在一起
- 执行代码
if判断进入true
执行f1
输出false
例子 2
if ( false ) {
function f1 () {
console.log( 'true' );
}
} else {
function f1 () {
console.log( 'false' );
}
}
f1();
分析
新版浏览器
- 预解析
无,f1为函数表达式,没有进行声明提升
- 执行代码
if判断
执行f1
输出false
旧版浏览器
- 预解析
函数名f1,声明提升,函数名和最后一个函数体链接在一起
- 执行代码
if判断
执行f1
输出false
例子 3
var num = 123;
function f1() {
console.log( num );
}
function f2() {
var num = 456;
f1();
}
f2();
分析
- 预解析
声明提升,变量num,函数名f1,f2,函数名和函数体链接
- 执行代码
num = 123
执行函数f2
变量num赋值456
执行函数f1
输出num,函数里面没有,向上寻找,上级为(0级作用域链),num=123,输出num为123
例子 4
var num = 123;
function f1() {
console.log( num );
}
function f2() {
num = 456;
f1(num);
}
f2();
分析
- 预解析
变量名num,函数名f1,f2声明提升,函数名和函数体链接一起
- 代码执行
num=123
执行函数f2
num=456,f2中没有num,向上级寻找得到num为123,将num = 123赋值,num = 456
执行f1, f1中没有num, 向上级寻找得到num为456
输出num,num=456,输出num456
JS面向对象之作用域的更多相关文章
- 带你一分钟理解闭包--js面向对象编程
上一篇<简单粗暴地理解js原型链--js面向对象编程>没想到能攒到这么多赞,实属意外.分享是个好事情,尤其是分享自己的学习感悟.所以网上关于原型链.闭包.作用域等文章多如牛毛,很多文章写得 ...
- JS面向对象(3) -- Object类,静态属性,闭包,私有属性, call和apply的使用,继承的三种实现方法
相关链接: JS面向对象(1) -- 简介,入门,系统常用类,自定义类,constructor,typeof,instanceof,对象在内存中的表现形式 JS面向对象(2) -- this的使用,对 ...
- 原生JS面向对象思想封装轮播图组件
原生JS面向对象思想封装轮播图组件 在前端页面开发过程中,页面中的轮播图特效很常见,因此我就想封装一个自己的原生JS的轮播图组件.有了这个需求就开始着手准备了,代码当然是以简洁为目标,轮播图的各个功能 ...
- js面向对象+一般方法的选项卡
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...
- Js面向对象编程
Js面向对象编程 1. 什么是面向对象编程? 我也不说不清楚什么是面向对象,反正就那么回事吧. 编程有时候是一件很快乐的事,写一些小游戏,用编程的方式玩游戏等等 2. Js如何定义一个 ...
- 拖拽系列二、利用JS面向对象OOP思想实现拖拽封装
接着上一篇拖拽系列一.JavaScript实现简单的拖拽效果这一篇博客将接着对上一节实现代码利用JS面向对象(OOP)思维对上一节代码进行封装; 使其模块化.避免全局函数污染.方便后期维护和调用:写到 ...
- JS面向对象笔记二
菜单导航,<JS面向对象笔记一>, 参考书籍:阮一峰之<JavaScript标准参考教程> 一.构造函数和new命令 二.this关键字 三.构造函数和new命令 四.构造函 ...
- 【JavaScript】 JS面向对象的模式与实践 (重点整治原型这个熊孩子 (/= _ =)/~┴┴ )
参考书籍 <JavaScript高级语言程序设计>—— Nicholas C.Zakas <你不知道的JavaScript> —— KYLE SIMPSON 在JS的面向 ...
- javascript(面向对象,作用域,闭包,设计模式等)
javascript(面向对象,作用域,闭包,设计模式等) 1. 常用js类定义的方法有哪些? 参考答案:主要有构造函数原型和对象创建两种方法.原型法是通用老方法,对象创建是ES5推荐使用的方法.目前 ...
随机推荐
- spring+jersey+c3p0构建restful webservice(数据源采用c3p0)
项目下载地址: http://files.cnblogs.com/files/walk-the-Line/payment.zip
- uva 11427
题目大意:每天晚上你都玩纸牌,如果第一次赢了就高高兴兴地去睡觉:如果输了就接着玩,假设每盘游戏获胜的的概率都是p,且各盘游戏相互独立.你是一个固执的完美主义者,因此会一直玩到当晚获胜局数的比例严格大于 ...
- [洛谷P1040] 加分二叉树
洛谷题目链接:加分二叉树 题目描述 设一个n个节点的二叉树tree的中序遍历为(1,2,3,-,n),其中数字1,2,3,-,n为节点编号.每个节点都有一个分数(均为正整数),记第i个节点的分数为di ...
- CentOS 7 安装python3.6.1
说明:CentOS 7最小化安装,默认Python 2.7 ,开发需要安装python3.6.1 . 1.检查python版本 [root@bogon ~]# python -VPython 2.7. ...
- CTL_CODE说明
DeviceIoControl函数的第二个参数IoControlCode就是由CTL_CODE宏定义的,下边我们可以了解一下CTL_CODE的内容. CTL_CODE:用于创建一个唯一的32位系统I/ ...
- efi转bios详细说明
前言 制作好的efi格式的ubuntu15.10系统放到服务器主板上启动不了,于是将其改为bios格式,发现问题解决了,成功登入系统.下面是操作过程的一个记录. 测试环境 目标环境 系统: Ubunt ...
- 基于vlc sdk的二次开发--环境搭建、编译
前言 关于 搭建.编译VLC,不同的平台有不同的方法,可以参考wiki. 其中在windows下编译VLC有两种方式,MSYS+MinGW和CygWin.通过测试,最后决定采用MSYS+MinGW搭建 ...
- python学习笔记 协程
在学习异步IO模型前,先来了解协程 协程又叫做微线程,Coroutine 子程序或者成为函数,在所有语言中都是层级调用,比如a调用b,b调用c.c执行完毕返回,b执行完毕返回,最后a执行完毕返回 所以 ...
- python unittest 快速入门
import unittest def add(x, y): return x + y class TestLearning(unittest.TestCase): def setUp(self): ...
- mysql的mysqladmin的用法
mysqladmin 适合于linux和windows系统 linux下:mysqladmin -u[username] -p[password] status windows下:先在安装目录找到my ...