1

我需要一个正则表达式来使用RegularExpressionAttribute验证用户名。它应该匹配不包含requestPathInvalidCharacters (<,>,*,%,&,:,\,?) 的任何默认值的用户名。

所以它应该匹配

阿法。

aaf8-a

阿法

它不应该匹配

aa<fa

aaf>a

aaf*a

阿法*

%a

aaf&a

aaf:一个

aaf\a

aaf?a

到目前为止,我一直在调整这个正则表达式,但在其中一种情况下它仍然失败......

^(?!.*(<|>|\*|%|&|:|\\|\?).*)$
    [Fact]
    public void CanValidateAgainstInvalidCharacter()
    {
        var result = true;
        Assert.True(result);
        var listWeDontWant = @"<,>,*,%,&,:,\,?";
        var inner = listWeDontWant.Split(',').Select(x => Regex.Escape(x)).Aggregate((s, s1) => s + "|" + s1);
        //inner = @"\\";
        var pattern = @"^(?!.*(" + inner + ").*)$";

        Debug.WriteLine(pattern);

        //var isMatch = ;

        //Debug.WriteLine(isMatch);
        pattern = @"^[^<>*%&:\\\?]+$";
        Assert.False(Regex.IsMatch("aaf\a", pattern));

      //  Assert.True(Regex.IsMatch("aafa.", pattern));
       // Assert.True(Regex.IsMatch("aaf8-a", pattern));
        //Assert.True(Regex.IsMatch("aafa", pattern));
        //Assert.True(Regex.IsMatch("aafa", pattern));
        //Assert.True(Regex.IsMatch("aa,fa", pattern));
        //Assert.True(Regex.IsMatch("aafa", pattern));

        //Assert.False(Regex.IsMatch("aa<fa", pattern));
        //Assert.False(Regex.IsMatch("aaf>a", pattern));
        //Assert.False(Regex.IsMatch("aaf*a", pattern));
        //Assert.False(Regex.IsMatch("aafa*", pattern));
        //Assert.False(Regex.IsMatch("aaf%a", pattern));
        //Assert.False(Regex.IsMatch("aaf&a", pattern));
        //Assert.False(Regex.IsMatch("aaf:a", pattern));
        //Assert.False(Regex.IsMatch("aaf\a", pattern));
        //Assert.False(Regex.IsMatch("aaf?a", pattern));

    }
4

2 回答 2

2

您期望"aaf\a"匹配失败,但它不会因为\a是转义字符“a”。因此,正则表达式匹配字符串,因为允许使用转义的“a”。我怀疑如果字符串包含反斜杠,您预计它会失败\。在这种情况下,更改您的示例字符串并转义反斜杠,或使用逐字字符串:

Regex.IsMatch("aaf\\a", pattern)  // escaped
Regex.IsMatch(@"aaf\a", pattern)  // verbatim string

随着这种变化的发生,该@"^[^<>*%&:\\\?]+$"模式将起作用并且您的断言应该通过。

至于您的否定前瞻模式,它目前接受的比它应该接受的更多,因为它断言是否可能匹配,但实际上并没有消耗任何文本,所以当这不是预期的时候它会产生积极的匹配。要解决此问题,请在模式中添加一些内容以实际匹配(并使用)文本。

// added ".+" before the final "$"
var pattern = @"^(?!.*(" + inner + ").*).+$";
于 2013-10-11T05:39:16.183 回答
1

那不应该只是

^[^<>*%&:\\\?]+$ 

或者

^[^<>*%&:\\\?]*$ 

如果空字符串有效

或者如果invalid包含 RequestPathInvalidCharacters ,则使用自动化

var regex = new Regex(string.Format("^[^{0}]*$",
                                    Regex.Escape(string.Join("",invalid.Split(',')))));
于 2013-10-09T07:17:41.643 回答