web-dev-qa-db-ja.com

ファイル内の行の順序を逆にするにはどうすればよいですか。

各行の内容を保持したまま、テキストファイル(または標準入力)の行の順序を逆にしたいのですが。

だから、すなわち、で始まる:

foo
bar
baz

で終わりたいのですが

baz
bar
foo

このための標準のUNIXコマンドラインユーティリティはありますか?

552
Scotty Allen

BSDテール:

tail -r myfile.txt

参照: FreeBSDNetBSDOpenBSD および OS X のマニュアルページ。

392
Jason Cohen

また言及する価値があります: tac (the、ahem、catの逆) coreutils の一部です。

あるファイルを別のファイルに反転する

tac a.txt > b.txt
1240

有名なsedのトリック :があります。

# reverse order of lines (emulates "tac")
# bug/feature in HHsed v1.5 causes blank lines to be deleted
sed '1!G;h;$!d'               # method 1
sed -n '1!G;h;$p'             # method 2

(説明:保留バッファーの先頭に非先頭行を追加し、行をスワップして保留バッファーを作成し、最後に行を出力します)

代わりに(より速い実行で) awk one-linersから

awk '{a[i++]=$0} END {for (j=i-1; j>=0;) print a[j--] }' file*

覚えていないのなら、

Perl -e 'print reverse <>'

GNUユーティリティを持つシステムでは、他の答えはより単純ですが、すべてがGNU/Linuxであるわけではありません...

147
ephemient

偶然vimを使っているのなら

:g/^/m0
53
DerMike
$ (tac 2> /dev/null || tail -r)

Linuxで動作するtacを試してください。それが機能しない場合は、BSDとOSXで動作するtail -rを使用してください。

38
DigitalRoss
tac <file_name>

例:

$ cat file1.txt
1
2
3
4
5

$ tac file1.txt
5
4
3
2
1
36
jins

コマンドの最後にput:| tac

tacは正確にあなたが求めていることをします、それは「各ファイルを標準出力に書き出す、最後の行を最初にする」です。

tacは猫の逆です:-)。

25

次のコマンドを試してください。

grep -n "" myfile.txt | sort -r -n | gawk -F : "{ print $2 }"
24
DG

Just Bash :)(4.0以降)

function print_reversed {
    local lines i
    readarray -t lines

    for (( i = ${#lines[@]}; i--; )); do
        printf '%s\n' "${lines[i]}"
    done
}

print_reversed < file
15
konsolebox

最も簡単な方法はtacコマンドを使用することです。 taccatの逆です。例:

$ cat order.txt
roger shah 
armin van buuren
fpga vhdl arduino c++ Java gridgain
$ tac order.txt > inverted_file.txt
$ cat inverted_file.txt
fpga vhdl arduino c++ Java gridgain
armin van buuren
roger shah 
11
Yekatandilburg

私は本当に " tail -r "の回答が好きですが、私のお気に入りのgawkの回答は....

gawk '{ L[n++] = $0 } 
  END { while(n--) 
        print L[n] }' file
9
Tim Menzies

_ edit _ 以下は、1から10までの数字のランダムにソートされたリストを生成します。

seq 1 10 | sort -R | tee /tmp/lst |cat <(cat /tmp/lst) <(echo '-------') **...**

ドットをリストを逆にする実際のコマンドに置き換えます。

tac

seq 1 10 | sort -R | tee /tmp/lst |cat <(cat /tmp/lst) <(echo '-------') \
<(tac)

python:sys.stdinで[:: - 1]を使用する

seq 1 10 | sort -R | tee /tmp/lst |cat <(cat /tmp/lst) <(echo '-------') \
<(python -c "import sys; print(''.join(([line for line in sys.stdin])[::-1]))")
2

シェルスクリプト内でtacを使用する可能性があるクロスOS(すなわちOSX、Linux)ソリューションでは、他の人が上記で言及したように自作を使用し、次にtacを別名で使用するだけです。

brew install coreutils
echo "alias tac='gtac'" >> ~/.bash_aliases (or wherever you load aliases)
source ~/.bash_aliases
tac myfile.txt
2
lacostenycoder

これはBSDとGNUの両方で動作します。

awk '{arr[i++]=$0} END {while (i>0) print arr[--i] }' filename
2
R. Kumar

ファイルを適切に修正したい場合は、次のコマンドを実行してください。

sed -i '1!G;h;$!d' filename

これにより、一時ファイルを作成してから元のファイルを削除または名前変更する必要がなくなり、同じ結果になります。例えば:

$tac file > file2
$sed -i '1!G;h;$!d' file
$diff file file2
$

Ephemientによる の答えに基づいて 、それは私が欲しかったことをほとんどしましたが、全くではありませんでした。

1
Mark Booth

最良の解決策:

tail -n20 file.txt | tac
1
ITNM

tail -r トリックを行います。

シーケンス1 20 | tail -r

0
Bohdan

面白いアイデアがたくさんあります。しかし、私の考えを試してみてください。テキストをこれにパイプします。

回転| tr '\ n' '〜' |回転| tr '〜' '\ n'

これは、文字「〜」がファイル内にないことを前提としています。これは、1961年に遡るすべてのUNIXシェルで動作するはずです。または、そのようなものです。

0
driver

Emacsユーザーの場合:C-x h(ファイル全体を選択)、次にM-x reverse-region。部分または線のみを選択して元に戻す場合にも機能します。

0
Marius Hofert

私も同じ質問をしましたが、最初の行(ヘッダー)も一番上になるようにしたいと思いました。だから私はawkの力を使う必要がありました

cat dax-weekly.csv | awk '1 { last = NR; line[last] = $0; } END { print line[1]; for (i = last; i > 1; i--) { print line[i]; } }'

PSはcygwinやgitbashでも動作します

0
KIC

vimstdinstdoutでそれを行うことができます。 exを使って POSIX準拠 にすることもできます。 vimexのビジュアルモードです。実際には、exvim -eまたはvim -E(改善されたexモード)と共に使用できます。 vimは、sedのようなツールとは異なり、編集のためにファイルをバッファリングしますが、ストリームにはsedが使用されるので便利です。あなたはawkを使うことができるかもしれませんが、手動ですべてを変数にバッファリングしなければならないでしょう。

アイデアは次のとおりです。

  1. Stdinから読む
  2. 各行について、1行目に移動します(反転)。コマンドはg/^/m0です。これは、各行gに対してグローバルに意味します。行の先頭に一致します。これは^に一致します。 1行目のm0であるアドレス0の後に移動します。
  3. すべて印刷するコマンドは%pです。これは、すべての行の範囲が%であることを意味します。 pという行を表示します。
  4. ファイルを保存せずに強制終了します。コマンドはq!です。これはqを終了することを意味します。強制的に!
# Generate a newline delimited sequence of 1 to 10
$ seq 10
1
2
3
4
5
6
7
8
9
10

# Use - to read from stdin.
# vim has a delay and annoying 'Vim: Reading from stdin...' output
# if you use - to read from stdin. Use --not-a-term to hide output.
# --not-a-term requires vim 8.0.1308 (Nov 2017)
# Use -E for improved ex mode. -e would work here too since I'm not
# using any improved ex mode features.
# each of the commands I explained above are specified with a + sign
# and are run sequentially.
$ seq 10 | vim - --not-a-term -Es +'g/^/m0' +'%p' +'q!'
10
9
8
7
6
5
4
3
2
1
# non improved ex mode works here too, -e.
$ seq 10 | vim - --not-a-term -es +'g/^/m0' +'%p' +'q!'

# If you don't have --not-a-term, use /dev/stdin
seq 10 | vim -E +'g/^/m0' +'%p' +'q!' /dev/stdin

# POSIX compliant (maybe)
# POSIX compliant ex doesn't allow using + sign to specify commands.
# It also might not allow running multiple commands sequentially.
# The docs say "Implementations may support more than a single -c"
# If yours does support multiple -c
$ seq 10 | ex -c "execute -c 'g/^/m0' -c '%p' -c 'q!' /dev/stdin

# If not, you can chain them with the bar, |. This is same as Shell
# piping. It's more like Shell semi-colon, ;.
# The g command consumes the |, so you can use execute to prevent that.
# Not sure if execute and | is POSIX compliant.
seq 10 | ex -c "execute 'g/^/m0' | %p | q!" /dev/stdin

これを再利用可能にする方法

私はvedを編集するためにvimを使用するためにsedstdinのようなvimエディタ)を呼び出すスクリプトを使用します。パスのvedというファイルにこれを追加します。

#!/usr/bin/env sh

vim - --not-a-term -Es "$@" +'%p | q!'

Vimでは10個のコマンドに制限されているので、私は+の代わりに+'%p' +'q!'コマンドを1つ使用しています。それでそれらをマージすることは"$@"が8の代わりに9の+コマンドを持つことを可能にします。

それからあなたはすることができます:

seq 10 | ved +'g/^/m0'

もしあなたがvim 8を持っていないなら、代わりにこれをvedに入れてください:

#!/usr/bin/env sh

vim -E "$@" +'%p | q!' /dev/stdin
0
dosentmatter