faqts : Computers : Programming : Languages : Python : Common Problems : Files

+ Search
Add Entry AlertManage Folder Edit Entry Add page to http://del.icio.us/
Did You Find This Entry Useful?

20 of 24 people (83%) answered Yes
Recently 8 of 10 people (80%) answered Yes

Entry

How can I open an existing file and truncate it?

May 22nd, 2000 04:56
unknown unknown, David Goodger


Problem:

When I do the following ->

f = open('file', 'w')
f.truncate(3)
f.close()

the file becomes corrupted. It has the wished size, but it is all null
bytes.

Solution:

I assume that you want to open an existing file and truncate it, leaving 
the existing (3 bytes of) data.

From the Python Library Reference (1.5.2p2), section 2.8, Built-in
Functions:

    open (filename[, mode[, bufsize]])

    Return a new file object ... The first two arguments are the same
    as for stdio's fopen(): ... mode indicates how the file is to be
    opened: 'r' for reading, 'w' for writing (truncating an existing
    file), and 'a' opens it for appending (which on some Unix systems
    means that all writes append to the end of the file, regardless of
    the current seek position).

    Modes 'r+', 'w+' and 'a+' open the file for updating (note that
    'w+' truncates the file). ...

I think your problem may be in your interpretation of the word 
"truncating" under the 'w' mode above. Since Python's open() modes are 
the same as stdio's fopen() in C, the following is relevant. From The C 
Programming Language, Second Edition (Kernighan & Ritchie), Appendix B, 
section B1.1, (<stdio.h>) File Operations:

    FILE *fopen ...
    
    ... Legal values for mode include:
    
        ...
        "w"        create text file for writing; discard previous
                contents if any
        ...
        "w+"    create text file for update; discard previous
                contents if any
        "a+"    append; open or create text file for update, writing
                at end
    
    Update mode permits reading and writing the same file; ...

So in the Python docs, "truncating" is used, whereas in C, it's "discard
previous contents if any". A bit misleading. Basically, modes "w" and 
"w+" both create a *new* file, clobbering/erasing any existing file with 
the same name. So your f.truncate(3) makes perfect sense: it sets the 
file length of a *new* file to 3 bytes, and since it *is* a new file, 
it's reasonable (downright friendly, in my opinion!) to put nulls in 
there.

I think what you want is to use the mode "a+" for a text file, or "a+b" 
for a binary file:

    >>> f1 = open("tmpfile", "w")
    >>> f1.write("123456789")
    >>> f1.close()
    >>> print open("tmpfile", "r").read()
    123456789
    >>> f2 = open("tmpfile", "a+")
    >>> f2.truncate(3)
    >>> f2.close()
    >>> print open("tmpfile", "r").read()
    123