it-swarm-ru.tech

Как получить обратное поведение для `tail` и` head`?

Есть ли способ head/tail документа и получить обратный вывод; потому что вы не знаете, сколько строк в документе?

То есть Я просто хочу получить все, кроме первых двух строк foo.txt, чтобы добавить к другому документу.

57
chrisjlee

Вы можете использовать это, чтобы убрать первую две строки:

tail -n +3 foo.txt

и это, чтобы убрать последнюю две строки:

head -n -2 foo.txt

(при условии, что файл заканчивается на \n для последнего)


Как и при стандартном использовании tail и ​​head, эти операции не являются разрушительными. Используйте >out.txt если вы хотите перенаправить вывод в новый файл:

tail -n +3 foo.txt >out.txt

В случае out.txt уже существует, он перезапишет этот файл. Используйте >>out.txt вместо >out.txt если вы хотите, чтобы вывод был добавлен к out.txt.

59
Stéphane Gimenez

Если вы хотите все, кроме первых N-1 строк, вызовите tail с количеством строк +N. (Число - это номер первой строки, которую вы хотите сохранить, начиная с 1, т.е. +1 означает начало сверху, +2 означает пропуск одной строки и т.д.).

tail -n +3 foo.txt >>other-document

Нет простого, портативного способа пропустить последние N строк. GNU head позволяет head -n +N как аналог tail -n +N. В противном случае, если у вас есть tac (например, GNU или Busybox)), вы можете объединить его с tail:

tac | tail -n +3 | tac

В частности, вы можете использовать фильтр awk (не проверено):

awk -vskip=2 '{
    lines[NR] = $0;
    if (NR > skip) print lines[NR-skip];
    delete lines[NR-skip];
}'

Если вы хотите удалить последние несколько строк из большого файла, вы можете определить смещение в байтах фрагмента для усечения, а затем выполнить усечение с помощью dd.

total=$(wc -c < /file/to/truncate)
chop=$(tail -n 42 /file/to/truncate | wc -c)
dd if=/dev/null of=/file/to/truncate seek=1 bs="$((total-chop))"

Вы не можете обрезать файл на месте в начале, но если вам нужно удалить первые несколько строк огромного файла, вы можете переместить содержимое .

9
Gilles 'SO- stop being evil'

Со страницы руководства tail (то есть GNU tail):

-n, --lines=K
   output the last K lines, instead of the last 10; or use -n +K to
   output lines starting with the Kth

Таким образом, следующее должно добавить все, кроме первых 2 строк somefile.txt до anotherfile.txt:

tail --lines=+3 somefile.txt >> anotherfile.txt
3
Steven Monday

Для удаления первых n строк GNU sed. Например, если n = 2

sed -n '1,2!p' input-file

! означает "исключить этот интервал". Как вы можете себе представить, более сложный результат может быть получен, например,

sed -n '3,5p;7p'

это покажет линию 3,4,5,7. Больше возможностей исходит от использования регулярных выражений вместо адресов.

Ограничение заключается в том, что номера строк должны быть известны заранее.

3
enzotib
_{   head -n2 >/dev/null
    cat  >> other_document
}   <infile
_

Если _<infile_ - обычный, lseek() - файл с возможностью, тогда да, во что бы то ни стало, не стесняйтесь. Выше приведена полностью поддерживаемая POSIXly конструкция.

2
mikeserv

Хотя tail -n +4 для вывода файла, начинающегося с 4-й строки (все, кроме первых 3 строк), является стандартным и переносимым, его head аналог (head -n -3, все, кроме последние 3 строки) нет.

Портативно, вы бы сделали:

sed '$d' | sed '$d' | sed '$d'

Или:

sed -ne :1 -e '1,3{N;b1' -e '}' -e 'P;N;D'

(имейте в виду, что в некоторых системах, где sed имеет шаблонное пространство ограниченного размера, оно не масштабируется до больших значений n).

Или:

awk 'NR>3 {print l[NR%3]}; {l[NR%3]=$0}'
1
Stéphane Chazelas

Вы можете использовать diff, чтобы сравнить вывод head/tail с исходным файлом, а затем удалить то же самое, получив обратное.

diff --unchanged-group-format='' foo.txt <(head -2 foo.txt)
1
sokoban

Мой подход похож на Жиля, но вместо этого я просто переворачиваю файл с помощью команды cat и pipe, используя команду head.

tac -r thefile.txt | head thisfile.txt (заменяет файлы)

0
Abe

Решение для BSD (macOS):

Удалить первые 2 строки:

tail -n $( echo "$(cat foo.txt | wc -l)-2" | bc )

Удалить последние 2 строки:

head -n $( echo "$(cat foo.txt | wc -l)-2" | bc )

... не очень элегантно, но выполняет свою работу!

0
spectrum

Надеюсь, я ясно понял вашу потребность.

У вас есть несколько способов выполнить ваш запрос:

tail -n$(expr $(cat /etc/passwd|wc -l) - 2) /etc/passwd

Где / etc/passwd ваш файл

Второе решение может быть полезным, если у вас большой файл:

my1stline=$(head -n1 /etc/passwd)
my2ndline=$(head -n2 /etc/passwd|grep -v "$my1stline")
cat /etc/passwd |grep -Ev "$my1stline|$my2ndline"
0
Mathieu Coavoux