I have a huge tab-separated file formatted like this
X column1 column2 column3
row1 0 1 2
row2 3 4 5
row3 6 7 8
row4 9 10 11
I would like to transpose it in an efficient way using only bash commands (I could write a ten or so lines Perl script to do that, but it should be slower to execute than the native bash functions). So the output should look like
X row1 row2 row3 row4
column1 0 3 6 9
column2 1 4 7 10
column3 2 5 8 11
I thought of a solution like this
cols=`head -n 1 input | wc -w`
for (( i=1; i <= $cols; i++))
do cut -f $i input | tr $'\n' $'\t' | sed -e "s/\t$/\n/g" >> output
done
But it's slow and doesn't seem the most efficient solution. I've seen a solution for vi in this post, but it's still over-slow. Any thoughts/suggestions/brilliant ideas? :-)
Have a look at GNU datamash which can be used like
datamash transpose
. A future version will also support cross tabulation (pivot tables)output
Performance against Perl solution by Jonathan on a 10000 lines file
EDIT by Ed Morton (@ghostdog74 feel free to delete if you disapprove).
Maybe this version with some more explicit variable names will help answer some of the questions below and generally clarify what the script is doing. It also uses tabs as the separator which the OP had originally asked for so it'd handle empty fields and it coincidentally pretties-up the output a bit for this particular case.
The above solutions will work in any awk (except old, broken awk of course - there YMMV).
The above solutions do read the whole file into memory though - if the input files are too large for that then you can do this:
which uses almost no memory but reads the input file once per number of fields on a line so it will be much slower than the version that reads the whole file into memory. It also assumes the number of fields is the same on each line and it uses GNU awk for
ENDFILE
andARGIND
but any awk can do the same with tests onFNR==1
andEND
.GNU datamash is perfectly suited for this problem with only one line of code and potentially arbitrarily large filesize!
Here is a moderately solid Perl script to do the job. There are many structural analogies with @ghostdog74's
awk
solution.With the sample data size, the performance difference between perl and awk was negligible (1 millisecond out of 7 total). With a larger data set (100x100 matrix, entries 6-8 characters each), perl slightly outperformed awk - 0.026s vs 0.042s. Neither is likely to be a problem.
Representative timings for Perl 5.10.1 (32-bit) vs awk (version 20040207 when given '-V') vs gawk 3.1.7 (32-bit) on MacOS X 10.5.8 on a file containing 10,000 lines with 5 columns per line:
Note that gawk is vastly faster than awk on this machine, but still slower than perl. Clearly, your mileage will vary.
If you only want to grab a single (comma delimited) line $N out of a file and turn it into a column:
Another
awk
solution and limited input with the size of memory you have.This joins each same filed number positon into together and in
END
prints the result that would be first row in first column, second row in second column, etc. Will output: