Как добавить текст в начало файла в Bash?

285

Привет, я хочу добавить текст в файл. Например, я хочу добавить задачи в начало файла todo.txt. Я знаю, echo 'task goes here' >> todo.txtно это добавляет строку в конец файла (не то, что я хочу).

user479534
источник
2
Существует эффективный способ сделать это для больших файлов: unix.stackexchange.com/questions/87772/…
Сиро Сантилли 新疆 改造 中心 法轮功 六四 事件
@ не могли бы вы взглянуть на этот метод superuser.com/a/1352628/11116
Эван Кэрролл,

Ответы:

374
echo 'task goes here' | cat - todo.txt > temp && mv temp todo.txt

или же

sed -i '1s/^/task goes here\n/' todo.txt

или же

sed -i '1itask goes here' todo.txt
Деннис Уильямсон
источник
4
первый работает отлично! не могли бы вы объяснить логику? Я не совсем уверен, как интерпретировать синтаксис.
user479534
44
@ user8347: Pipe ( |) сообщение ( echo '...'), для catкоторого используется -(стандартный ввод) в качестве первого файла и todo.txtв качестве второго. catconCATenates несколько файлов. Отправьте вывод ( >) в файл с именем temp. Если нет ошибок ( &&), catпереименуйте ( mv) tempфайл обратно в исходный файл ( todo.txt).
Деннис Уильямсон
1
@itaifrenkel: Я бы посмотрел, что ты сделал, но если catполучит буквальную обратную косую черту, он не будет преобразован в новую строку. Что-то еще, должно быть, сделало это. Вместо этого catпопробуйте подключиться, hexdump -Cчтобы увидеть, действительно ли вы отправляете обратную косую черту и n или символ новой строки. Вы также можете попытаться cat -eпоказать окончание строк.
Деннис Уильямсон
1
Использование 2 и 3 (мне кажется, что 3 проще) позволяет добавлять текст ко многим файлам одновременно.
Феликс
4
@Kira: 1Средство делает следующую команду только в первой строке файла, а iкоманда вставляется. Посмотрите справочную страницу в разделе «Адреса» и в разделе «Команды с нулевым или одноадресным адресом».
Деннис Уильямсон
73

На мой взгляд, более простой вариант:

echo -e "task goes here\n$(cat todo.txt)" > todo.txt

Это работает, потому что команда внутри $(...)выполняется прежде, чем todo.txtбудет перезаписана> todo.txt

В то время как другие ответы работают хорошо, я нахожу это намного легче запомнить, потому что я использую echo и cat каждый день.

РЕДАКТИРОВАТЬ: Это решение очень плохая идея, если есть какие-либо обратные слеши todo.txt, потому что благодаря -eфлагу echo будет их интерпретировать. Еще один, гораздо более простой способ получить перевод строки в предисловие ...

echo "task goes here
$(cat todo.txt)" > todo.txt

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

(echo 'task goes here' && cat todo.txt) > todo.txt
echo 'task goes here'$'\n'"$(cat todo.txt)" > todo.txt

Кроме того, если вы заботитесь о том, будет ли добавлена ​​новая строка в конце todo.txt, не используйте их. Ну, кроме второго до последнего. Это не портит конец.

Джон Альбертс
источник
1
Я не выполняю $ (...) вообще.
SCL
2
Это может сработать лучше (или вообще) с двойными кавычками вместо одинарных…
ℝaphink
5
Не -eпреобразует ли также escape-последовательности внутри todo.txt?
mk12,
2
Обходные решения yield -> cat: <имя файла назначения>: входной файл - выходной файл ...
здесь
printf будет намного более согласованным на разных платформах и, как правило, будет работать более плавно, чем echo -e
Питер Берг
28

moreutilsЕсть хороший инструмент под названием sponge:

echo "task goes here" | cat - todo.txt | sponge todo.txt

Он «впитает» STDIN и затем запишет в файл, что означает, что вам не нужно беспокоиться о временных файлах и их перемещении.

Вы можете получить moreutilsдоступ ко многим дистрибутивам Linux, через apt-get install moreutilsили в OS X, используя Homebrew , с помощью brew install moreutils.

slhck
источник
Я хотел бы пойти (echo 'ble'; cat todo.txt):-)
Сиро Сантилли 新疆 改造 中心 法轮功 六四 事件
11

Вы можете использовать Vim в режиме Ex:

ex -s -c '1i|task goes here' -c x todo.txt
  1. 1 выберите первую строку

  2. i вставить

  3. x сохранить и закрыть

Стивен Пенни
источник
5

Вы можете создать новый временный файл.

echo "new task" > new_todo.txt
cat todo.txt >> new_todo.txt
rm todo.txt
mv new_todo.txt todo.txt

Вы также можете использовать sedили awk. Но в основном происходит то же самое.

Кит
источник
1
Скажем, у вас недостаточно места на диске, так что new_todo.txtзаписывается только частично. Ваше решение, похоже, потеряет исходный файл.
NPE
У кого заканчивается дисковое пространство? ;-) Это просто простой пример.
Кит
2
@Keith Кто-то работает на виртуальной машине и не ожидал, что ей понадобится особенно большой виртуальный диск. Или кто-то перемещает большой файл. В любом случае, реальный аргумент против этого - права доступа к каталогу; если у вас нет разрешения на создание новых файлов в данном каталоге, единственная команда, которая будет успешно выполнена в вашем сценарии, - rmэто исходный файл.
Парфянский выстрел
3

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

echo "$(echo 'task goes here' | cat - todo.txt)" > todo.txt

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

Rucent88
источник
Просто чтобы задать очевидный вопрос: где предел символов переменных оболочки?
Никда
Насколько я знаю, он ограничен только объемом доступной памяти. Я заполнил переменные более 100 МБ в памяти. text=$(cat file), Будьте осторожны, чтобы использовать только текст, потому что переменные оболочки не являются двоичными чистыми mywiki.wooledge.org/BashFAQ/058
Rucent88
2

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

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

Остерегайтесь потери данных, сохраняя исходный файл при создании нового, если файловая система окажется переполненной во время процесса. например:

cat <(echo task go there) todo.txt > todo.txt.new && mv todo.txt.new todo.txt
jlliagre
источник
Downvoters могут объяснить свою мотивацию. Ни один из оставшихся ответов, включая принятый, в моем ответе не противоречит.
jlliagre
Это трудно разобрать, так как <...> выглядят как скобки, что, я полагаю, не так. Пробел между <и (может помочь?
Дамблдад
Это не работает для меня. echo HOME=\"/g/Users/timregan/\" | cat - 'F:\Program Files\Git\etc\profile'работает, но cat <echo HOME=\"/g/Users/timregan/\" 'F:\Program Files\Git\etc\profile'выдает ошибку "echo: нет такого файла или каталога"
dumbledad
@dumbledad Вы думаете над моим ответом. Вам нечего в этом разбирать. Пробел между <и (нарушает синтаксис. Попробуйтеcat <(echo HOME=\"/g/Users/timregan/\") 'F:\Program Files\Git\etc\profile'
jlliagre
1

Вы можете использовать tee:

echo 'task goes here' | cat - todo.txt | tee todo.txt
Раду Рэдяну
источник
3
Нет, вы не можете askubuntu.com/a/752451
Стивен Пенни
0

GitBash + Windows10 + Multline :

Вот версия, которая позволяет вам использовать многострочные строки .

##############################################
## This section for demo purpose only,      ##
## So you can save entire file as           ##
## whatever.sh and run it.                  ##
##                                          ##
##############################################
> MY_TARGET_FILE.txt ##Make Or Clear File
echo "[STARTER_CONTENT]" >> MY_TARGET_FILE.txt
##############################################

## Below is main code:

##################################################
TARGET_FILE_VARIABLE="MY_TARGET_FILE.txt"
ADD_TO_HEAD_VARIABLE=$(cat << "HEREDOC_HEAD_TEXT"
//|  +-------------------------------------+   |//
//|  |                                     |   |//
//|  |     MESSAGE_FOR_HEAD_OF_FILE        |   |//
//|  |                                     |   |//
//|  +-------------------------------------+   |//
HEREDOC_HEAD_TEXT
)
##################################################
TAR=$TARGET_FILE_VARIABLE                       ##
TEX=$ADD_TO_HEAD_VARIABLE                       ##
echo "$TEX" | cat - $TAR > TEMP && mv TEMP $TAR ##
##################################################

## Expected contents of MY_TARGET_FILE.txt :
## //|  +-------------------------------------+   |//
## //|  |                                     |   |//
## //|  |     MESSAGE_FOR_HEAD_OF_FILE        |   |//
## //|  |                                     |   |//
## //|  +-------------------------------------+   |//
## [STARTER_CONTENT]
J MADISON
источник