Java:拆分以逗号分隔的字符串,但忽略引号中的逗号

Jas*_*n S 238 java regex string

我有一个模糊的字符串:

foo,bar,c;qual="baz,blurb",d;junk="quux,syzygy"
Run Code Online (Sandbox Code Playgroud)

我想用逗号分割 - 但我需要在引号中忽略逗号.我怎样才能做到这一点?好像正则表达式方法失败了; 我想我可以在看到引号时手动扫描并进入不同的模式,但是使用预先存在的库会更好.(编辑:我想我的意思是已经是JDK的一部分或已经是Apache Commons等常用库的一部分的库.)

上面的字符串应该分成:

foo
bar
c;qual="baz,blurb"
d;junk="quux,syzygy"
Run Code Online (Sandbox Code Playgroud)

注意:这不是CSV文件,它是包含在具有更大整体结构的文件中的单个字符串

Bar*_*ers 418

尝试:

public class Main { 
    public static void main(String[] args) {
        String line = "foo,bar,c;qual=\"baz,blurb\",d;junk=\"quux,syzygy\"";
        String[] tokens = line.split(",(?=(?:[^\"]*\"[^\"]*\")*[^\"]*$)", -1);
        for(String t : tokens) {
            System.out.println("> "+t);
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

输出:

> foo
> bar
> c;qual="baz,blurb"
> d;junk="quux,syzygy"
Run Code Online (Sandbox Code Playgroud)

换句话说:只有当逗号为零或前面有偶数引号时才能在逗号上拆分.

或者,眼睛有点友善:

public class Main { 
    public static void main(String[] args) {
        String line = "foo,bar,c;qual=\"baz,blurb\",d;junk=\"quux,syzygy\"";

        String otherThanQuote = " [^\"] ";
        String quotedString = String.format(" \" %s* \" ", otherThanQuote);
        String regex = String.format("(?x) "+ // enable comments, ignore white spaces
                ",                         "+ // match a comma
                "(?=                       "+ // start positive look ahead
                "  (?:                     "+ //   start non-capturing group 1
                "    %s*                   "+ //     match 'otherThanQuote' zero or more times
                "    %s                    "+ //     match 'quotedString'
                "  )*                      "+ //   end group 1 and repeat it zero or more times
                "  %s*                     "+ //   match 'otherThanQuote'
                "  $                       "+ // match the end of the string
                ")                         ", // stop positive look ahead
                otherThanQuote, quotedString, otherThanQuote);

        String[] tokens = line.split(regex, -1);
        for(String t : tokens) {
            System.out.println("> "+t);
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

其产生与第一个例子相同.

编辑

正如@MikeFHay在评论中提到的那样:

我更喜欢使用Guava的Splitter,因为它有更健全的默认值(参见上面关于空匹配被修剪的讨论String#split(),所以我做了:

Splitter.on(Pattern.compile(",(?=(?:[^\"]*\"[^\"]*\")*[^\"]*$)"))
Run Code Online (Sandbox Code Playgroud)

  • @Alex,是的,逗号*匹配,但空匹配不在结果中.将"-1"添加到split方法参数:`line.split(regex,-1)`.请参阅:http://docs.oracle.com/javase/6/docs/api/java/lang/String.html#split(java.lang.String,%20int) (5认同)
  • 效果很好!我更喜欢使用Guava的Splitter,因为它有saner默认值(参见上面关于空字符串被String#split修剪的讨论),所以我做了`Splitter.on(Pattern.compile(",(?=([^ \"]*\"[^ \"]*\")*[^ \"]*$)"))`. (2认同)
  • **警告!!!!这个正则表达式很慢!!!**它有O(N ^ 2)行为,因为每个逗号的前瞻一直看到字符串的结尾.使用此正则表达式导致大型Spark作业减速4倍(例如45分钟 - > 3小时).更快的替代方法是`findAllIn("?s)(?:\".*?\"| [^ \",]*)*")`结合后处理步骤跳过第一个(总是 - 每个非空字段后面的字段. (2认同)

Fab*_*eeg 43

虽然我确实喜欢正则表达式,但是对于这种依赖于状态的标记化,我认为一个简单的解析器(在这种情况下比这个单词要简单得多)可能是一个更清晰的解决方案,特别是在可维护性方面. ,例如:

String input = "foo,bar,c;qual=\"baz,blurb\",d;junk=\"quux,syzygy\"";
List<String> result = new ArrayList<String>();
int start = 0;
boolean inQuotes = false;
for (int current = 0; current < input.length(); current++) {
    if (input.charAt(current) == '\"') inQuotes = !inQuotes; // toggle state
    boolean atLastChar = (current == input.length() - 1);
    if(atLastChar) result.add(input.substring(start));
    else if (input.charAt(current) == ',' && !inQuotes) {
        result.add(input.substring(start, current));
        start = current + 1;
    }
}
Run Code Online (Sandbox Code Playgroud)

如果你不关心在引号内保留逗号,你可以通过用引号替换引号中的逗号来简化这种方法(不处理起始索引,没有最后一个字符特殊情况),然后用逗号分割:

String input = "foo,bar,c;qual=\"baz,blurb\",d;junk=\"quux,syzygy\"";
StringBuilder builder = new StringBuilder(input);
boolean inQuotes = false;
for (int currentIndex = 0; currentIndex < builder.length(); currentIndex++) {
    char currentChar = builder.charAt(currentIndex);
    if (currentChar == '\"') inQuotes = !inQuotes; // toggle state
    if (currentChar == ',' && inQuotes) {
        builder.setCharAt(currentIndex, ';'); // or '?', and replace later
    }
}
List<String> result = Arrays.asList(builder.toString().split(","));
Run Code Online (Sandbox Code Playgroud)

  • 请记住,如果逗号是最后一个字符,它将在最后一个项目的字符串值中。 (2认同)

Jon*_*erg 21

http://sourceforge.net/projects/javacsv/

https://github.com/pupi1985/JavaCSV-Reloaded (前一个库的分支,允许生成的输出\r\n在不运行Windows时具有Windows行终止符)

http://opencsv.sourceforge.net/

适用于Java的CSV API

你能推荐一个Java库来读取(并可能写)CSV文件吗?

Java lib或app将CSV转换为XML文件?

  • 不一定......我的技能通常是足够的,但他们受益于磨练. (7认同)
  • 认识到OP正在解析CSV文件的好电话.外部库非常适合此任务. (3认同)

Mar*_*ski 8

我不建议Bart的正则表达式答案,我发现在这种特殊情况下解析解决方案更好(正如Fabian提出的那样).我已经尝试了正则表达式解决方案和自己的解析实现我发现:

  1. 解析比使用反向引用正则表达式分裂要快得多 - 短字符串快〜20倍,长字符串快〜40倍.
  2. 正则表达式在最后一个逗号后找不到空字符串.这不是原始问题,但这是我的要求.

我的解决方案和测试如下.

String tested = "foo,bar,c;qual=\"baz,blurb\",d;junk=\"quux,syzygy\",";
long start = System.nanoTime();
String[] tokens = tested.split(",(?=([^\"]*\"[^\"]*\")*[^\"]*$)");
long timeWithSplitting = System.nanoTime() - start;

start = System.nanoTime(); 
List<String> tokensList = new ArrayList<String>();
boolean inQuotes = false;
StringBuilder b = new StringBuilder();
for (char c : tested.toCharArray()) {
    switch (c) {
    case ',':
        if (inQuotes) {
            b.append(c);
        } else {
            tokensList.add(b.toString());
            b = new StringBuilder();
        }
        break;
    case '\"':
        inQuotes = !inQuotes;
    default:
        b.append(c);
    break;
    }
}
tokensList.add(b.toString());
long timeWithParsing = System.nanoTime() - start;

System.out.println(Arrays.toString(tokens));
System.out.println(tokensList.toString());
System.out.printf("Time with splitting:\t%10d\n",timeWithSplitting);
System.out.printf("Time with parsing:\t%10d\n",timeWithParsing);
Run Code Online (Sandbox Code Playgroud)

当然,如果你对它的丑陋感到不舒服,你可以自由地改变切换到这个片段中的else-ifs.注意用分离器切换后没有断裂.StringBuilder被设计为StringBuffer,以提高速度,线程安全性无关紧要.

  • 有关时间分裂与解析的有趣观点.但是,声明#2不准确.如果你在Bart的答案中为split方法添加一个`-1`,你将捕获空字符串(包括最后一个逗号后的空字符串):`line.split(regex,-1)` (2认同)