Я нашел несколько решений, но результаты, которые я получаю, не соответствуют ожиданиям.

Я хочу взять строку и разделить ее запятыми, за исключением случаев, когда запятые заключены в двойные кавычки. Я хотел бы игнорировать пробелы. Я могу жить с потерей двойных кавычек в процессе, но это не обязательно.

CSV лучший способ сделать это? Будет ли решение регулярного выражения лучше?

#!/usr/local/bin/python2.7

import csv

s = 'abc,def, ghi, "jkl, mno, pqr","stu"'

result = csv.reader(s, delimiter=',', quotechar='"')

for r in result: 
    print r

# Should display:
# abc
# def
# ghi
# jkl, mno, pqr
# stu
#
# But I get:
# ['a']
# ['b']
# ['c']
# ['', '']
# ['d']
# ['e']
# ['f']
# ['', '']
# [' ']
# ['g']
# ['h']
# ['i']
# ['', '']
# [' ']
# ['jkl, mno, pqr']
# ['', '']
# ['stu']

print r[1]  # Should be "def" but I get and "list index out of range" error.
0
Calab 24 Окт 2018 в 09:29

2 ответа

Лучший ответ

Вы можете использовать регулярное выражение

".+?"|[\w-]+

Это будет соответствовать двойным кавычкам, за которыми следуют любые символы, пока не будет найдена следующая двойная кавычка - ИЛИ, это будет соответствовать символам слова (без запятых и кавычек).

https://regex101.com/r/IThYf7/1

import re
s = 'abc,def, ghi, "jkl, mno, pqr","stu"'
for r in re.findall(r'".+?"|[\w-]+', s):
    print(r)

Если вы хотите избавиться от " в цитируемых разделах, лучшее, что я мог выяснить, используя модуль regex (чтобы можно было использовать \K), было:

(?:^"?|, ?"?)\K(?:(?<=").+?(?=")|[\w-]+)

https://regex101.com/r/IThYf7/3

2
CertainPerformance 24 Окт 2018 в 07:31

Помимо использования csv у вас может быть еще один хороший подход, который поддерживается новым модулем regex (т.е. pip install regex):

"[^"]*"(*SKIP)(*FAIL)|,\s*

"[^"]*"(*SKIP)(*FAIL) # match everything between two double quotes and "forget" about them
|                     # or
,\s*                  # match a comma and 0+ whitespaces

Python
import regex as re

rx = re.compile(r'"[^"]*"(*SKIP)(*FAIL)|,\s*')
string = 'abc,def, ghi, "jkl, mno, pqr","stu"'

parts = rx.split(string)
print(parts)

Это дает

['abc', 'def', 'ghi', '"jkl, mno, pqr"', '"stu"']

См. демонстрацию на regex101.com .

0
Jan 24 Окт 2018 в 06:45
52962287