Macji Pro2010

记录一些应该记录的东东

JavaScript中的作用域与闭包

March 19, 2008 - by Macji - HTML/CSS/JS/PHP - javascript, closure

Mike West说:

Scope is one of the foundational aspects of the JavaScript language, and probably the one I’ve struggled with the most when building complex programs. I can’t count the number of times I’ve lost track of what the this keyword refers to after passing control around from function to function, and I’ve often found myself contorting my code in all sorts of confusing ways, trying to retain some semblance of sanity in my understanding of which variables were accessible where.

要学会JavaScript,必须得先去理解它的作用域是怎样的。


没有块级作用域


JavaScript变量的作用域只有全局和局部作用域。全局作用域是定义在JavaScript中(非函数体内),可以在任何地方改变它的值,这是非常危险的。局部作用域是定义在函数体内的,它的作用域是该函数的范围。外界不能访问他,只允许函数内部访问。JavaScript的变量没有块级作用域。

var g = "G";

function F() {
    alert(g); //显示G
    if ( true ) {
        var num = 10;
    }
    alert(num); //显示10
}

F();
alert(num + "1");//显示错误,num没有定义

我们再看一个例子

var p = "G";
function F2() {
    alert(p);//undefined
    var p = "L";
    alert(p);//L
}
F2();

为什么会这样呢?我们看一张图(画不好图-_-!)。这就它查找变量的方式。



词法作用域


JavaScript函数的作用域是通过词法来划分作用域。也就是说函数是定义在函数时的那个作用域里运行的。每定义一个函数,那么JavaScript解析器就保存了该函数的作用域链了。执行该函数时,它就跑到定义那函数的地方去运行,也可以理解成引用。我们先看一个例子。

function F4() {
    var a = 10;
    function F5() {
        return a;
    }
    alert(F5());//10
}
F4();

当运行F4()时,它跑到function F4的地方,然后一行一行执行下来,定义a = 10;定一个function F5;运行alert(F5()),它先在F5函数体内找a的值,找不到,再向上一层,在F4体内查找a,找到了,那么alert 10出来(如果没有找到,那么再向上,找到全局时,还没有找到,那么就是没有定义)。


垃圾回收机制与闭包


JavaScript采用自动回收,也就是当一个对象没有对其他引用的时候,那么销毁它。这很好理解,那闭包很难用专业词汇解释。其实也很容易理解。函数一 被 A 引用,函数一 体内的 函数二 又被 B 引用,那么当A引用完毕后,函数一也该销毁了,但不是这样的,它体内的函数二被B引用着,所以函数一没有被销毁,这些综合体我们叫他闭包。(解释的不好,建议找些专业的解释理解一下。)这是一种很强大的技术,理解了作用域和垃圾回收机制,这也很好理解了。在浏览器里,所有用户定义的函数都是闭包,但像上述中的函数二被第三者引用,这种闭包才有意思。


一个闭包的例子



Realazy翻译的一篇关于作用域的文章

1 条评论»

仿flash动画导航(html+css+js)

February 1, 2008 - by Macji - HTML/CSS/JS/PHP - xhtml, javascript, animation

废话不多说,先给出CSS

body{margin:100px 0}
.clear:after{content:"."; display:block; height:0; clear:both; visibility:hidden}.clear{display:inline-block}.clear{display:block}
div#nav{height:70px; background:url(img/nav_bg.png) repeat-x}
div#nav ul{width:960px; margin:0 auto; list-style:none}
div#nav ul li{float:left; height:35px; overflow:hidden; padding:0 2px 0 0; font: bold 12px/35px Arial; background:url(img/nav_right.png) repeat-y right 0}
div#nav ul li a{float:left; height:100%; padding:0 20px; background:url(img/nav_sub.png) repeat-x; color:#fff; text-decoration:none}
div#nav ul li a.hover{clear:both; background-position:0 -35px}
div#nav ul li.on a{background-position:0 -35px}
div#nav ul li.nobg{background:none}

html


<div id="nav">
<ul class="clear">
<li class="on"><a href="javascript:;" mce_href="javascript:;">首页</a></li>
<li><a href="javascript:;" mce_href="javascript:;">麦鸡</a></li>

<li><a href="javascript:;" mce_href="javascript:;">麦鸡</a></li>
<li><a href="javascript:;" mce_href="javascript:;">麦鸡</a></li>
<li><a href="javascript:;" mce_href="javascript:;">MacJi</a></li>

<li><a href="javascript:;" mce_href="javascript:;">麦鸡</a></li>
<li><a href="javascript:;" mce_href="javascript:;">麦鸡</a></li>
<li class="nobg"><a href="http://www.macji.com" mce_href="http://www.macji.com">麦鸡的博客</a></li>

</ul>
</div>

JavaScript

function nav(c, config){
  this.config = config || {speed: 10, num: 2};
  this.container = (typeof(c)=="object") ? c : document.getElementById(c);
  this.lineHeight = this.container.offsetHeight;
  this.scrollTimeId = null;
  var _this = this;  

  this.__construct = function (){
    var inner,el,href;
    inner = _this.container.childNodes[0].innerHTML;
    href = _this.container.childNodes[0].href;
    el = document.createElement("a");
    el.innerHTML = inner;
    el.href = href;
    el.className = 'hover';
    _this.container.appendChild(el);  

    //注册事件
    _this.container.onmouseover = function (){_this.start()};
    _this.container.onmouseout  = function (){_this.end()};
  }();  

  this.start = function (){
    _this.clear();
    _this.scrollTimeId = setTimeout(function(){_this.scrollUp();}, _this.config.speed);
  };  

  this.end = function (){
    _this.clear();
    _this.scrollTimeId = setTimeout(function(){_this.scrollDown();}, _this.config.speed);
  };  

  this.scrollUp = function (){
    var c = _this.container;  

    if(c.scrollTop >= _this.lineHeight){c.scrollTop = _this.lineHeight;return;}
    c.scrollTop += _this.config.num;
    _this.scrollTimeId = setTimeout(function(){_this.scrollUp();}, _this.config.speed);
  };  

  this.scrollDown = function (){
    var c = _this.container;  

    if(c.scrollTop <= 0){c.scrollTop = 0;return;}
    c.scrollTop -= _this.config.num;
    _this.scrollTimeId = setTimeout(function(){_this.scrollDown();}, _this.config.speed);
  };  

  this.clear = function (){clearTimeout(_this.scrollTimeId)};
}

调用方法

(function(){
  var container = document.getElementById('nav');
  var el_li = container.getElementsByTagName('li');
  var arr = [];
  for( var i = 0; i < el_li.length; i++){
    //如果不是当前页面(className  == 'on')就实例化
    if(el_li[i].className == 'on') continue;
    arr[i] = new nav(el_li[i], {speed: 10, num: 4});
  }
})();

查看效果

2 条评论»

用javascript来实现动画导航

November 27, 2007 - by Macji - HTML/CSS/JS/PHP - javascript, animation

google是个大公司,全世界都有google的脚印,韩国的google动画效果非常不错,蓝色理想论坛里已经有人挖过来了,可惜js写的太多了,那自己写一个吧?好,就这么干!


原理


小时候,总喜欢看动画片吧,动画片是怎样实现的呢?记得妈妈说是一张画一张画切换过去(啊?那一部葫芦兄弟要画多少副画啊? -_-! ),其实我们现在做的也是这样,用一个图片,这个图片里有很多个小图,来显示动画轨迹.按时间来移动图片,那图片不是会动了啊?(不知道,表达清楚了没…语文很重要啊!!)


准备


我们需要一张图片,一个大脑,一张会笑的脸(不笑效果就出不来了….)!!下面是我准备的图片(ps水平有限^_^)…

代码


我们看到上面的图片,想象下,它动起来是多么的优美啊…


css

.Gnb_btn_div{
	width:90px;
	height:75px;
	overflow:hidden;
	display:block;
	position:absolute;
}     

.Gnb_btn_img{
	width:100%;
	height:525px;
	display:block;
	overflow:hidden;
	text-indent:-500px;
}
#gnb_btn_01 .Gnb_btn_img {
	background-image:url(http://www.wler.cn/blog/img/friend.gif)
}

javascript

<script type="text/javascript">
// <![CDATA[
function GNB(_7c){
	//初始化一些参数
	this.iImgNum=7;			//小图片个数
	this.iImgHeight=75;		//小图片高度
	this.iOverSpeed=50;		//鼠标经过时候切换的时间
	this.iOutSpeed=50;		//鼠标离开时候切换的时间
	this.eventObj=_7c;		//取得图片对象     

	this.MouseOverFlag=false;
	this.imageIndex=0;
	if(this.eventObj==null){return;}
	this.eventObj.parentClass=this;this.eventAssign();
}     

GNB.prototype.eventAssign=function(){//注册事件
	this.eventObj.onmouseover=this.menuMouseOver;
	this.eventObj.onmouseout=this.menuMouseOut;
};     

GNB.prototype.menuMouseOver=function(){//鼠标经过
	if(this.parentClass.MouseOverFlag!=false){return;}
	this.parentClass.MouseOverFlag=true;
	this.parentClass.clearTimeOut();
	this.parentClass.menuMouseOverTimer();
};     

GNB.prototype.menuMouseOut=function(){//鼠标离开
	this.parentClass.MouseOverFlag=false;
	this.parentClass.clearTimeOut();
	this.parentClass.menuMouseOutTimer();
};     

GNB.prototype.menuMouseOverTimer=function(){//经过图片位置递增
	var _7d=this;
	if(this.imageIndex>=this.iImgNum){return;}
	this.eventObj.scrollTop=this.imageIndex*this.iImgHeight;
	this.imageIndex++;
	this.setTimerID=setTimeout(function(){_7d.menuMouseOverTimer();},this.iOverSpeed);
};     

GNB.prototype.menuMouseOutTimer=function(){////经过图片位置递减
	var _7e=this;if(this.imageIndex<0){return;}
	this.eventObj.scrollTop=this.imageIndex*this.iImgHeight;
	this.imageIndex--;
	this.setTimerID=setTimeout(function(){_7e.menuMouseOutTimer();},this.iOutSpeed);
};     

GNB.prototype.clearTimeOut=function(){//取消定时
	clearTimeout(this.setTimerID);
};
// ]]>
</script>

xhtml

<div class="Gnb_btn_div" id="gnb_btn_01">

<a class="Gnb_btn_img" href="#1" mce_href="#1">找朋友</a>
</div>     

<script type="text/javascript">
// <![CDATA[
var GNB1=new GNB(document.getElementById("gnb_btn_01"));//实例单个按钮,当然也可以多个
// ]]>
</script>

演示

1 条评论»