ES5允许直接使用保留字作为属性名,但却不允许直接使用保留字作为函数名
设现有类NSMap,若要给NSMap的原型加delete方法,如
function NSMap(){
}
NSMap.prototype.delete=function delete(){
};
则浏览器解析报错
SyntaxError: Unexpected token delete
那么,为什么native code的Map可以办到?
后来想到标识符可以由除ASCII特殊字符以外的大部分Unicode字符组成,方案来了:
//1.从保留字中随便挑个字符出来,如字符t,算出字符t的十六进制charCode,
"t".charCodeAt(0).toString(16)//"74"
//2.尝试将t用\x74表示,使用dele\x74e作为函数名,不行
NSMap.prototype.delete=function dele\x74e(){
};
//3.尝试将t用\u0074表示,使用dele\u0074e作为函数名,总算不报错了
NSMap.prototype.delete=function dele\u0074e(){
};
//(这里有点不解,\x74和\u0074不是同一个意思吗,有可能是因解析器的性能考虑而不支持\x)
//4.如果函数是独立声明的,引用函数也不得直接使用字面保留字
function dele\u0074e(){
}
NSMap.prototype.delete=dele\u0074e;
//用得较多时,可以参照Chrome底层JavaScript源码那样写
InstallFunctions(NSMap.prototype,DONT_ENUM,[
"extends",function ex\u0074ends(){
},
"delete",function dele\u0074e(){
}
]);
另附上一个转义函数
function toCharCodeString(){
return Array.prototype.map.call(new String(this),function(c){
var code=c.charCodeAt(0),
hex=code.toString(16);
//return code>0xff?
// "\\u"+"000".substr(0,4-hex.length)+hex:
// "\\x"+"0".substr(0,2-hex.length)+hex;
return "\\u"+"000".substr(0,4-hex.length)+hex;
}).join("");
}
toCharCodeString.call("delete"); // "\u0064\u0065\u006c\u0065\u0074\u0065"
toCharCodeString.call("Unicode字符"); // "\u0055\u006e\u0069\u0063\u006f\u0064\u0065\u5b57\u7b26"