The csv
module provide tools for working with and manipulating csv(comma separated values) files.
csv files are used to store tabular data in plain text formats. csv data consists of records and each record is made up of fields. Normally, a line in a csv file represents a single record while fields in a record are separated by commas.
The following is an example of csv data.
first_name,last_name,age,gender
John,Main,30,male
Jane,Doe,24,female
Morrison,Smith,22,male
Brian,Gates,32,male
Mary,Reagan,25,female
Typically, files with a .csv
extension are normally associated with csv data.
Reading csv data
To read data from a csv file, use the reader()
function. The function has the following basic syntax:
csv.reader(iterable)
The reader()
function takes any iterable that yields a line during each iteration. This can be a file object, a list or any other iterable made of lines. It returns an iterator of lists, where each list represents a record in the csv data.
Consider if the previous csv data exists in a file called demo.csv
.
import csv
with open('demo.csv') as file:
reader = csv.reader(file)
for row in reader:
print(row)
['first_name', 'last_name', 'age', 'gender']
['John', 'Main', '30', 'male']
['Jane', 'Doe', '24', 'female']
['Morrison', 'Smith', '22', 'male']
['Brian', 'Gates', '32', 'male']
['Mary', 'Reagan', '25', 'female']
In the above example, we used data stored in a file object but as earlier mentioned, you can literally use any iterable. In the following example, we use a list instead of a file object.
import csv
data = """first_name,last_name,age,gender
John,Main,30,male
Jane,Doe,24,female
Morrison,Smith,22,male
Brian,Gates,32,male
Mary,Reagan,25,female"""
reader = csv.reader(data.splitlines())
for row in reader:
print(row)
['first_name', 'last_name', 'age', 'gender']
['John', 'Main', '30', 'male']
['Jane', 'Doe', '24', 'female']
['Morrison', 'Smith', '22', 'male']
['Brian', 'Gates', '32', 'male']
['Mary', 'Reagan', '25', 'female']
Writing csv data
The writer()
function creates an object for writing csv data. It has the following basic syntax:
csv.writer(fileobj)
Where the fileobj
is the opened file where the csv data will be written to.
After creating the writer object, we can then use writerow()
method to write a single record into the file or writerows()
to write multiple rows at once.
import csv
with open('demo.csv', 'a') as file:
writer = csv.writer(file)
writer.writerow(('Ruth', 'Miller', 24, 'female'))
writer.writerow(('Brandy', 'Jones', 22, 'male'))
Note that in the above example, we opened the demo.csv
file with the "a"
mode. This ensures that the new rows are appended to the file, if we had used the "w"
mode instead, the existing rows would have been overwritten.
import csv
with open('demo.csv') as file:
print(file.read())
first_name,last_name,age,gender
John,Main,30,male
Jane,Doe,24,female
Morrison,Smith,22,male
Brian,Gates,32,male
Mary,Reagan,25,female
Ruth,Miller,24,female
Brandy,Jones,22,male
As you can see above, the new records have been added successfully.
To add multiple rows at once, use the writer.writerows()
method.
import csv
rows = [
('Stephen', 'King', 32, 'male'),
('Molly', 'Jones', 22, 'female')
]
with open('demo.csv', 'a') as file:
writer = csv.writer(file)
writer.writerows(rows)
Quoting behavior
In cases where you want records of certain types to be stored in quotes, you can specify the quoting
flag when creating the writer
object. For example, if you want string items to be in quotes but numeric values not to be, you can specify the quoting parameter as csv.QUOTE_NONNUMERIC
.
import csv
rows = [
('Stephen', 'King', 32, 'male'),
('Molly', 'Jones', 22, 'female')
]
with open('demo.csv', 'w') as file:
writer = csv.writer(file, quoting = csv.QUOTE_NONNUMERIC)
writer.writerows(rows)
If you open the demo.csv
file, you will see that in the written records , string fields are quoted.
import csv
with open('demo.csv') as file:
print(file.read())
"Stephen","King",32,"male"
"Molly","Jones",22,"female"
All the quoting options are as shown in the following table:
QUOTE_ALL |
Quote any field regardless of type |
QUOTE_NONE |
Don't quote any field. |
QUOTE_MINIMAL |
Quote only fields containing some special characters. This is the default quoting behavior. |
QUOTE_NONNUMERIC |
Quote all fields except those with numeric values like integers and floats. |
csv dialects
Separating csv fields with a comma is just the popular convention, there is really no well defined standard on csv format. For example, you could as well separate the fields with spaces instead of commas. This means that any csv parser needs to be very flexible when dealing with csv delimiters.
Dialects allow us to specify to the parser the delimiter and other important parameters that will be used when parsing the csv files. The csv.list_dialects()
function returns a list of the registered dialects.
import csv
print(csv.list_dialects())
['excel', 'excel-tab', 'unix']
The default dialect is excel
, in which the fields are delimited with commas.
If for example, you have csv data in which the fields are delimited by a pipe character "|"
or any other character instead of a comma, as shown below:
first_name|last_name|age|gender
John|Main|30|male
Jane|Doe|24|female
Morrison|Smith|22|male
Brian|Gates|32|male
Mary|Reagan|25|female
You can register a new dialect using the csv.register_dialect()
. You can then use the registered dialect with reader
and writer
objects.
import csv
data = """first_name|last_name|age|gender
John|Main|30|male
Jane|Doe|24|female
Morrison|Smith|22|male
Brian|Gates|32|male
Mary|Reagan|25|female"""
csv.register_dialect('pipe', delimiter = '|')
print("Dialects: ", csv.list_dialects())
reader = csv.reader(data.splitlines(), dialect = 'pipe') #use the dialect
for row in reader:
print(row)
Apart from delimiter
, you can specify other dialect parameters depending on the nature of your csv data. The supported parameters are shown in the following tables:
parameter | default value | description |
---|---|---|
delimiter |
, |
Field separator. |
quotechar |
" |
The character to enclose fields where quoting is allowed. |
lineterminator |
\r \n |
The string/character used to terminate a line |
doublequote |
True |
Whether quotchar parameters are doubled. |
escapechar |
None |
Character used to indicate an escape sequence |
skipintialspace |
False |
Ignore whitespace after the delimiter |
Automatic dialect detection
In cases where the csv format is not well known, we can use the csv.Sniffer
class, which automatically constructs a dialect object given a sample of csv data.
Consider if we have the following csv data.
first_name&last_name&age&gender
John&Main&30&male
Jane&Doe&24&female
Morrison&Smith&22&male
Brian&Gates&32&male
Mary&Reagan&25&female
In the above example, we used the "&"
character as the delimiter.
import csv
data="""first_name&last_name&age&gender
John&Main&30&male
Jane&Doe&24&female
Morrison&Smith&22&male
Brian&Gates&32&male
Mary&Reagan&25&female"""
sniffer=csv.Sniffer()
dialect=sniffer.sniff(sample=data)
print("delimiter: ", dialect.delimiter)
reader = csv.reader(data.splitlines(), dialect = dialect)
for row in reader:
print(row)
As you can see, the sniffer correctly detected that "&"
is used as the delimiter character. However, you should keep in mind that the sniffer is not always correct, it can only be said to be making educated guesses.
Using named fields
You can use csv.DictReader
and csv.DictWriter
classes to use named fields when reading or writing csv data.
The two classes allows you to specify in advance a list of field names to alias the fields in the csv data.
csv.DictReader
The DictReader
class allows you to read rows with the fields having an alias name.
Consider the following example:
import csv
data = """first_name,last_name,age,gender
John,Main,30,male
Jane,Doe,24,female
Morrison,Smith,22,male
Brian,Gates,32,male
Mary,Reagan,25,female"""
fields = 'fname', 'lname', 'age', 'gender'
reader = csv.DictReader(data.splitlines(), fieldnames = fields)
for row in reader:
print(row['fname'], row['lname'])
Iterating through the DictReader
object will yield a list of dictionaries, where each dictionary represents a row in the csv data.
csv.DictWriter
DictWriter
objects writes data into a csv file with argument given as dictionaries. Consider the following example:
import csv
fields = 'fname', 'lname', 'age', 'gender'
data_to_wite = [
{'fname':'Stephen', 'lname':'King', 'age':32, 'gender':'male'},
{'age':22, 'lname':'Jones', 'fname':'Molly', 'gender':'female'},
]
with open('pynerds.txt', 'w') as file:
writer = csv.DictWriter(file, fieldnames = fields)
writer.writerows(data_to_wite)
As shown above, when using named fields, the order is not important as long as the field names are correct.
import csv
with open('demo.csv') as file:
print(file.read())
Stephen,King,32,male
Molly,Jones,22,female