Python file.tell gives wrong value location

前端 未结 2 424
没有蜡笔的小新
没有蜡笔的小新 2021-01-03 07:03

I am trying to extract a number of locations from an existing file using Python. This is my current code for extracting the locations:

    self.fh = open( f         


        
相关标签:
2条回答
  • 2021-01-03 07:27

    The cause is (rather obscurely) explained in the docs for a file object's next() method:

    When a file is used as an iterator, typically in a for loop (for example, for line in f: print line), the next() method is called repeatedly. This method returns the next input line, or raises StopIteration when EOF is hit. In order to make a for loop the most efficient way of looping over the lines of a file (a very common operation), the next() method uses a hidden read-ahead buffer. As a consequence of using a read-ahead buffer, combining next() with other file methods (like readline()) does not work right. However, using seek() to reposition the file to an absolute position will flush the read-ahead buffer.

    The values returned by tell() reflect how far this hidden read-ahead buffer has gotten, which will typically be up to a few thousand bytes beyond the characters your program has actually retrieved.

    There's no portable way around this. If you need to mix tell() with reading lines, then use the file's readline() method instead. The tradeoff is that, in return for getting usable tell() results, iterating over a large file with readline() is typically significantly slower than using for line in file_object:.

    Code

    Concretely, change the loop to this:

    line = self.fh.readline()
    while line:
        if p.search(line):
            self.porSnipStartFPtr = self.fh.tell()
            sys.stdout.write("found regPorSnip")
        line = fh.readline()
    

    I'm not sure that's what you really want, though: tell() is capturing the position of the start of the next line. If want the position of the start of the line, then you need to change the logic, like so:

    pos = self.fh.tell()
    line = self.fh.readline()
    while line:
        if p.search(line):
            self.porSnipStartFPtr = pos
            sys.stdout.write("found regPorSnip")
        pos = self.fh.tell()
        line = fh.readline()
    

    or do it with a "loop and a half":

    while True:
        pos = self.fh.tell()
        line = self.fh.readline()
        if not line:
            break
        if p.search(line):
            self.porSnipStartFPtr = pos
            sys.stdout.write("found regPorSnip")
    
    0 讨论(0)
  • 2021-01-03 07:30

    I guess I dont understand the issue

    >>> fh = open('test.txt')
    >>> fh.tell()
    0L
    >>> fh.read(1)
    '"'
    >>> fh.tell()
    1L
    >>> fh.read(5)
    'a" \n"'
    >>> fh.tell()
    7L
    
    0 讨论(0)
提交回复
热议问题