Prototype String对象 学习


Posted in Javascript onJuly 19, 2009
//String对象的静态方法 
Object.extend(String, { 
interpret: function(value) { 
return value == null ? '' : String(value); 
}, 
specialChar: { 
'\b': '\\b', 
'\t': '\\t', 
'\n': '\\n', 
'\f': '\\f', 
'\r': '\\r', 
'\\': '\\\\' 
} 
}); Object.extend(String.prototype, (function() { 
//内部方法,为gsub和sub函数初始化replacement参数 
function prepareReplacement(replacement) { 
if (Object.isFunction(replacement)) return replacement; 
var template = new Template(replacement); 
return function(match) { return template.evaluate(match) }; 
} 
//用replacement替换所有符合pattern的字符串 
//注意当replacement不为函数时,这里面会有一个递归调用,其中参数pattern会变为Template.pattern 
//可以参考Template的evaluate方法,这里构造的非常巧妙,在while循环里的else部分会处理这个递归调用的结果 
function gsub(pattern, replacement) { 
var result = '', source = this, match; 
replacement = prepareReplacement(replacement); 
if (Object.isString(pattern)) 
pattern = RegExp.escape(pattern); 
//如果pattern参数为null或者'',用把整个字符串按照单个字符分割,并且在每个字符的前后添加replacement 
if (!(pattern.length || pattern.source)) { 
replacement = replacement(''); 
return replacement + source.split('').join(replacement) + replacement; 
} 
while (source.length > 0) { 
     //如果source匹配pattern 
if (match = source.match(pattern)) { 
         //取出匹配之前的字符串 
result += source.slice(0, match.index); 
        //匹配替换 
result += String.interpret(replacement(match)); 
        //把匹配字符之后的部分赋给source,进行下一次匹配 
source = source.slice(match.index + match[0].length); 
} else { //如果source不匹配pattern,则直接把source添加到结果上,并把source置空,结束循环 
result += source, source = ''; 
} 
} 
return result; 
} 
//基本意思和gsub一样,只不过多一个count参数,表示要替换几次 
function sub(pattern, replacement, count) { 
replacement = prepareReplacement(replacement); 
count = Object.isUndefined(count) ? 1 : count; 
return this.gsub(pattern, function(match) { 
if (--count < 0) return match[0]; 
return replacement(match); 
}); 
} 
//对符合pattern的字符串进行iterator调用 
function scan(pattern, iterator) { 
this.gsub(pattern, iterator); 
return String(this); 
} 
//按照给定长度截断字符串 
function truncate(length, truncation) { 
length = length || 30; 
truncation = Object.isUndefined(truncation) ? '...' : truncation; 
return this.length > length ? 
this.slice(0, length - truncation.length) + truncation : String(this); 
} 
//剔除字符串前后空格 
function strip() { 
return this.replace(/^\s+/, '').replace(/\s+$/, ''); 
} 
//把字符串的html标记剔除 
function stripTags() { 
return this.replace(/<\w+(\s+("[^"]*"|'[^']*'|[^>])+)?>|<\/\w+>/gi, ''); 
} 
//把字符串中的script标记剔除 
function stripScripts() { 
return this.replace(new RegExp(Prototype.ScriptFragment, 'img'), ''); 
} 
//获取字符串中的script内容 
function extractScripts() { 
var matchAll = new RegExp(Prototype.ScriptFragment, 'img'); 
var matchOne = new RegExp(Prototype.ScriptFragment, 'im'); 
return (this.match(matchAll) || []).map(function(scriptTag) { 
return (scriptTag.match(matchOne) || ['', ''])[1]; 
}); 
} 
//执行字符串中的script内容 
function evalScripts() { 
return this.extractScripts().map(function(script) { return eval(script) }); 
} 
//转义HTML内容,例如把'<>&'等特殊字符替换成标准的HTML表达形式 
function escapeHTML() { 
escapeHTML.text.data = this; 
return escapeHTML.div.innerHTML; 
} 
function unescapeHTML() { 
var div = document.createElement('div'); 
div.innerHTML = this.stripTags(); 
return div.childNodes[0] ? (div.childNodes.length > 1 ? 
$A(div.childNodes).inject('', function(memo, node) { return memo+node.nodeValue }) : 
div.childNodes[0].nodeValue) : ''; 
} 
//按照separator参数把字符串分割成查询参数形式 
function toQueryParams(separator) { 
var match = this.strip().match(/([^?#]*)(#.*)?$/); 
if (!match) return { }; 
return match[1].split(separator || '&').inject({ }, function(hash, pair) { 
if ((pair = pair.split('='))[0]) { 
var key = decodeURIComponent(pair.shift()); 
var value = pair.length > 1 ? pair.join('=') : pair[0]; 
if (value != undefined) value = decodeURIComponent(value); 
if (key in hash) { 
if (!Object.isArray(hash[key])) hash[key] = [hash[key]]; 
hash[key].push(value); 
} 
else hash[key] = value; 
} 
return hash; 
}); 
} 
function toArray() { 
return this.split(''); 
} 
//返回字符串的字符 
function succ() { 
return this.slice(0, this.length - 1) + 
String.fromCharCode(this.charCodeAt(this.length - 1) + 1); 
} 
//获得重复的字符串 
function times(count) { 
return count < 1 ? '' : new Array(count + 1).join(this); 
} 
//把css样式类型的字符串转换成脚本形式 
function camelize() { 
var parts = this.split('-'), len = parts.length; 
if (len == 1) return parts[0]; 
var camelized = this.charAt(0) == '-' 
? parts[0].charAt(0).toUpperCase() + parts[0].substring(1) 
: parts[0]; 
for (var i = 1; i < len; i++) 
camelized += parts[i].charAt(0).toUpperCase() + parts[i].substring(1); 
return camelized; 
} 
//首字母大写 
function capitalize() { 
return this.charAt(0).toUpperCase() + this.substring(1).toLowerCase(); 
} 
//'borderBottomWidth'.underscore(); 
// -> 'border_bottom_width' 
function underscore() { 
return this.gsub(/::/, '/').gsub(/([A-Z]+)([A-Z][a-z])/,'#{1}_#{2}').gsub(/([a-z\d])([A-Z])/,'#{1}_#{2}').gsub(/-/,'_').toLowerCase(); 
} 
//'border_bottom_width'.dasherize(); 
// -> 'border-bottom-width' 
function dasherize() { 
return this.gsub(/_/,'-'); 
} 
//Returns a debug-oriented version of the string(返回一个用来调式的字符串) 
function inspect(useDoubleQuotes) { 
var escapedString = this.gsub(/[\x00-\x1f\\]/, function(match) { 
var character = String.specialChar[match[0]]; 
return character ? character : '\\u00' + match[0].charCodeAt().toPaddedString(2, 16); 
}); 
if (useDoubleQuotes) return '"' + escapedString.replace(/"/g, '\\"') + '"'; 
return "'" + escapedString.replace(/'/g, '\\\'') + "'"; 
} 
function toJSON() { 
return this.inspect(true); 
} 
function unfilterJSON(filter) { 
return this.sub(filter || Prototype.JSONFilter, '#{1}'); 
} 
function isJSON() { 
var str = this; 
if (str.blank()) return false; 
str = this.replace(/\\./g, '@').replace(/"[^"\\\n\r]*"/g, ''); 
return (/^[,:{}\[\]0-9.\-+Eaeflnr-u \n\r\t]*$/).test(str); 
} 
//Strips comment delimiters around Ajax JSON or JavaScript responses. This security method is called internally. 
function evalJSON(sanitize) { 
var json = this.unfilterJSON(); 
try { 
if (!sanitize || json.isJSON()) return eval('(' + json + ')'); 
} catch (e) { } 
throw new SyntaxError('Badly formed JSON string: ' + this.inspect()); 
} 
function include(pattern) { 
return this.indexOf(pattern) > -1; 
} 
function startsWith(pattern) { 
return this.indexOf(pattern) === 0; 
} 
function endsWith(pattern) { 
var d = this.length - pattern.length; 
return d >= 0 && this.lastIndexOf(pattern) === d; 
} 
function empty() { 
return this == ''; 
} 
function blank() { 
return /^\s*$/.test(this); 
} 
//和Template的evaluate方法一样 
function interpolate(object, pattern) { 
return new Template(this, pattern).evaluate(object); 
} 
return { 
gsub: gsub, 
sub: sub, 
scan: scan, 
truncate: truncate, 
strip: String.prototype.trim ? String.prototype.trim : strip, 
stripTags: stripTags, 
stripScripts: stripScripts, 
extractScripts: extractScripts, 
evalScripts: evalScripts, 
escapeHTML: escapeHTML, 
unescapeHTML: unescapeHTML, 
toQueryParams: toQueryParams, 
parseQuery: toQueryParams, 
toArray: toArray, 
succ: succ, 
times: times, 
camelize: camelize, 
capitalize: capitalize, 
underscore: underscore, 
dasherize: dasherize, 
inspect: inspect, 
toJSON: toJSON, 
unfilterJSON: unfilterJSON, 
isJSON: isJSON, 
evalJSON: evalJSON, 
include: include, 
startsWith: startsWith, 
endsWith: endsWith, 
empty: empty, 
blank: blank, 
interpolate: interpolate 
}; 
})()); 
Object.extend(String.prototype.escapeHTML, { 
div: document.createElement('div'), 
text: document.createTextNode('') 
}); 
String.prototype.escapeHTML.div.appendChild(String.prototype.escapeHTML.text); 
//以下估计是解决浏览器兼容问题 
if ('<\n>'.escapeHTML() !== '<\n>') { 
String.prototype.escapeHTML = function() { 
return this.replace(/&/g,'&').replace(/</g,'<').replace(/>/g,'>'); 
}; 
} 
if ('<\n>'.unescapeHTML() !== '<\n>') { 
String.prototype.unescapeHTML = function() { 
return this.stripTags().replace(/</g,'<').replace(/>/g,'>').replace(/&/g,'&'); 
}; 
}

blank
camelize
capitalize
dasherize
empty
endsWith
escapeHTML
evalJSON
evalScripts
extractScripts
gsub
include
inspect
interpolate
isJSON
parseQuery
scan
startsWith
strip
stripScripts
stripTags
sub
succ
times
toArray
toJSON
toQueryParams
truncate
underscore
unescapeHTML
unfilterJSON
下面只给出一些重要方法的例子,简单方法就略过了

escapeHTML() /unescapeHTML() :

'<div class="article">This is an article</div>'.escapeHTML(); 
// -> "<div class="article">This is an article</div>" 'x > 10'.unescapeHTML() 
// -> 'x > 10' '<h1>Pride & Prejudice</h1>'.unescapeHTML() 
// -> 'Pride & Prejudice'

evalJSON() /evalScripts()

String对象里面的有几个方法是为了防止XSS Attack攻击的,有兴趣的可以搜一下,下面给出XSS的概念:

Cross-site scripting (XSS) is a type of computer security vulnerability typically found in web applications which allow code injection by malicious web users into the web pages viewed by other users.

var person = '{ "name": "Violet", "occupation": "character" }'.evalJSON(); 
person.name; 
//-> "Violet" person = 'grabUserPassword()'.evalJSON(true); 
//-> SyntaxError: Badly formed JSON string: 'grabUserPassword()' 
person = '/*-secure-\n{"name": "Violet", "occupation": "character"}\n*/'.evalJSON() 
person.name; 
//-> "Violet"

'lorem... <script type="text/javascript"><!-- 
2 + 2 
// --></script>'.evalScripts(); 
// -> [4] '<script type="text/javascript"><!-- 
2 + 2 
// --></script><script type="text/javascript"><!-- 
alert("hello world!") 
// --></script>'.evalScripts(); 
// -> [4, undefined] (and displays 'hello world!' in the alert dialog)

gsub() /sub() :
var mouseEvents = 'click dblclick mousedown mouseup mouseover mousemove mouseout'; mouseEvents.gsub(' ', ', '); 
// -> 'click, dblclick, mousedown, mouseup, mouseover, mousemove, mouseout' mouseEvents.gsub(/\w+/, function(match){return 'on' + match[0].capitalize()}); 
// -> 'onClick onDblclick onMousedown onMouseup onMouseover onMousemove onMouseout' 
var markdown = '![a pear](/img/pear.jpg) ![an orange](/img/orange.jpg)'; 
markdown.gsub(/!\[(.*?)\]\((.*?)\)/, function(match){ return '<img alt="' + match[1] + '" src="' + match[2] + '" src="' + match[2] + '" />'; }); 
// -> '<img alt="a pear" src="/img/pear.jpg" src="img/pear.jpg" /> <img alt="an orange" src="/img/orange.jpg" src="img/orange.jpg" />' 
//================================================== 
var fruits = 'apple pear orange'; 
fruits.sub(' ', ', '); // -> 'apple, pear orange' 
fruits.sub(' ', ', ', 1); // -> 'apple, pear orange' 
fruits.sub(' ', ', ', 2); // -> 'apple, pear, orange' 
fruits.sub(/\w+/, function(match){return match[0].capitalize() + ','}, 2); 
// -> 'Apple, Pear, orange' 
var markdown = '![a pear](/img/pear.jpg) ![an orange](/img/orange.jpg)'; 
markdown.sub(/!\[(.*?)\]\((.*?)\)/, function(match){ return '<img alt="' + match[1] + '" src="' + match[2] + '" src="' + match[2] + '" />'; }); 
// -> '<img alt="a pear" src="/img/pear.jpg" src="img/pear.jpg" /> ![an orange](/img/orange.jpg)' 
markdown.sub(/!\[(.*?)\]\((.*?)\)/, '<img alt="#{1}" src="#{2}" src="#{2}" />'); 
// -> '<img alt="a pear" src="/img/pear.jpg" src="img/pear.jpg" /> ![an orange](/img/orange.jpg)'

interpolate() :
"#{animals} on a #{transport}".interpolate({ animals: "Pigs", transport: "Surfboard" }); 
//-> "Pigs on a Surfboard"

scan() :
var fruits = []; 
'apple, pear & orange'.scan(/\w+/, function(match){ fruits.push(match[0])}); fruits.inspect() 
// -> ['apple', 'pear', 'orange']

times() :
"echo ".times(3); //-> "echo echo echo "

toQueryParams():
'section=blog&id=45'.toQueryParams(); 
// -> {section: 'blog', id: '45'} 'section=blog;id=45'.toQueryParams(); 
// -> {section: 'blog', id: '45'} 
'http://www.example.com?section=blog&id=45#comments'.toQueryParams(); 
// -> {section: 'blog', id: '45'} 
'section=blog&tag=javascript&tag=prototype&tag=doc'.toQueryParams(); 
// -> {section: 'blog', tag:['javascript', 'prototype', 'doc']} 
'tag=ruby%20on%20rails'.toQueryParams(); 
// -> {tag: 'ruby on rails'} 
'id=45&raw'.toQueryParams(); 
// -> {id: '45', raw: undefined}

Javascript 相关文章推荐
javascript面向对象之共享成员属性与方法及prototype关键字用法
Jan 13 Javascript
jQuery基础知识点总结(DOM操作)
Jun 01 Javascript
Javascript 获取鼠标当前的位置实现方法
Oct 27 Javascript
12 款 JS 代码测试必备工具(翻译)
Dec 13 Javascript
使用jquery+iframe做一个ajax上传效果(实例)
Aug 24 jQuery
js中json对象和字符串的理解及相互转化操作实现方法
Sep 22 Javascript
原生JavaScript实现Ajax异步请求
Nov 19 Javascript
在element-ui的select下拉框加上滚动加载
Apr 18 Javascript
Vue实现拖放排序功能的实例代码
Jul 08 Javascript
Vue中跨域及打包部署到nginx跨域设置方法
Aug 26 Javascript
layui使用form表单实现post请求页面跳转的方法
Sep 14 Javascript
vue-cli4.5.x快速搭建项目
May 30 Vue.js
Prototype Template对象 学习
Jul 19 #Javascript
Prototype Number对象 学习
Jul 19 #Javascript
Prototype ObjectRange对象学习
Jul 19 #Javascript
Prototype RegExp对象 学习
Jul 19 #Javascript
Prototype Class对象学习
Jul 19 #Javascript
javascript iframe内的函数调用实现方法
Jul 19 #Javascript
9个javascript语法高亮插件 推荐
Jul 18 #Javascript
You might like
留言板翻页的实现详解
2006/10/09 PHP
php的declare控制符和ticks教程(附示例)
2014/03/21 PHP
smarty中post用法实例
2014/11/28 PHP
详解JavaScript数组和字符串中去除重复值的方法
2016/03/07 Javascript
jQuery实现响应鼠标事件的图片透明效果【附demo源码下载】
2016/06/16 Javascript
javascript 中的事件委托详解
2016/10/25 Javascript
AngularJS的Filter的示例详解
2017/03/07 Javascript
Vue项目webpack打包部署到服务器的实例详解
2017/07/17 Javascript
Angular4绑定html内容出现警告的处理方法
2017/11/03 Javascript
解决vue-cli + webpack 新建项目出错的问题
2018/03/20 Javascript
Vue中的v-for指令不起效果的解决方法
2018/09/27 Javascript
详解Vue2 添加对scss的支持
2019/01/02 Javascript
webpack常用构建优化策略小结
2019/11/21 Javascript
解决Vue 移动端点击出现300毫秒延迟的问题
2020/07/21 Javascript
用python 制作图片转pdf工具
2015/01/30 Python
用Python实现斐波那契(Fibonacci)函数
2016/03/25 Python
基于Python的接口测试框架实例
2016/11/04 Python
Python 结巴分词实现关键词抽取分析
2017/10/21 Python
pandas Dataframe行列读取的实例
2018/06/08 Python
python实现连续变量最优分箱详解--CART算法
2019/11/22 Python
python add_argument()用法解析
2020/01/29 Python
pycharm 代码自动补全的实现方法(图文)
2020/09/18 Python
selenium设置浏览器为headless无头模式(Chrome和Firefox)
2021/01/08 Python
python 逆向爬虫正确调用 JAR 加密逻辑
2021/01/12 Python
Html5移动端获奖无缝滚动动画实现示例
2018/06/25 HTML / CSS
某公司.Net方向面试题
2014/04/24 面试题
电子商务专业个人的自我评价
2013/12/19 职场文书
测量工程专业求职信
2014/02/24 职场文书
浪费资源的建议书
2014/03/12 职场文书
工作散漫检讨书
2014/09/16 职场文书
廉政文化进校园广播稿
2014/10/20 职场文书
个人贷款收入证明
2014/10/26 职场文书
现实表现材料范文
2014/12/23 职场文书
个人求职自荐信范文
2015/03/06 职场文书
php字符串倒叙
2021/04/01 PHP
python实现剪贴板的操作
2021/07/01 Python