在csv中写入io.BytesIO在python3中失败

时间:2016-06-22 16:57:22

标签: python string python-3.x csv bytesio

我正在尝试编写python 2/3兼容代码来将字符串写入csv文件对象。这段代码:

line_as_list = [line.encode() for line in line_as_list]
writer_file =  io.BytesIO()
writer = csv.writer(writer_file, dialect=dialect, delimiter=self.delimiter)
for line in line_as_list:
    assert isinstance(line,bytes)
    writer.writerow(line)

在Python3上给出了这个错误:

>           writer.writerow(line)
E           TypeError: a bytes-like object is required, not 'str'

但断言对类型没有问题,为什么csv会产生错误?

我不能仅将BytesIO用于Python 2和3吗?这里的问题在哪里?

1 个答案:

答案 0 :(得分:18)

在Python3中csv.writer期望在文本模式下打开类似文件的对象。 在Python2中,csv.writer期望以二进制模式打开类似文件的对象。

因此,在Python3中,使用io.StringIO,而在Python2中使用io.BytesIO

import io
import csv
import sys
PY3 = sys.version_info[0] == 3

line_as_list = [u'foo', u'bar']
encoding = 'utf-8'

if PY3:
    writer_file =  io.StringIO()
else:
    writer_file =  io.BytesIO()
    line_as_list = [line.encode(encoding) for line in line_as_list]

writer = csv.writer(writer_file, dialect='excel', delimiter=',')
writer.writerow(line_as_list)
content = writer_file.getvalue()

if PY3:
    content = content.encode(encoding)

print(type(content))
print(repr(content))

在Python3中,上面的代码打印

<class 'bytes'>
b'foo,bar\r\n'

在Python2中,上面的代码打印

<type 'str'>
'foo,bar\r\n'