更改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\"));
    

要回复问题请先登录注册