如何解决更改RegExp标志
| 因此,基本上我自己编写了此函数,以便能够计算字符串中子字符串的出现次数:String.prototype.numberOf = function(needle) {
var num = 0,lastIndex = 0;
if(typeof needle === \"string\" || needle instanceof String) {
while((lastIndex = this.indexOf(needle,lastIndex) + 1) > 0)
{num++;} return num;
} else if(needle instanceof RegExp) {
// needle.global = true;
return this.match(needle).length;
} return 0;
};
该方法本身执行得很好,并且基于RegExp和基于String的搜索在执行时间方面都相当可比(在整个Ray Bradbury的“ 451 Fahrenheit”中搜索所有“ the”的时间均为〜2ms) s)。
但是,令我困扰的是无法更改所提供的RegExp实例的标志。没有将提供的正则表达式的全局标志设置为true的情况下,在此函数中调用String.prototype.match是没有意义的,因为这样只会注意到第一次出现。您当然可以在传递给该函数的每个RegExp上手动设置该标志,但是我希望能够克隆然后操纵提供的正则表达式的标志。
令人惊讶的是,我不允许这样做,因为RegExp.prototype.global标志(更确切地说是所有标志)似乎是只读的。因此,注释掉的行8。
所以我的问题是:是否有一种很好的方法来更改RegExp对象的标志?
我真的不想做这样的事情:
if(!expression.global)
expression = eval(expression.toString() + \"g\");
某些实现可能不支持RegExp.prototype.toString,而仅从Object.prototype继承它,或者它可能是完全不同的格式。首先,这似乎是一种不良的编码实践。
解决方法
首先,当
needle
是不匹配的正则表达式时,您当前的代码将无法正常工作。即以下行:
return this.match(needle).length;
如果没有匹配项,match
方法将返回null
。当(未成功)访问null
的length
属性时,将生成JavaScript错误。这很容易解决,如下所示:
var m = this.match(needle);
return m ? m.length : 0;
现在解决眼前的问题。当您说global
,ignoreCase
和multiline
是只读属性时,您是对的。唯一的选择是创建一个新的RegExp。因为正则表达式源字符串存储在re.source
属性中,所以这很容易做到。这是函数的经过测试的修改后的版本,可以纠正上述问题并在needle
尚未设置global
标志时创建一个新的RegExp对象:
String.prototype.numberOf = function(needle) {
var num = 0,lastIndex = 0;
if (typeof needle === \"string\" || needle instanceof String) {
while((lastIndex = this.indexOf(needle,lastIndex) + 1) > 0)
{num++;} return num;
} else if(needle instanceof RegExp) {
if (!needle.global) {
// If global flag not set,create new one.
var flags = \"g\";
if (needle.ignoreCase) flags += \"i\";
if (needle.multiline) flags += \"m\";
needle = RegExp(needle.source,flags);
}
var m = this.match(needle);
return m ? m.length : 0;
}
return 0;
};
,var globalRegex = new RegExp(needle.source,\"g\");
现场演示编辑:m只是为了说明您可以设置多个修饰符
var regex = /find/;
var other = new RegExp(regex.source,\"gm\");
alert(other.global);
alert(other.multiline);
,您没有太多可以做的,但我强烈建议您避免使用eval
。您可以扩展RegExp原型以帮助您。
RegExp.prototype.flags = function () {
return (this.ignoreCase ? \"i\" : \"\")
+ (this.multiline ? \"m\" : \"\")
+ (this.global ? \"g\" : \"\");
};
var reg1 = /AAA/i;
var reg2 = new RegExp(reg1.source,reg1.flags() + \'g\');
,r = new Regexp(r.source,r.flags + (r.global ? \"\" : \"g\"));
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。