Prototype源码浅析 String部分(二)


Posted in Javascript onJanuary 16, 2012
格式 camelize | capitalize |  underscore |  dasherize  | inspect          
变形 toArray |  succ  | times
这里面一个有用的方法是inspect,按照参考手册的说明,他的作用是“返回该字符串针对调试的字符串表现形式(即用单引号或双引号包括起来,并使用 '\' 对特殊字符进行转义)”,在Object的toJSON里面也涉及到这个方法。

既然涉及到需要转义的字符,我们自然要一份转义字符信息,下面直接给出:

String.specialChar = { 
'\b': '\\b', 
'\t': '\\t', 
'\n': '\\n', 
'\f': '\\f', 
'\r': '\\r', 
'\\': '\\\\' 
}

【在JSON.js里面,多了一个'"',因为JSON里面string里面是不能出现"的,所以需要转义】

第一步,当然是要替换特殊的转义字符,初始版本:

function inspect() { 
return this.replace(/[\b\t\n\f\r\\]/,function(a){ 
return String.specialChar[a]; 
}); 
}

对于JSON形式来说,双引号是必须的,因此,我们应该可以选择自己的返回形式,所以,给inspect一个参数useDoubleQuotes,默认是用单引号返回字符串的。

function inspect(useDoubleQuotes) { 
var escapedString = this.replace(/[\b\t\n\f\r\\]/,function(a){ 
return String.specialChar[a]; 
}); 
if (useDoubleQuotes){ 
return '"' + escapedString.replace(/"/g, '\\"') + '"'; 
} 
return "'" + escapedString.replace(/'/g, '\\\'') + "'"; 
}

现在这跟源码中的功能差不多,不过Prototype源码中的实现方式并不是这样的,主要区别在于escapedString这一段。源码中直接列出来所有的控制字符,表示为[\x00-\x1f],外加'\'就是[\x00-\x1f\\],因此改造上面的初始版本就是:

function inspect(useDoubleQuotes) { 
var escapedString = this.replace(/[\x00-\x1f\\]/g, function(character) { 
if (character in String.specialChar) { 
return String.specialChar[character]; 
} 
return character ; 
}); 
if (useDoubleQuotes) return '"' + escapedString.replace(/"/g, '\\"') + '"'; 
return "'" + escapedString.replace(/'/g, '\\\'') + "'"; 
} 
[html] 
附,ASCII控制字符编码表,对应\x00-\x1f: 


如果发现除了String.specialChar中的字符,还有其他的控制字符,源码中也有一步处理,就是将控制字符转变为unicode的表示形式,因为这个方法本身就是要获得字符串的形式。 
比如垂直制表符'\v'。'\v'.inspect() -->'\u000b' 
完整版本: 
[code] 
function inspect(useDoubleQuotes) { 
var escapedString = this.replace(/[\x00-\x1f\\]/g, function(character) { 
if (character in String.specialChar) { 
return String.specialChar[character]; 
} 
return '\\u00' + character.charCodeAt().toPaddedString(2, 16); 
}); 
if (useDoubleQuotes) return '"' + escapedString.replace(/"/g, '\\"') + '"'; 
return "'" + escapedString.replace(/'/g, '\\\'') + "'"; 
}

其中toPaddedString(length[, radix])将当前 Number 对象转换为字符串,如果转换后的字符串长度小于 length 指定的值,则用 0 在左边补足其余的位数。可选的参数 radix 用于指定转换时所使用的进制。这是Prototype中Number的一个扩展,暂时知道即可。

因此'\v'.charCodeAt().toPaddedString(2, 16)就是将'\v'的字符编码转换成16进制的两位编码符[操作字符不会范围有限制,因此不会超出],最后冠以'\u00'开头即可。

方法说明:
toArray:将字符串拆分为字符数组。
succ:根据 Unicode 字母表转换字符串最后的字符为后续的字符
times:将字符串重复。

对应具体的实现也很简单,String部分的重要之处在于后面的脚本,JSON和替换处理,其他都是增强性质的。

function toArray() { 
return this.split(''); 
}

其中split('')就将字符串打撒为单个字符,并以数组形式返回,如果还要再增强,可以给一个参数给toArray来指定分隔符。

function toArray(pattern) { 
return this.split(pattern); 
} 
console.log(toArray.call('my name is xesam',' '));//["my", "name", "is", "xesam"]

就是对split的使用而已,不过源码中并没有这么做,因为并没有这个必要。
function succ() { 
return this.slice(0, this.length - 1) + String.fromCharCode(this.charCodeAt(this.length - 1) + 1); 
}

这里主要的就是fromCharCode和charCodeAt方法的使用。从代码中也可以看出,两者的明显区别是fromCharCode是String的静态方法,而charCodeAt是字符串的方法(挂在String.prototype上面)。然后两者的作用正好相反,下面是http://www.w3school.com.cn给出的解释:

fromCharCode() 可接受一个指定的 Unicode 值,然后返回一个字符串。

charCodeAt() 方法可返回指定位置的字符的 Unicode 编码。这个返回值是 0 - 65535 之间的整数。

具体到succ,以字符串‘hello xesam'为例,先获取除结尾字符外的所有字符‘hello xesa',然后加上Unicode表中‘m'后面的一个字符‘n',因此结果就是‘hello xesan'
以此为基础,我们要打印从‘a'到‘z'的所有字母,可以用以下的函数:

function printChar(start,end){ 
var s = (start + '').charCodeAt() 
var e = (end + '').charCodeAt(); 
if(s > e){ 
s = [e,e=s][0]; 
} 
for(var i = s ;i <= e; i++){ 
console.log(String.fromCharCode(i)); 
} 
} 
printChar('a','z');

function times(count) { 
return count < 1 ? '' : new Array(count + 1).join(this); 
}

times作用是重复整个字符串,其主要思想就是将当前字符作为数组的连接符调用join来获得预期结果。当然用循环添加也可以,不过没这么简洁。
如果要重复字符串里面的每个字符,可以用相同的思想:

String.prototype.letterTimes = function(count){ 
var arr = []; 
arr.length = count + 1; 
return this.replace(/\w/g,function(a){ 
return arr.join(a); 
}) 
} 
console.log('xesam'.letterTimes(3));//xxxeeesssaaammm

camelize | capitalize | underscore | dasherize这四个主要是关于变量名转换的。
camelize : 将一个用横线分隔的字符串转换为 Camel 形式
capitalize :将一个字符串的首字母转换为大写,其它的字母全部转为小写。
underscore :将一个 Camel 形式的字符串转换为以下划线("_")分隔的一系列单词。
dasherize :将字符串中的下划线全部替换为横线("_" 替换为 "-")。

最明显的,可以用在CSS属性与DOM的style属性的相互转换中【class与float不属于此范畴】。对应到上面的方法中,将CSS属性转换为对应的DOM的style属性可以使用camelize 方法,但是反过来却没有这个方法,因此必须连续调用underscore -> dasherize 方法才行。

function camelize() { 
return this.replace(/-+(.)?/g, function(match, chr) { 
return chr ? chr.toUpperCase() : ''; 
}); 
}

核心是replace方法的使用,其他挺简单,参见《浅析字符串的replace方法应用》

function capitalize() { 
return this.charAt(0).toUpperCase() + this.substring(1).toLowerCase(); 
}

这里注意charAt(charAt() 方法可返回指定位置的字符。)与charCodeAt的区别就可以了。

function underscore() { 
return this.replace(/::/g, '/') 
.replace(/([A-Z]+)([A-Z][a-z])/g, '$1_$2') 
.replace(/([a-z\d])([A-Z])/g, '$1_$2') 
.replace(/-/g, '_') 
.toLowerCase(); 
}

实例来说明步骤:
'helloWorld::ABCDefg'.underscore() 
//'helloWorld::ABCDefg' 
.replace(/::/g, '/') //'helloWorld/ABCDefg' 
.replace(/([A-Z]+)([A-Z][a-z])/g, '$1_$2')//helloWorld/ABC_Defg 
.replace(/([a-z\d])([A-Z])/g, '$1_$2') //hello_World/ABC_Defg 
.replace(/-/g, '_') //hello_World/ABC_Defg 
.toLowerCase(); //hello_world/abc_defg

这个方法只适合Camel 形式的,就是得有‘峰'。
function dasherize() { 
return this.replace(/_/g, '-'); 
}

这个就是单纯的字符替换而已。
来自小西山子
Javascript 相关文章推荐
json属性名为什么要双引号(个人猜测)
Jul 31 Javascript
Jquery代码实现图片轮播效果(一)
Aug 12 Javascript
JS实现的通用表单验证插件完整实例
Aug 20 Javascript
理解JavaScript中worker事件api
Dec 25 Javascript
jqPlot jQuery绘图插件的使用
Jun 18 Javascript
easyui messager alert 三秒后自动关闭提示的实例
Nov 07 Javascript
开源免费天气预报接口API及全国所有地区代码(国家气象局提供)
Dec 26 Javascript
vue如何自动化打包测试环境和正式环境的dist/test文件
Jun 06 Javascript
vue移动端实现手机左右滑动入场动画
Jun 17 Javascript
JS实现网站吸顶条
Jan 08 Javascript
Node.JS获取GET,POST数据之queryString模块使用方法详解
Feb 06 Javascript
Vue 解决在element中使用$notify在提示信息中换行问题
Nov 11 Javascript
深入理解JavaScript系列(11) 执行上下文(Execution Contexts)
Jan 15 #Javascript
深入理解JavaScript系列(10) JavaScript核心(晋级高手必读篇)
Jan 15 #Javascript
深入理解JavaScript系列(9) 根本没有“JSON对象”这回事!
Jan 15 #Javascript
深入理解JavaScript系列(8) S.O.L.I.D五大原则之里氏替换原则LSP
Jan 15 #Javascript
深入理解JavaScript系列(7) S.O.L.I.D五大原则之开闭原则OCP
Jan 15 #Javascript
深入理解JavaScript系列(6):S.O.L.I.D五大原则之单一职责SRP
Jan 15 #Javascript
深入理解JavaScript系列(6) 强大的原型和原型链
Jan 15 #Javascript
You might like
在Nginx上部署ThinkPHP项目教程
2015/02/02 PHP
laravel开发环境homestead搭建过程详解
2020/07/03 PHP
javascript与cookie 的问题详解
2013/11/11 Javascript
jQuery获取Radio,CheckBox选择的Value值(示例代码)
2013/12/12 Javascript
详解JavaScript中的4种类型识别方法
2015/09/14 Javascript
js判断手机浏览器操作系统和微信浏览器的方法
2016/04/30 Javascript
客户端验证用户名和密码的方法详解
2016/06/16 Javascript
jQuery dataTables与jQuery UI 对话框dialog的使用教程
2016/09/02 Javascript
angular ngClick阻止冒泡使用默认行为的方法
2016/11/03 Javascript
面包屑导航详解
2017/12/07 Javascript
微信小程序实现红包雨功能
2018/07/11 Javascript
如何将百度地图包装成Vue的组件的方法步骤
2019/02/12 Javascript
JS实现的冒泡排序,快速排序,插入排序算法示例
2019/03/02 Javascript
Vue实现计算器计算效果
2020/08/17 Javascript
[04:45]DOTA2-DPC中国联赛正赛 iG vs LBZS 赛后选手采访
2021/03/11 DOTA
python实现堆栈与队列的方法
2015/01/15 Python
Python文件与文件夹常见基本操作总结
2016/09/19 Python
Python科学计算之Pandas详解
2017/01/15 Python
Python实现七彩蟒蛇绘制实例代码
2018/01/16 Python
python学生信息管理系统(初级版)
2018/10/17 Python
Python JSON格式数据的提取和保存的实现
2019/03/22 Python
python日志logging模块使用方法分析
2019/05/23 Python
python用quad、dblquad实现一维二维积分的实例详解
2019/11/20 Python
python数据库开发之MongoDB安装及Python3操作MongoDB数据库详细方法与实例
2020/03/18 Python
HTML5仿微信聊天界面、微信朋友圈实例代码
2018/01/29 HTML / CSS
Miller Harris官网:英国小众香水品牌
2020/09/24 全球购物
杭州联环马网络笔试题面试题
2013/08/04 面试题
秘书专业自荐信范文
2013/12/26 职场文书
物流业务员岗位职责
2014/02/08 职场文书
高中运动会入场词
2014/02/14 职场文书
人事文员岗位职责
2015/02/04 职场文书
虎兄虎弟观后感
2015/06/12 职场文书
家庭贫困证明
2015/06/16 职场文书
新手开公司创业注意事项有哪些?
2019/07/29 职场文书
FFmpeg视频处理入门教程(新手必看)
2022/01/22 杂记
Windows和Linux上部署Golang并运行程序
2022/04/22 Servers