Cheap way to search a large text file for a string

后端 未结 9 1033
隐瞒了意图╮
隐瞒了意图╮ 2020-11-27 04:15

I need to search a pretty large text file for a particular string. Its a build log with about 5000 lines of text. Whats the best way to go about doing that? Using regex sho

相关标签:
9条回答
  • 2020-11-27 04:49

    If it is "pretty large" file, then access the lines sequentially and don't read the whole file into memory:

    with open('largeFile', 'r') as inF:
        for line in inF:
            if 'myString' in line:
                # do_something
    
    0 讨论(0)
  • 2020-11-27 04:52

    I've had a go at putting together a multiprocessing example of file text searching. This is my first effort at using the multiprocessing module; and I'm a python n00b. Comments quite welcome. I'll have to wait until at work to test on really big files. It should be faster on multi core systems than single core searching. Bleagh! How do I stop the processes once the text has been found and reliably report line number?

    import multiprocessing, os, time
    NUMBER_OF_PROCESSES = multiprocessing.cpu_count()
    
    def FindText( host, file_name, text):
        file_size = os.stat(file_name ).st_size 
        m1 = open(file_name, "r")
    
        #work out file size to divide up to farm out line counting
    
        chunk = (file_size / NUMBER_OF_PROCESSES ) + 1
        lines = 0
        line_found_at = -1
    
        seekStart = chunk * (host)
        seekEnd = chunk * (host+1)
        if seekEnd > file_size:
            seekEnd = file_size
    
        if host > 0:
            m1.seek( seekStart )
            m1.readline()
    
        line = m1.readline()
    
        while len(line) > 0:
            lines += 1
            if text in line:
                #found the line
                line_found_at = lines
                break
            if m1.tell() > seekEnd or len(line) == 0:
                break
            line = m1.readline()
        m1.close()
        return host,lines,line_found_at
    
    # Function run by worker processes
    def worker(input, output):
        for host,file_name,text in iter(input.get, 'STOP'):
            output.put(FindText( host,file_name,text ))
    
    def main(file_name,text):
        t_start = time.time()
        # Create queues
        task_queue = multiprocessing.Queue()
        done_queue = multiprocessing.Queue()
        #submit file to open and text to find
        print 'Starting', NUMBER_OF_PROCESSES, 'searching workers'
        for h in range( NUMBER_OF_PROCESSES ):
            t = (h,file_name,text)
            task_queue.put(t)
    
        #Start worker processes
        for _i in range(NUMBER_OF_PROCESSES):
            multiprocessing.Process(target=worker, args=(task_queue, done_queue)).start()
    
        # Get and print results
    
        results = {}
        for _i in range(NUMBER_OF_PROCESSES):
            host,lines,line_found = done_queue.get()
            results[host] = (lines,line_found)
    
        # Tell child processes to stop
        for _i in range(NUMBER_OF_PROCESSES):
            task_queue.put('STOP')
    #        print "Stopping Process #%s" % i
    
        total_lines = 0
        for h in range(NUMBER_OF_PROCESSES):
            if results[h][1] > -1:
                print text, 'Found at', total_lines + results[h][1], 'in', time.time() - t_start, 'seconds'
                break
            total_lines += results[h][0]
    
    if __name__ == "__main__":
        main( file_name = 'testFile.txt', text = 'IPI1520' )
    
    0 讨论(0)
  • 2020-11-27 04:54

    5000 lines isn't big (well, depends on how long the lines are...)

    Anyway: assuming the string will be a word and will be seperated by whitespace...

    lines=open(file_path,'r').readlines()
    str_wanted="whatever_youre_looking_for"
    
    
        for i in range(len(lines)):
            l1=lines.split()
            for p in range(len(l1)):
                if l1[p]==str_wanted:
                    #found
                    # i is the file line, lines[i] is the full line, etc.
    
    0 讨论(0)
  • 2020-11-27 04:57

    I like the solution of Javier. I did not try it, but it sounds cool!

    For reading through a arbitary large text and wanting to know it a string exists, replace it, you can use Flashtext, which is faster than Regex with very large files.

    Edit:

    From the developer page:

    >>> from flashtext import KeywordProcessor
    >>> keyword_processor = KeywordProcessor()
    >>> # keyword_processor.add_keyword(<unclean name>, <standardised name>)
    >>> keyword_processor.add_keyword('Big Apple', 'New York')
    >>> keyword_processor.add_keyword('Bay Area')
    >>> keywords_found = keyword_processor.extract_keywords('I love Big Apple and Bay Area.')
    >>> keywords_found
    >>> # ['New York', 'Bay Area']
    

    Or when extracting the offset:

    >>> from flashtext import KeywordProcessor
    >>> keyword_processor = KeywordProcessor()
    >>> keyword_processor.add_keyword('Big Apple', 'New York')
    >>> keyword_processor.add_keyword('Bay Area')
    >>> keywords_found = keyword_processor.extract_keywords('I love big Apple and Bay Area.', span_info=True)
    >>> keywords_found
    >>> # [('New York', 7, 16), ('Bay Area', 21, 29)]
    
    0 讨论(0)
  • 2020-11-27 05:02

    I'm surprised no one mentioned mapping the file into memory: mmap

    With this you can access the file as if it were already loaded into memory and the OS will take care of mapping it in and out as possible. Also, if you do this from 2 independent processes and they map the file "shared", they will share the underlying memory.

    Once mapped, it will behave like a bytearray. You can use regular expressions, find or any of the other common methods.

    Beware that this approach is a little OS specific. It will not be automatically portable.

    0 讨论(0)
  • 2020-11-27 05:03

    The following function works for textfiles and binary files (returns only position in byte-count though), it does have the benefit to find strings even if they would overlap a line or buffer and would not be found when searching line- or buffer-wise.

    def fnd(fname, s, start=0):
        with open(fname, 'rb') as f:
            fsize = os.path.getsize(fname)
            bsize = 4096
            buffer = None
            if start > 0:
                f.seek(start)
            overlap = len(s) - 1
            while True:
                if (f.tell() >= overlap and f.tell() < fsize):
                    f.seek(f.tell() - overlap)
                buffer = f.read(bsize)
                if buffer:
                    pos = buffer.find(s)
                    if pos >= 0:
                        return f.tell() - (len(buffer) - pos)
                else:
                    return -1
    

    The idea behind this is:

    • seek to a start position in file
    • read from file to buffer (the search strings has to be smaller than the buffer size) but if not at the beginning, drop back the - 1 bytes, to catch the string if started at the end of the last read buffer and continued on the next one.
    • return position or -1 if not found

    I used something like this to find signatures of files inside larger ISO9660 files, which was quite fast and did not use much memory, you can also use a larger buffer to speed things up.

    0 讨论(0)
提交回复
热议问题