正则表达式正面看起来很困难

时间:2012-03-02 13:53:53

标签: java regex

我的目标是在小数值中匹配前0和零之后的所有内容。如果第一个小数位为零,那么我也想匹配小数。如果没有小数则不捕获任何内容。

以下是我想要的一些例子:

180.570123 // should capture the "0123" on the end
180.570    // should capture the "0" on the end
180.0123   // should capture the ".0123" on the end
180.0      // should capture the ".0" on the end
180123     // should capture nothing
180        // should capture nothing

如果第一个小数位为0,那么匹配很容易:

(\.0.*)

当第一个小数位不是0时,我的问题是匹配。我相信正面看后会解决这个问题,但我无法让它正常工作。这是我试过的一个正则表达式:

(?<=^.*\..*)0.*

这个正则表达式最终将用于Java。

更新

我将使用这个正则表达式来使用Java的replaceAll方法去掉字符串末尾的数字和可能的小数点。我将通过用空字符串替换捕获组来完成此操作。这是我想要的更好的例子。

String case1 = "180.570123";
String case2 = "180.570";
String case3 = "180.0123";
String case4 = "180.0";
String case5 = "180123";
String case6 = "180";

String result = null;

result = case1.replaceAll( "THE REGEX I NEED", "" );
System.out.println( result ); // should print 180.57

result = case2.replaceAll( "THE REGEX I NEED", "" );
System.out.println( result ); // should print 180.57

result = case3.replaceAll( "THE REGEX I NEED", "" );
System.out.println( result ); // should print 180

result = case4.replaceAll( "THE REGEX I NEED", "" );
System.out.println( result ); // should print 180

result = case5.replaceAll( "THE REGEX I NEED", "" );
System.out.println( result ); // should print 180123

result = case6.replaceAll( "THE REGEX I NEED", "" );
System.out.println( result ); // should print 180

另外,我正在http://gskinner.com/RegExr/

测试这些正则表达式

4 个答案:

答案 0 :(得分:3)

您可以使用以下表达式:

\.[1-9]*(0\d*)

你想要的是第一个捕获组。 (小数点除外。)

如果您想捕获小数点,可以使用:

(?:\.[1-9]+|(?=\.))(\.?0\d*)

示例(online):

Pattern p = Pattern.compile("(?:\\.[1-9]+|(?=\\.))(\\.?0\\d*)");

String[] strs = {"180.570123", "180.570", "180.0123", "180.0", "180123", "180", "180.2030405"};

for (String s : strs) {
    Matcher m = p.matcher(s);
    System.out.printf("%-12s: Match: %s%n", s,
        m.find() ? m.group(1) : "n/a");
}

输出:

180.570123  : Match: 0123
180.570     : Match: 0
180.0123    : Match: .0123
180.0       : Match: .0
180123      : Match: n/a
180         : Match: n/a
180.2030405 : Match: 030405

答案 1 :(得分:1)

我会编写一个小函数来进行提取而不是正则表达式。

private String getZeroPart(final String s) {
        final String[] strs = s.split("\\.");
        if (strs.length != 2 || strs[1].indexOf("0") < 0) {
            return null;
        } else {
            return strs[1].startsWith("0") ? "." + strs[1] : strs[1].substring(strs[1].indexOf("0"));
        }
    }

测试它:

final String[] ss = { "180.570123", "180.570", "180.0123", 
"180.0", "180123", "180", "180.2030405","180.5555" };

        for (final String s : ss) {
            System.out.println(getZeroPart(s));
        }

输出:

0123
0
.0123
.0
null
null
030405
null

<强>更新

基于问题的编辑。对方法进行一些更改以获得正确的数字:

private String cutZeroPart(final String s) {
    final String[] strs = s.split("\\.");
    if (strs.length != 2 || strs[1].indexOf("0") < 0) {
        return s;
    } else {
        return strs[1].startsWith("0") ? strs[0] : s.substring(0, strs[0].length() + strs[1].indexOf("0") + 1);
    }
}

输出:

180.570123 -> 180.57
180.570 -> 180.57
180.0123 -> 180
180.0 -> 180
180123 -> 180123
180 -> 180
180.2030405 -> 180.2
180.5555 -> 180.5555

答案 2 :(得分:0)

Lookbehinds可能有点矫枉过正。这个效果很好 - &gt;

/\.\d*(0\d*)/

答案 3 :(得分:0)

你可以试试这个:

        Matcher m = Pattern.compile("(?<=(\\.))(\\d*?)(0.*)").matcher("180.2030405");
        String res="";
        if(m.find()){
            if(m.group(2).equals("")){
                res = "."+m.group(3);
            }
            else{
                res = m.group(3);
            }
        }