KiMoGiGi 技术文集

不在乎选择什么,而在乎坚持多久……

IT博客 首页 联系 聚合 管理
  185 Posts :: 14 Stories :: 48 Comments :: 0 Trackbacks
我们知道JScript的对象(不只是Object)可以像html元素对象一样,添加任意的自定义属性值。也就是说JScript的对象,天生都是Key/Value map。并且这种map结构由于使用Native的代码实现,检索效率非常之高,我曾在这篇文章里讨论过。不过在遍历这样的map时,expando导入属性和prototype导入属性存在了混淆。

    由于JScript的prototype特性对对象的扩充非常的方便,所以我们在制作一些JScript类库的时候,一般都会使用prototype特性为对象添加方法,比如我们对Object进行如下prototype扩充:
Object.prototype.Clone = function() {};
Object.prototype.Call 
= function() {};
Object.prototype.OtherMethod 
= function(){};

    这个时候如果再使用Object作为map结构来使用,我们就会遇到遍历这个map的错误,看下面的代码:
var objMap = {};
objMap['abc'] 
= '1.abc';
objMap['def'] 
= '2.def';
objMap['ghi'] 
= '3.ghi';
objMap['jkl'] 
= '4.jkl';    

    遍历这个集合:
function DisplayMap(map)
{
    
var values = [];
    
for ( var key in map )
    {
        values.push(map[key]);
    }
    
return values;
}
Display(objMap);

    我们发现,在values里的值居然是:function(){},function() {},function() {},1.abc,2.def,3.ghi,4.jkl。真是郁闷! 其实这就是 for in 语句的效果,JScript也就是这么设计的,我们没有办法改变。那么我门能不能只取出objMap中expando进去的属性呢?

    由于prototype属性的优先级很高,在对象实例生成的时候就expand到对象实例中去了。所以我们建立的任何一个对象,都会包含相同的prototype属性。这样一来就好办了,我们把objMap中的prototype属性找出来过滤掉就行了呗。参考代码如下:
function GetExpandoValues(map)
{
    
var values = [];
    
var obj = new map.constructor();
    
for ( var key in map )
    {
        
if ( obj[key] !== map[key] )
        {
            values.push(map[key]);
        }
    }
    
return values;
}
GetExpandoValues(objMap);

    获得结果为:1.abc,2.def,3.ghi,4.jkl

    
posted on 2006-12-03 01:15 KiMoGiGi 阅读(300) 评论(0)  编辑 收藏 引用 所属分类: JavaScript
只有注册用户登录后才能发表评论。