设为首页 收藏本站
查看: 590|回复: 0

[经验分享] 比较javascript学python-2 闭包实现

[复制链接]

尚未签到

发表于 2017-4-29 14:14:27 | 显示全部楼层 |阅读模式
  python源码剖析是本好书,看了能睡不着觉的,可惜为何没有javascript源码剖析,c系能手啥时能来研究v8?

  


  承接 比较javascript学 python-1 对象与类
,先给出两个语言使用闭包的例子:

  代码举例:


  javascript:



function outer(){
var value="inner";
function inner(){
alert(value);
}
return inner;
}
var c=outer();
c();
   

  python:



def outer():
value="inner"
def inner():
print value
return inner
c=outer()
c()
 
   实现:


  


  javascript:


  


  根据 ecmascript5 10.3 章 (链接待引):函数运行时会产生 execution context,而execution context中包含 LexicalEnvironment (Identifies the Lexical Environment used to resolve identifier references made by code within this execution context.) 为链状结构,用来解析运行时函数内的变量引用,上例图解:

  


DSC0000.jpg

  


  可见,嵌套函数内的变量访问复杂度是O(函数的嵌套层次), Nicholas C. Zakas专门提到了这个问题(链接待引),当需要多次访问某个外部作用域变量时,要先copy一个到本函数局部变量中来。


  对应上例的小题大做改动:


function outer(){
var value="inner";
function inner(){
var innervalue=value;
//多次访问
alert(innervalue);
alert(innervalue);
alert(innervalue);
}
return inner;
}
var c=outer();
c();
  PS:对于一些注重性能的javascript引擎譬如webkit,会索引变量,避免传统的作用域链查询,使得变量在作用域链中的深度于chrome,safari中变得无关紧要。(IE还是很紧要的:))

  来源 High Performance Javascript:

DSC0001.png

  python:


  


  根据 python源码剖析(链接待引)
,函数运行时会产生 PyFrameObject (等同ecmascript规范描述的 execution context),而闭包涉及外层作用域变量直接被(指针引用?)放到了内层函数的 PyFrameObject 中的自由变量部分tuple结构中,解析时直接访问本PyFrameObject的tuple结构即可。

DSC0002.jpg

  


  可见,嵌套函数内的变量访问复杂度是O(1),但是python由于没有变量声明语句存在以下问题:

  python:


  (由于没有申明变量关键字!造成不能修改上层作用域的变量,只能引用,直接赋值同名变量就算覆盖了)


def outerFunc():
a=1;
def innerFunc():
a=2
innerFunc()
print a

outerFunc()
  
 
而只能变通一下:



def outerFunc():
a=[1];
def innerFunc():
a[0]=2
innerFunc()   
print a
outerFunc()
   Javascript:



(function(){
var a=1;
(function(){
a=2;
})();
alert(a);
})();
  updated: 2011-05-25
  从黑客与画家一书中看到这也就是经典的累加器例子:

  javascript 可以:

function foo(n){
return function(i){
return n+=i;
};
}
   而 python 则必须为:

def foo(n):
s=[n]
def bar(i) :
s[0]+=i
return s[0]
return bar
 
   python 也可以绕过用对象属性来代替原来的词法作用域的变量:

def foo(n):
class acc:
//构造器
def __init__(self,s)
self.s=s
def inc(self,i):
self.s+=i
return self.s
//词法作用域变量放到对象属性内维护
return acc(n).inc
  或override 隐藏的 __call__,直接封存变量到对象属性

class foo:
def __init__(self,n):
self.n=n
//作为函数调用时
def __call__(self,i)
self.n+=i
return self.n
inc=foo(5)
inc(1) //=>6
inc(2) //=>8
   但这只能解决一层词法作用域问题
  java 就更不行,只有个近似实现
  可以在函数中搞个匿名接口子类的对象,而这个匿名接口子类可访问函数内 final 的变量:

public interface inc{
public int call(int i);
}
public static inc foo(final int n){
return new inc(){
int s=n;
public int call(int i){
return s+=i;
}
}
}
 

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-370866-1-1.html 上篇帖子: python学习——一些有趣的讨论 下篇帖子: 《征服Python》笔记--前8章
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表