Я хотел бы изменить порядок строк в текстовом файле (или stdin), сохраняя содержимое каждой строки.
Итак, то есть, начиная с:
foo
bar
baz
Я хочу закончить с
baz
bar
foo
Существует ли для этого стандартная утилита командной строки UNIX?
Я хотел бы изменить порядок строк в текстовом файле (или stdin), сохраняя содержимое каждой строки.
Итак, то есть, начиная с:
foo
bar
baz
Я хочу закончить с
baz
bar
foo
Существует ли для этого стандартная утилита командной строки UNIX?
BSD tail:
tail -r myfile.txt
Ссылка: FreeBSD, NetBSD, OpenBSD и OS X страницы руководства.
Также стоит упомянуть: tac
(the, ahem, reverse от cat
). Часть coreutils.
tac a.txt > b.txt
Там известные 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...
Если вы оказались в vim
, используйте
:g/^/m0
$ (tac 2> /dev/null || tail -r)
Попробуйте tac
, который работает в Linux, и если это не сработает, используйте tail -r
, который работает на BSD и OSX.
tac <file_name>
Пример:
$ cat file1.txt
1
2
3
4
5
$ tac file1.txt
5
4
3
2
1
в конце вашей команды поставьте: | tac
| tac
tac делает именно то, что вы просите, он записывает каждый ФАЙЛ в стандартный вывод, в последнюю строку первым.
Так это противоположность кошки :-).
Попробуйте выполнить следующую команду:
grep -n "" myfile.txt | sort -r -n | gawk -F : "{ print $2 }"
Просто Баш :) (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
Самый простой способ - использовать команду 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
Мне очень нравится ответ "tail -r", но мой любимый ответ gawk -....
gawk '{ L[n++] = $0 }
END { while(n--)
print L[n] }' file
ИЗМЕНИТЬ следующее генерирует беспорядочно отсортированный список чисел от 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]))")
Для решения перекрестных ОС (например, 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
Это будет работать как на BSD, так и на GNU.
awk '{arr[i++]=$0} END {while (i>0) print arr[--i] }' filename
Лучшее решение:
tail -n20 file.txt | tac
Если вы хотите изменить файл на месте, вы можете запустить
sed -i '1!G;h;$!d' filename
Это избавляет от необходимости создавать временный файл, а затем удалять или переименовывать оригинал и имеет тот же результат. Например:
$tac file > file2
$sed -i '1!G;h;$!d' file
$diff file file2
$
На основании ответа от ephemient, который сделал почти, но не совсем то, что я хотел.
Для пользователей Emacs: Cx h
(выбрать весь файл), а затем Mx reverse-region
. Также работает только для выбора частей или линий и их возврата.
Я вижу много интересных идей. Но попробуй мою идею. Направьте ваш текст в это:
рев | tr '\n' '~' | рев | tr '~' '\n'
который предполагает, что символ '~' отсутствует в файле.
У меня был тот же вопрос, но я также хотел, чтобы первая строка (заголовок) оставалась на вершине. Поэтому мне нужно было использовать мощность 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
Вы можете сделать это с помощью vim
stdin
и stdout
. Вы также можете использовать ex
чтобы быть POSIX-совместимым. vim
это просто визуальный режим для ex
. Фактически, вы можете использовать ex
с vim -e
или vim -e
(улучшенный режим ex
). vim
полезен, потому что в отличие от таких инструментов, как sed
он буферизует файл для редактирования, а sed
используется для потоков. Возможно, вы сможете использовать awk
, но вам придется вручную буферизовать все в переменной.
Идея состоит в том, чтобы сделать следующее:
g/^/m0
. Это означает глобально для каждой строки g
; соответствует началу строки, которая соответствует чему-либо ^
; переместите его после адреса 0, который является строкой 1 m0
.%p
. Это означает для диапазона всех линий %
; выведите строку p
.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
tail -r работает в большинстве систем Linux и MacOS
сек 1 20 | хвост -r
sort -r < filename
или
rev < filename