正则表达式允许一个字符(它不应该)为什么?

时间:2013-09-05 22:20:43

标签: javascript jquery regex jquery-inputmask

您好我正在尝试创建一个识别正在输入的金钱和数字的正则表达式。我必须允许数字,因为我希望以编程方式输入非格式化的数字,然后我将自己格式化。出于某种原因,我的正则表达式允许使用单字母字符作为可能的输入。

[\$]?[0-9,]*\.[0-9][0-9]

据我所知,我的正则表达式接受添加多个逗号并且小数点后还需要两位数的情况。我已经知道如何解决这个问题。我把它缩小到可能*\.作为问题

修改

我发现正则表达式有效[\$]?([0-9,])*[\.][0-9]{2},但我仍然不知道它是如何或为什么首先失败

我正在使用.formatCurrency()将输入格式化为货币格式。它可以找到here,但它仍然允许我使用字母字符,所以我必须使用$(this).inputmask('Regex', { regex: "[\$]?([0-9,])*[\.][0-9]{2}" });进一步屏蔽它,其中找到输入掩码here$(this)是引用文本类型的输入元素。我的代码看起来像这样

<input type="text" id="123" data-Money="true">

 //in the script
 .find("input").each(function () {          
        if ($(this).attr("data-Money") == "true") {                            
            $(this).inputmask('Regex', { regex: "[\$]?([0-9,])*[\.][0-9]{2}" });
            $(this).on("blur", function () {
                $(this).formatCurrency();
            });

我希望这会有所帮助。我尝试创建一个JSfiddle但Idk如何添加外部库/插件/扩展

2 个答案:

答案 0 :(得分:3)

您在示例脚本中使用的“正则表达式”不是RegExp:

$(this).inputmask('Regex', { regex: "[\$]?([0-9,])*[\.][0-9]{2}" });

相反,它是一个字符串,其中包含一个模式,在某些时候被您的库转换为真正的RegExp

var RE=!(value instanceof RegExp) ? new RegExp(value) : value;

在字符串中,反斜杠\用于表示特殊字符,例如\n来表示换行符。在句号的开头添加反斜杠,即\.,没有任何作用,因为不需要“逃避”句号。

因此,从String创建的RegExp根本没有看到反斜杠。

不使用String作为正则表达式,而是使用JavaScript的文字正则表达式分隔符。

所以而不是:

$(this).inputmask('Regex', { regex: "[\$]?([0-9,])*[\.][0-9]{2}" });

使用

$(this).inputmask('Regex', { regex: /[\$]?([0-9,])*[\.][0-9]{2}/ });

我相信你的“正则表达”会像你期望的那样表现出来。

(注意使用正斜杠/来分隔您的模式,JavaScript将使用它来提供真正的RegExp。)

答案 1 :(得分:1)

首先,您可以将'[0-9]'替换为'\ d'。所以我们可以更干净地重写你的第一个正则表达式

\$?[\d,]*\.\d\d

打破这个局面:

\$?       - A literal dollar sign, zero or one
[\d,]*    - Either a digit or a comma, zero or more
\.        - A literal dot, required
\d        - A digit, required
\d        - A digit, required

由此,我们可以看到最小法定字符串为\.\d\d,长度为三个字符。你给的正则表达式永远不会验证任何一个字符串。

看看你的第二个正则表达式,

[\$]?     - A literal dollar sign, zero or one
([0-9,])* - Either a digit or a comma, subexpression for later use, zero or more
[\.]      - A literal dot, required
[0-9]{2}  - A digit, twice required

这与上面的最小匹配字符串完全相同 - \.\d\d

编辑如上所述,根据您可能需要转义斜杠的语言,以确保在处理字符串时语言不会误解它们。

另外,另外,下面的正则表达式可能更接近你需要的东西。

[A-Z]{3} ?(\d{0,3}(?:([,. ])\d{3}(?:\2\d{3})*)?)(?!\2)[,.](\d\d)\b

说明:

[A-Z]{3}       - Three letters; for an ISO currency code
 ?             - A space, zero or more; for readability
(              - Capture block; to catch the integer currency amount
  \d{0,3}        - A digit, between one and three; for the first digit block
  (?:            - Non capturing block (NC)
    ([,. ])        - A comma, dot or space; as a thousands delimiter
    \d{3}          - A digit, three; the first possible whole thousands
    (?:            - Non capturing block (NC) 
      \2             - Match 2; the captured thousands delimiter above
      \d{3}          - A digits, three
    )*             - The above group, zero or more, i.e. as many thousands as we want
  )?              - The above (NC) group, zero or one, ie. all whole thousands
)              - The above group, i.e everything before the decimal
[.,]           - A comma or dot, as a decimal delimiter
(\d{2})        - Capture, A digit, two; ie. the decimal portion
\b             - A word boundry; to ensure that we don't catch another
                 digit in the wrong place.

负面前瞻是由John Kugelman在this question中的回答提供的。

这正确匹配(括号括在方括号中):

[AUD 1.00]
[USD 1,300,000.00]
[YEN 200 000.00]
I need [USD 1,000,000.00], all in non-sequential bills.

但不是:

GBP 1.000
YEN 200,000