Hello regular expression experts,
There has never been a string manipulation problem I couldn't resolve with regular expressions until now, at least in an elegant manner using just one step. Here is the sample data I'm working with:
0,"section1","(7) Delivery of 'certificate' outside the United States prohibited. Since both section 339 of the 1940 statute, 68/ and section 341 of the present law are explicit in their statement that the certificate shall be furnished the citizen, only if such individual is at the time within the United States, it is clear that the document could not and cannot be delivered outside the United States.",http://www.google.com/
1,"section2",,http://www.google.com/
2,"section3",",,",http://www.google.com/
This is a section of a much larger CSV file. With one elegant regular expression, I'd like to replace only all the commas that occur within the double quotes with an underscore character (_). It is important that the regular expression does NOT replace any commas outside the quotes because that would mess up the CSV data structure.
Thanks, Tom
--
CLARIFICATION:
Sorry guys, I posted the question without fully clarifying my situation, so let me summarize below:
""
or """
etc., so they are easily replaced beforehand).Using the sample text above, here is what it SHOULD look like after running the regular expression replacement (there should be a total of 5 replacements):
0,"section1","(7) Delivery of 'certificate' outside the United States prohibited. Since both section 339 of the 1940 statute_ 68/ and section 341 of the present law are explicit in their statement that the certificate shall be furnished the citizen_ only if such individual is at the time within the United States_ it is clear that the document could not and cannot be delivered outside the United States.",http://www.google.com/
1,"section2",,http://www.google.com/
2,"section3","__",http://www.google.com/
To use RegEx, the first argument of replace will be replaced with regex syntax, for example /regex/ . This syntax serves as a pattern where any parts of the string that match it will be replaced with the new substring. The string 3foobar4 matches the regex /\d. *\d/ , so it is replaced.
$ means "Match the end of the string" (the position after the last character in the string). Both are called anchors and ensure that the entire string is matched instead of just a substring.
[] denotes a character class. () denotes a capturing group. [a-z0-9] -- One character that is in the range of a-z OR 0-9. (a-z0-9) -- Explicit capture of a-z0-9 .
I'll help you, but you have to promise to stop using the word "elegant". It's been working too hard lately, and deserves a rest. :P
(?m),(?=[^"]*"(?:[^"\r\n]*"[^"]*")*[^"\r\n]*$)
This matches a comma if, between the comma and the end of the record, there's an odd number of quotation marks. I'm assuming a standard CSV format, in which a record ends at the next line separator that isn't enclosed in quotes. Line separators are legal inside quoted fields, as are quotes if they're escaped with another quote.
Depending on which regex flavor you're using, you may have to use \r?$
instead of just $
. In .NET, for example, only the linefeed (\n
) is considered a line separator. But in Java, $
matches before the \r
in \r\n
, but not between the \r
and the \n
(unless you set UNIX_LINES mode).
Regular expressions are not particularly good at matching balanced text (i.e. starting and ending quotes).
A naïve approach would be to repeatedly apply something like this (until it no longer matched):
s/(^[^"]*(?:"[^"]*"[^"]*)*?)"([^",]*),([^"]*)"/$1"$2_$3"/
But that wouldn't work with escaped quotes. The best (i.e. simplest, most readable, and most maintanable) solution is to use a CSV file parser, go through all the field values one by one (replacing commas with underscores as you go), then write it back out to the file.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With