So we have these big data sets that people say are the code of life. In a simple representation they add up to about 3 GB of space. While it is quite possible to load up the whole human genome into RAM and have capacity to spare, I was wondering if there were painless ways to access compressed forms of the data. This lead me to Mikhail Korobov's blog and to bcolz. Here's a quick shoot out between plain Python strings, numpy and bcolz as used to store and access human chromosome 11 which is about 135 million base pairs long:
My take away, for my particular application is that I will stick to plain old Pythons strings - I tend to take short slices of the data (about 100 basepairs long) and that puts me in the range where Python strings are faster than numpy arrays. Memory is cheap.
Code follows:
My take away, for my particular application is that I will stick to plain old Pythons strings - I tend to take short slices of the data (about 100 basepairs long) and that puts me in the range where Python strings are faster than numpy arrays. Memory is cheap.
Code follows:
"""Access times for single characters along the length of the string and for 100 item slices along the length of the string.""" import pylab import numpy import bcolz chrom_11_str = None # Ugh, globals for timeit chrom_11_bcolz = None chrom_11_numpy = None def load_data(): global chrom_11_str, chrom_11_bcolz, chrom_11_numpy with open('chr11.fa', 'r') as fp: _ = fp.readline() # The header chrom_11_str = ''.join([ln.strip() for ln in fp.readlines()]) chrom_11_numpy = numpy.fromstring(chrom_11_str, dtype='c') chrom_11_bcolz = bcolz.carray(chrom_11_numpy) def time_things(): from timeit import timeit n_loops = 1000 return [[[timeit("a = {:s}[0:{:d}]".format(dta, n), "from __main__ import {:s}".format(dta), number=n_loops) / n_loops for n in [10 ** m for m in range(6)]], [timeit("a = {:s}[100000000:100000000+{:d}]".format(dta, n), "from __main__ import {:s}".format(dta), number=n_loops) / n_loops for n in [10 ** m for m in range(6)]]] for dta in ['chrom_11_str', 'chrom_11_bcolz', 'chrom_11_numpy']] def plot_things(t): #t_str, t_bcolz, t_numpy = t sl_size = [10 ** m for m in range(6)] location = ['(start)', '(end)'] for n in [0, 1]: pylab.subplot(1,2,n+1) for t_value, t_name in zip(t, ['str', 'bcolz', 'numpy']): pylab.plot(sl_size, t_value[n], label=t_name + location[n]) #pylab.plot(t_value[1], 'b', label=t_name + '(finish)') pylab.setp(pylab.gca(), xlabel='Slice size', ylabel='Time (s)', xscale='log', yscale='log', ylim=[1e-9, 1e-3]) pylab.legend(loc='lower right') load_data() t = time_things() plot_things(t)
Comments
Post a Comment