codecamp

CoffeeScript作用域

先看几个例子:
age 变量在函数定义之前定义的例子
age = 99
r = -> age = 0
r()
console.log “i am #{age} years old”
#puts “i am 99 years old”


age 变量在函数定义之后定义的例子


r = -> age = 0
age = 99
r()
console.log “i am #{age} years old”
#also puts “i am 99 years old”


age 变量在函数外无定义的例子


r = -> age = 0
r()
console.log “i am #{age} years old”
#ReferenceError: age is not defined


很显然 CoffeeScript 中的变量作用域与 Javascript 中是一样的。其中有三条规则:
每个函数都是一个独立作用域,创建一个函数是创建作用域的唯一方法;(可参考 javascript 核心函数的说明)
变量的作用域仅在其被定义的范围内有效;
在变量的作用域以外,其是不可见的。
那为什么函数中直接用 age 做变量并赋值却没有改变全局变量 age (r函数之外的那个变量)的值呢?
因为 coffee 程序解析成 js 程序时,r 函数内通过 var 关键字重新定义了 age,使得 age 成为 r 函数内的一个变量,此变量的作用域仅为 r 函数内部,而不影响外部的 age 变量,以上第一个例子解析成 js 就是:


var r, age;
age = 99;
r = function() {
var age;
return age = 0;
};
r();
console.log(“i am ” + age + ” years old”);
如果将例子中 r 函数中的语句改为:
age = 99
r = -> age++
r()
console.log “i am #{age} years old”
#puts “i am 100 years old”


或者其他非赋值语句,此时的 age 则为全局中的 age 变量。



CoffeeScript编译器会考虑所有变量,保证每个变量都在词法域里适当地被定义,你永远不需要自己去写 var。那么在不同的上下文环境下出现了同名变量,CoffeeScript是怎么处理的呢,看个例子:


#编译前
outer = 1
fn = ->
 inner = -1
 outer = 10
 return null
inner = 4


//编译后
var fn, inner, outer;


outer = 1;


fn = function() {
 var inner;
 inner = -1;
 outer = 10;
 return null;
};


inner = 4;
在最外层的作用域下声明了outer、inner、fn,变量的定义都被推到相关的顶层作用域了。在fn函数内部只声明了inner,没有声明outer。实际上CoffeeScript变量定义是跟作用域和声明顺序有关的,看CoffeeScript代码fn的代码块,因为在fn里outer定义之前,函数外已经定义了outer,所以outer不在重新声明。 那为啥外面也声明了inner,fn内部又重新定义了inner,因为顺序的原因,函数fn内部的声明在前,所以编译后的代码重新定义了inner。函数外部是拿不到函数内的变量的,所以外部的inner在外围的作用域下也声明了inner。大家看下下面的代码,有助于理解:


外部先声明:


#编译前
outer = 1
inner = 4
fn = ->
 inner = -1
 outer = 10
 return null


//编译后
 var fn, inner, outer;


 outer = 1;


 inner = 4;


 fn = function() {
 inner = -1;
 outer = 10;
 return null;
};


函数内部先声明:


#编译前
 fn = ->
 inner = -1
 outer = 10
 return null
 outer = 1
 inner = 4


//编译后
 var fn, inner, outer;


 fn = function() {
 var inner, outer;
 inner = -1;
 outer = 10;
 return null;
};


 outer = 1;
 inner = 4;


CoffeeScript格式
CoffeeScript赋值
温馨提示
下载编程狮App,免费阅读超1000+编程语言教程
取消
确定
目录

关闭

MIP.setData({ 'pageTheme' : getCookie('pageTheme') || {'day':true, 'night':false}, 'pageFontSize' : getCookie('pageFontSize') || 20 }); MIP.watch('pageTheme', function(newValue){ setCookie('pageTheme', JSON.stringify(newValue)) }); MIP.watch('pageFontSize', function(newValue){ setCookie('pageFontSize', newValue) }); function setCookie(name, value){ var days = 1; var exp = new Date(); exp.setTime(exp.getTime() + days*24*60*60*1000); document.cookie = name + '=' + value + ';expires=' + exp.toUTCString(); } function getCookie(name){ var reg = new RegExp('(^| )' + name + '=([^;]*)(;|$)'); return document.cookie.match(reg) ? JSON.parse(document.cookie.match(reg)[2]) : null; }