linux sort 多列正排序,倒排序

轉載:https://segmentfault.com/a/1190000005713784
sort是在Linux裏很是經常使用的一個命令,管排序segmentfault

sort將文件的每一行做爲一個單位,相互比較,比較原則是從首字符向後,依次按ASCII碼值進行比較,最後將他們按升序輸出。
使用方法:sort [選項]... [文件]...
長選項必須用的參數在使用短選項時也是必須的。順序選項:
-b, --ignore-leading-blanks ignore leading blanks
-d, --dictionary-order consider only blanks and alphanumeric characters
-f, --ignore-case fold lower case to upper case characters
-g, --general-numeric-sort compare according to general numerical value
-i, --ignore-nonprinting consider only printable characters
-M, --month-sort compare (unknown) < `JAN' < ... < `DEC'
-n, --numeric-sort compare according to string numerical value
-r, --reverse reverse the result of comparisons

Other options:

-c, --check check whether input is sorted; do not sort
-k, --key=POS1[,POS2] start a key at POS1, end it at POS2 (origin 1)
-m, --merge merge already sorted files; do not sort
-o, --output=FILE write result to FILE instead of standard output
-s, --stable stabilize sort by disabling last-resort comparison
-S, --buffer-size=SIZE use SIZE for main memory buffer
-t, --field-separator=SEP use SEP instead of non-blank to blank transition
-T, --temporary-directory=DIR use DIR for temporaries, not $TMPDIR or/tmp;
multiple options specify multiple directories
-u, --unique with -c, check for strict ordering;
without -c, output only the first of an equal run
-z, --zero-terminated end lines with 0 byte, not newline
--help 顯示此幫助信息並退出
--version 輸出版本號信息並退出

sort的-u選項

在輸出行中去除重複行跨域

[ericshenMacPro@root duweixin]$ cat duweixin.net.txt
banana
apple
pear
orange
pear
[ericshenMacPro@root duweixin]$ sort duweixin.net.txt
apple
banana
orange
pear
pear
[ericshenMacPro@root duweixin]$ sort -u duweixin.net.txt
apple
banana
orange
pear

pear因爲重複被-u選項無情的刪除了。bash

sort的-r選項

sort默認的排序方式是升序,若是想改爲降序,就加個-r。app

[ericshenMacPro@root duweixin]$ cat duweixin.net.txt
1
3
5
2
4
[ericshenMacPro@root duweixin]$ sort duweixin.net.txt
1
2
3
4
5
#倒序加-r
[ericshenMacPro@root duweixin]$ sort -r duweixin.net.txt
5
4
3
2
1

sort的-o選項

因爲sort默認是把結果輸出到標準輸出,因此須要用重定向才能將結果寫入文件,形如sort oldfile > newfile
可是,若是你想把排序結果輸出到原文件中,用重定向可就不行了。
```
[ericshenMacPro@root duweixin]sort -r duweixin.txt > duweixin.txt [ericshenMacPro@root duweixin] cat duweixin.txt
[ericshenMacPro@root duweixin]$
看,居然將duweixin清空了。ide

加上-o選項,解決了這個問題,讓你放心的將結果寫入原文件。學習

[ericshenMacPro@root duweixin]$ cat number.txt
1
3
5
2
4
[ericshenMacPro@root duweixin]$ sort -r number.txt -o number.txt
[ericshenMacPro@root duweixin]$ cat number.txt
5
4
3
2
1

sort的-n選項

你有沒有遇到過10比2小的狀況。我反正遇到過。出現這種狀況是因爲排序程序將這些數字按字符來排序了,排序程序會先比較1和2,顯然1小,因此就將10放在2前面嘍。這也是sort的一向做風。
咱們若是想改變這種現狀,就要使用-n選項,來告訴sort,「要以數值來排序」!google

[ericshenMacPro@root duweixin]$ cat duweixin.net.txt
1
10
19
11
2
5
[ericshenMacPro@root duweixin]$ sort duweixin.net.txt
1
10
11
19
2
5
[ericshenMacPro@root duweixin]$ sort -n duweixin.net.txt
1
2
5
10
11
19

sort的-t選項和-k選項

若是有一個文件的內容是這樣:.net

[ericshenMacPro@root duweixin]$ cat facebook.txt
banana:30:5.5
apple:10:2.5
pear:90:2.3
orange:20:3.4

這個文件有三列,列與列之間用冒號隔開了,第一列表示水果類型,第二列表示水果數量,第三列表示水果價格。
那麼我想以水果數量來排序,也就是以第二列來排序,如何利用sort實現?
幸虧,sort提供了-t選項,後面能夠設定間隔符。(是否是想起了cut和paste的-d選項,共鳴~~)
指定了間隔符以後,就能夠用-k來指定列數了。code

[ericshenMacPro@root duweixin]$ sort -n -k 2 -t : facebook.txt
apple:10:2.5
orange:20:3.4
banana:30:5.5
pear:90:2.3

咱們使用冒號做爲間隔符,並針對第二列來進行數值升序排序,結果很使人滿意。
其餘的sort經常使用選項排序

-f會將小寫字母都轉換爲大寫字母來進行比較,亦即忽略大小寫
-c會檢查文件是否已排好序,若是亂序,則輸出第一個亂序的行的相關信息,最後返回1
-C會檢查文件是否已排好序,若是亂序,不輸出內容,僅返回1
-M會以月份來排序,好比JAN小於FEB等等
-b會忽略每一行前面的全部空白部分,從第一個可見字符開始比較。

有時候學習腳本,你會發現sort命令後面跟了一堆相似-k1,2,或者-k1.2
-k3.4的東東,有些匪夷所思。今天,咱們就來搞定它—-k選項!

多列排序

準備素材

第一個域是公司名稱,第二個域是公司人數,第三個域是員工平均工資。

$ cat duweixin.net.txt
google 110 5000
baidu 100 5000
guge 50 3000
sohu 100 4500

我想讓這個文件按公司的字母順序排序,也就是按第一個域進行排序:(這個duweixin.net.txt文件有三個域)

$ sort -t ‘ ‘ -k 1 duweixin.net.txt
baidu 100 5000
google 110 5000
guge 50 3000
sohu 100 4500

就直接用-k 1設定就能夠了。(其實此處並不嚴格,稍後你就會知道)

我想讓duweixin.net.txt按照公司人數排序

$ sort -n -t ‘ ‘ -k 2 duweixin.net.txt
guge 50 3000
baidu 100 5000
sohu 100 4500
google 110 5000

可是,此處出現了問題,那就是baidu和sohu的公司人數相同,都是100人,這個時候怎麼辦呢?按照默認規矩,是從第一個域開始進行升序排序,所以baidu排在了sohu前面。

我想讓facebook.txt按照公司人數排序 ,人數相同的按照員工平均工資升序排序:

$ sort -n -t ‘ ‘ -k 2 -k 3 duweixin.net.txt
guge 50 3000
sohu 100 4500
baidu 100 5000
google 110 5000

加了一個-k2 -k3就解決了問題。對滴,sort支持這種設定,就是說設定域排序的優先級,先以第2個域進行排序,若是相同,再以第3個域進行排序。

我想讓facebook.txt按照員工工資降序排序,若是員工人數相同的,則按照公司人數升序排序

$ sort -n -t ‘ ‘ -k 3r -k 2 duweixin.net.txt
baidu 100 5000
google 110 5000
sohu 100 4500
guge 50 3000

此處有使用了一些小技巧,你仔細看看,在-k 3後面偷偷加上了一個小寫字母r。你想一想,再結合咱們上一篇文章,能獲得答案麼?揭曉:r和-r選項的做用是同樣的,就是表示逆序。由於sort默認是按照升序排序的,因此此處須要加上r表示第三個域(員工平均工資)是按照降序排序。此處你還能夠加上n,就表示對這個域進行排序時,要按照數值大小進行排序,舉個例子吧:

$ sort -t ‘ ‘ -k 3nr -k 2n duweixin.net.txt
baidu 100 5000
google 110 5000
sohu 100 4500
guge 50 3000

去掉了最前面的-n選項,而是將它加入到了每個-k選項中了。

-k選項的具體語法格式

要繼續往下深刻的話,就不得不來點理論知識。你須要瞭解-k選項的語法格式,以下:

[ FStart [ .CStart ] ] [ Modifier ] [ , [ FEnd [ .CEnd ] ][ Modifier ] ]

這個語法格式能夠被其中的逗號(「,」)分爲兩大部分,Start部分和End部分。
先給你灌輸一個思想,那就是「若是不設定End部分,那麼就認爲End被設定爲行尾」。這個概念很重要的,但每每你不會重視它。
Start部分也由三部分組成,其中的Modifier部分就是咱們以前說過的相似n和r的選項部分。咱們重點說說Start部分的FStart和C.Start。
C.Start也是能夠省略的,省略的話就表示從本域的開頭部分開始。以前例子中的-k 2和-k 3就是省略了C.Start的例子嘍。
FStart.CStart,其中FStart就是表示使用的域,而CStart則表示在FStart域中從第幾個字符開始算「排序首字符」。
同理,在End部分中,你能夠設定FEnd.CEnd,若是你省略.CEnd,則表示結尾到「域尾」,即本域的最後一個字符。或者,若是你將CEnd設定爲0(零),也是表示結尾到「域尾」。

從公司英文名稱的第二個字母開始進行排序:

$ sort -t ‘ ‘ -k 1.2 duweixin.net.txt
baidu 100 5000
sohu 100 4500
google 110 5000
guge 50 3000

使用了-k 1.2,這就表示對第一個域的第二個字符開始到本域的最後一個字符爲止的字符串進行排序。你會發現baidu由於第二個字母是a而名列榜首。sohu和 google第二個字符都是o,但sohu的h在google的o前面,因此二者分別排在第二和第三。guge只能屈居第四了。

只針對公司英文名稱的第二個字母進行排序,若是相同的按照員工工資進行降序排序:

$ sort -t ‘ ‘ -k 1.2,1.2 -k 3,3nr duweixin.net.txt
baidu 100 5000
google 110 5000
sohu 100 4500
guge 50 3000

因爲只對第二個字母進行排序,因此咱們使用了-k 1.2,1.2的表示方式,表示咱們「只」對第二個字母進行排序。(若是你問「我使用-k 1.2怎麼不行?」,固然不行,由於你省略了End部分,這就意味着你將對從第二個字母起到本域最後一個字符爲止的字符串進行排序)。對於員工工資進行排 序,咱們也使用了-k 3,3,這是最準確的表述,表示咱們「只」對本域進行排序,由於若是你省略了後面的3,就變成了咱們「對第3個域開始到最後一個域位置的內容進行排序」 了。

在modifier部分還能夠用到哪些選項?

能夠用到b、d、f、i、n 或 r。
其中n和r你確定已經很熟悉了。
b表示忽略本域的簽到空白符號。
d表示對本域按照字典順序排序(即,只考慮空白和字母)。
f表示對本域忽略大小寫進行排序。
i表示忽略「不可打印字符」,只針對可打印字符進行排序。(有些ASCII就是不可打印字符,好比a是報警,b是退格,n是換行,r是回車等等)
10、思考思考關於-k和-u聯合使用的例子:

$ cat duweixin.net.txt
google 110 5000
baidu 100 5000
guge 50 3000
sohu 100 4500

這是最原始的duweixin.net.txt文件。

$ sort -n -k 2 duweixin.net.txt
guge 50 3000
baidu 100 5000
sohu 100 4500
google 110 5000

$ sort -n -k 2 -u duweixin.net.txt
guge 50 3000
baidu 100 5000
google 110 5000

當設定以公司員工域進行數值排序,而後加-u後,sohu一行就被刪除了!原來-u只識別用-k設定的域,發現相同,就將後續相同的行都刪除。

$ sort  -k 1 -u duweixin.net.txt
baidu 100 5000
google 110 5000
guge 50 3000
sohu 100 4500

$ sort  -k 1.1,1.1 -u duweixin.net.txt
baidu 100 5000
google 110 5000
sohu 100 4500

這個例子也同理,開頭字符是g的guge就沒有幸免於難。

$ sort -n -k 2 -k 3 -u duweixin.net.txt
guge 50 3000
sohu 100 4500
baidu 100 5000
google 110 5000

咦!這裏設置了兩層排序優先級的狀況下,使用-u就沒有刪除任何行。原來-u是會權衡全部-k選項,將都相同的纔會刪除,只要其中有一級不一樣都不會輕易刪除的:)(不信,你能夠本身加一行sina 100 4500試試看)

最詭異的排序:

$ sort -n -k 2.2,3.1 duweixin.net.txt
guge 50 3000
baidu 100 5000
sohu 100 4500
google 110 5000

以第二個域的第二個字符開始到第三個域的第一個字符結束的部分進行排序。
第一行,會提取0 3,第二行提取00 5,第三行提取00 4,第四行提取10 5。
又由於sort認爲0小於00小於000小於0000….
所以0 3確定是在第一個。10 5確定是在最後一個。但爲何00 5卻在00 4前面呢?(你能夠本身作實驗思考一下。)
答案揭曉:原來「跨域的設定是個假象」,sort只會比較第二個域的第二個字符到第二個域的最後一個字符的部分,而不會把第三個域的開頭字符歸入比較範圍。當發現00和00相同時,sort就會自動比較第一個域去了。固然baidu在sohu前面了。用一個範例便可證明:

sort -n -k 2.2,3.1 -k 1,1r duweixin.net.txt
guge 50 3000
sohu 100 4500
baidu 100 5000
google 110 5000
相關文章
相關標籤/搜索