Как изменить порядок строк в файле?

Я хотел бы изменить порядок строк в текстовом файле (или stdin), сохраняя содержимое каждой строки.

Итак, то есть, начиная с:

foo
bar
baz

Я хочу закончить с

baz
bar
foo

Существует ли для этого стандартная утилита командной строки UNIX?

Ответ 1

BSD tail:

tail -r myfile.txt

Ссылка: FreeBSD, NetBSD, OpenBSD и OS X страницы руководства.

Ответ 2

Также стоит упомянуть: tac (the, ahem, reverse от cat). Часть coreutils.

Переключение одного файла на другой

tac a.txt > b.txt

Ответ 3

Там известные 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:

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

Если вы не можете вспомнить это,

perl -e 'print reverse <>'

В системе с утилитами GNU другие ответы проще, но не во всем мире есть GNU/Linux...

Ответ 4

Если вы оказались в vim, используйте

:g/^/m0

Ответ 5

$ (tac 2> /dev/null || tail -r)

Попробуйте tac, который работает в Linux, и если это не сработает, используйте tail -r, который работает на BSD и OSX.

Ответ 6

tac <file_name>

Пример:

$ cat file1.txt
1
2
3
4
5

$ tac file1.txt
5
4
3
2
1

Ответ 7

в конце вашей команды поставьте: | tac | tac

tac делает именно то, что вы просите, он записывает каждый ФАЙЛ в стандартный вывод, в последнюю строку первым.

Так это противоположность кошки :-).

Ответ 8

Попробуйте выполнить следующую команду:

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

Ответ 9

Просто Баш :) (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

Ответ 10

Самый простой способ - использовать команду tac. tac является cat обратным. Пример:

$ 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

Мне очень нравится ответ "tail -r", но мой любимый ответ gawk -....

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

Ответ 12

ИЗМЕНИТЬ следующее генерирует беспорядочно отсортированный список чисел от 1 до 10:

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

где точки заменяются фактической командой, которая меняет список

нолики

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

python: используя [:: - 1] в sys.stdin

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]))")

Ответ 13

Для решения перекрестных ОС (например, OSX, Linux), которые могут использовать tac внутри оболочки script, используйте homebrew, как упоминалось выше, а затем просто псевдоним tac, например:

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

Ответ 14

Это будет работать как на BSD, так и на GNU.

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

Ответ 15

Лучшее решение:

tail -n20 file.txt | tac

Ответ 16

Если вы хотите изменить файл на месте, вы можете запустить

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

Это избавляет от необходимости создавать временный файл, а затем удалять или переименовывать оригинал и имеет тот же результат. Например:

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

На основании ответа от ephemient, который сделал почти, но не совсем то, что я хотел.

Ответ 17

Для пользователей Emacs: Cx h (выбрать весь файл), а затем Mx reverse-region. Также работает только для выбора частей или линий и их возврата.

Ответ 18

Я вижу много интересных идей. Но попробуй мою идею. Направьте ваш текст в это:

рев | tr '\n' '~' | рев | tr '~' '\n'

который предполагает, что символ '~' отсутствует в файле.

Ответ 19

У меня был тот же вопрос, но я также хотел, чтобы первая строка (заголовок) оставалась на вершине. Поэтому мне нужно было использовать мощность 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

Ответ 20

Вы можете сделать это с помощью vim stdin и stdout. Вы также можете использовать ex чтобы быть POSIX-совместимым. vim это просто визуальный режим для ex. Фактически, вы можете использовать ex с vim -e или vim -e (улучшенный режим ex). vim полезен, потому что в отличие от таких инструментов, как sed он буферизует файл для редактирования, а sed используется для потоков. Возможно, вы сможете использовать awk, но вам придется вручную буферизовать все в переменной.

Идея состоит в том, чтобы сделать следующее:

  1. Читать со стандартного ввода
  2. Для каждой строки переместите ее в строку 1 (для реверса). Команда g/^/m0. Это означает глобально для каждой строки g; соответствует началу строки, которая соответствует чему-либо ^; переместите его после адреса 0, который является строкой 1 m0.
  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 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 editor как sed), чтобы использовать vim для редактирования stdin. Добавьте это в файл с именем ved на вашем пути:

#!/usr/bin/env sh

vim - --not-a-term -Es "[email protected]" +'%p | q!'

Я использую одну команду + вместо +'%p' +'q!' потому что vim ограничивает вас до 10 команд. Таким образом, объединение их позволяет "[email protected]" иметь 9 + команд вместо 8.

Тогда вы можете сделать:

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

Если у вас нет vim 8, поместите это в ved:

#!/usr/bin/env sh

vim -E "[email protected]" +'%p | q!' /dev/stdin

Ответ 21

tail -r работает в большинстве систем Linux и MacOS

сек 1 20 | хвост -r

Ответ 22

sort -r < filename

или

rev < filename