Принятый ответ неверен (есть такая опция для cp). Смотрите второй ответ Пола Уиппа.
Роненц
1
@Ronenz, я согласен, что в GNU стоит упомянуть вариант cp(как ответил Пол Уипп), но он не такой общий, как запрошенный OP. Он может копировать a/foo/bar/fileдо , b/foo/bar/fileно он не может скопировать fileв b/foo/bar/file. (т.е. он работает только для создания родительских каталогов, которые уже существовали в случае с первым файлом, но не может создавать произвольные каталоги)
Sudo Bash
Это похоже на хороший запрос функции для cp. Есть ли шанс, что мы сможем получить такой ключ командной строки в ближайшем будущем?
Я не думаю, что вам нужно test -d: mkdir -pвсе еще успешно, даже если каталог уже существует.
Эфимент
упс, верно. Но тогда test может быть встроенным в bash (особенно если он написан как [[-d "$ d"]]), а mkdir не может ;-)
Майкл Крелин - хакер
1
mkdirявляется встроенным в BusyBox;)
ephemient
7
@holms, $dэто переменная, которая предположительно содержит соответствующий каталог. Я имею в виду, если вам придется повторять это трижды, вы, скорее всего, сначала назначите его переменной.
Майкл Крелин - хакер
237
Если оба из следующих условий верны:
Вы используете версию GNU cp(а не, например, версию Mac), и
Вы копируете из какой-то существующей структуры каталогов, и вам просто нужно ее восстановить
--parents
Form the name of each destination file by appending to the target
directory a slash and the specified name of the source file. The
last argument given to `cp' must be the name of an existing
directory. For example, the command:
cp --parents a/b/c existing_dir
copies the file `a/b/c' to `existing_dir/a/b/c', creating any
missing intermediate directories.
Пример:
/tmp $ mkdir foo
/tmp $ mkdir foo/foo
/tmp $ touch foo/foo/foo.txt
/tmp $ mkdir bar
/tmp $ cp --parents foo/foo/foo.txt bar
/tmp $ ls bar/foo/foo
foo.txt
Это не работает на Mac OS X, так что я думаю, что это специфично для Linux.
Олт
7
Я бы сказал, что гну конкретных. Если у вас есть, macportsвы можете установить coreutils и его gcp.
Майкл Крелин - хакер
16
Чувак, у linux есть все
Зигги
19
Я чувствую, что это ответ на немного другой вопрос, хотя это тот ответ, который я искал (и, следовательно, тот, за которого я проголосовал). Я думаю, что это решение, предполагающее, что вы хотите, чтобы родительские каталоги назначения были такими же, как исходные родительские каталоги, что, вероятно, и является тем случаем, в котором большинство людей, читающих это, интересны.
David Winiecki
7
Предполагается, что директория 'bar' уже существует, но исходный вопрос подразумевает, как автоматически создать 'bar', когда она указана в качестве пункта назначения и ее еще нет. Ответ на этот вопрос предоставлен @ MichaelKrelin-hacker.
thdoan
70
Короткий ответ
Для копирования myfile.txtв /foo/bar/myfile.txt, использования:
mkdir -p /foo/bar && cp myfile.txt $_
Как это работает?
Для этого есть несколько компонентов, поэтому я расскажу о синтаксисе шаг за шагом.
MkDir утилита, как указано в стандарте POSIX , делает каталоги. -pАргумент, согласно документации, может привести MkDir к
Создайте все недостающие промежуточные компоненты пути
Это означает , что при вызове mkdir -p /foo/bar, MkDir будет создавать /fooи/foo/bar , если /fooеще не существует. (Без -pэтого он вместо этого выдаст ошибку.
Наконец, $_мы передаем в качестве второго аргумента аргумент cp«специальный параметр», который может быть полезен для избежания повторения длинных аргументов (например, путей к файлам) без необходимости их сохранения в переменной. Согласно инструкции Bash , это:
расширяется до последнего аргумента предыдущей команды
В данном случае это то, к /foo/barчему мы перешли mkdir. Таким образом, cpкоманда расширяется до cp myfile.txt /foo/bar, которая копирует myfile.txtво вновь созданный /foo/barкаталог.
Обратите внимание, что $_это не является частью стандарта POSIX , поэтому теоретически вариант Unix может иметь оболочку, которая не поддерживает эту конструкцию. Однако я не знаю никаких современных оболочек, которые не поддерживают $_; конечно Bash, Dash и Zsh все делают.
Последнее замечание: команда, которую я дал в начале этого ответа, предполагает, что в именах ваших каталогов нет пробелов. Если вы имеете дело с именами с пробелами, вам нужно заключить их в кавычки, чтобы разные слова не рассматриваются как разные аргументы mkdirили cp. Таким образом, ваша команда будет выглядеть так:
mkdir -p "/my directory/name with/spaces"&& cp "my filename with spaces.txt""$_"
Кроме того: я обычно разочарован отсутствием подробностей в вопросах о командах оболочки Bash или Unix в Stack Overflow, которые часто выбрасывают сложную и чрезвычайно плотную однострочную строку, которую сложно распознать, если вы еще не мастер Unix. Я попытался пойти на противоположную крайность здесь и объяснил каждую деталь синтаксиса и ссылку на соответствующие места, чтобы найти документацию о том, как этот материал работает. Я надеюсь, что это поможет, по крайней мере, некоторым людям. Кроме того, следует отдать должное: я использовал этот подход из этого ответа на повторяющийся вопрос: stackoverflow.com/a/14085147/1709587
Марк Эмери
Я никогда не видел $ _ раньше. что оно относится к? папка, использованная в последней команде?
thebunnyrules
10
@thebunnyrules Но ... но ... есть "Как это работает?" раздел в моем ответе, который буквально содержит несколько абзацев, специально посвященных только что заданному вопросу. Это чувствует себя проигнорированным и нелюбимым. :(
Марк Амери
Извини за это. Ты совершенно прав. Я уже знал все остальное в вашем ответе, поэтому просто быстро просмотрел его. Я должен был прочитать это более внимательно.
thebunnyrules
Ух ты, ты действительно вложил много работы в этот ответ ... Спасибо за это. Было приятно узнать о: $ _. Я могу видеть, что теперь я использую это совсем немного. Я написал скрипт для автоматизации всего процесса и выложил в этой теме.
Попробуйте
39
Такой старый вопрос, но, возможно, я могу предложить альтернативное решение.
Вы можете использовать installпрограмму для копирования вашего файла и создания пути назначения «на лету».
вам нужно указать также имя файла назначения , а не только путь назначения
целевой файл будет исполняемым (по крайней мере, насколько я видел из моих тестов)
Вы можете легко изменить # 2, добавив -mопцию для установки разрешений для файла назначения (пример: -m 664создаст файл назначения с разрешениями rw-rw-r--, точно так же как создание нового файла с touch).
На самом деле не работал для моего использования, но крутой трюк! Я буду иметь в виду.
thebunnyrules
обратите внимание, что эта команда создает файлы назначения с разрешениями 755( rwxr-xr-x), что, вероятно, нежелательно. Вы можете указать что-то еще с помощью -mпереключателя, но я не смог найти способ просто сохранить права доступа к файлу :(
törzsmókus
19
Функция оболочки, которая делает то, что вы хотите, называя ее «скрытой» копией, потому что она вырывает дыру в файле для хранения:
@Kalmi, для правильной цитаты вы также хотели бы привести $2в качестве аргумента dirname. Как mkdir -p "$(dirname "$2")". Без этой цитаты $2в cpэто бесполезно;)
Михаил Крелин - хакер
3
Обратите внимание, что в этом ответе предполагается, что $ 2 еще не является целевым каталогом, в противном случае вы просто захотите указать «mkdir -p» $ 2 »&& cp« $ 1 »« $ 2 »в качестве тела функции
user467257
Я думаю, что в этом есть ошибка, как указывает @ user467257. Я не думаю, что это поправимо, поскольку $ 2 является неоднозначным между целевым каталогом и целевым файлом в этом сценарии. Я разместил альтернативный ответ, который обращается к этому.
Богатое
@ Рич, я не согласен с тем, что это невозможно исправить. Следующее прекрасно работает как для файлов, так и для каталогов: mkdir -p dirname "$2"&& cp -r "$ 1" "$ 2"; Обратите внимание, что обратные пометки вокруг dirname $2не отображаются, потому что SO анализирует их как маркеры кода.
dirnameдаст вам родителя каталога или файла назначения. Затем mkdir -p `dirname ...` создаст этот каталог, гарантируя, что при вызове cp -r будет создан правильный базовый каталог.
Преимущество этого над --parents в том, что он работает для случая, когда последним элементом в пути назначения является имя файла.
на самом деле, это совсем не одно и то же. Он требует, чтобы вы дважды передали имя файла (отсюда и флаг '-t'), и он установил биты выполнения в файле (отсюда и пример '-m'). он не должен быть главным ответом, поскольку он на самом деле не отвечает на вопрос - в то время как мой ответ.
Spongman
1
Это работает для одного файла. Просто учтите, что thereэто последний файл, а не последний подкаталог.
Квантур
6
При всем моем уважении к ответам выше, я предпочитаю использовать rsync следующим образом:
$ rsync -a directory_name /path_where_to_inject_your_directory/
Я попробовал это и получил следующий результат: rsync -a bull / some / hoop / ti / doop / ploop РЕЗУЛЬТАТ rsync: mkdir "/ some / hoop / ti / doop / ploop" не удалось: такой файл или каталог (2) ошибка rsync отсутствует : ошибка в IO файла (код 11) в main.c (675) [Receiver = 3.1.2]
Если я правильно понимаю ваше предложение, вы предлагаете указать в моей команде полный путь к источнику, также называемому «$ (pwd) / bull», а не просто имя файла: bull? Дело в том, что ошибка в rsync создает директорию назначения, а не источник (бык). Мне кажется, rsync использует простой mkdir, а не mkdir -p.
thebunnyrules
1
Кстати, отличная утилита, я могу начать использовать в своих скриптах. Спасибо за рекомендацию.
thebunnyrules
1
rsync более предсказуемо, чем cp. Например, если я хочу, чтобы cp /from/somedir /to-dircp вел себя по-другому, если он /to-dirуже существует: если /to-dirсуществует, то cpсоздаст /to-dir/some-dir, в противном случае /from/somedirпомещается только содержимое /to-dir. Тем не менее, rsyncведет себя так же в случае, то есть, /to-dir/some-dirбудет всегда создаваться.
JoeAC
5
Просто чтобы возобновить и дать полное рабочее решение, в одну строку. Будьте осторожны, если вы хотите переименовать ваш файл, вы должны включить способ предоставить чистый путь к каталогу mkdir. $ fdst может быть файлом или каталогом. Следующий код должен работать в любом случае.
Спасибо! Это то, что мне было нужно. В моем случае я не знаю, есть ли у места назначения каталог или нет, и существует ли каталог, или нет, поэтому этот код дает мне родительский каталог, который я могу создать безопасно, если он не существует
xbmono
4
Просто добавьте следующее в ваш .bashrc, настройте, если вам нужно. Работает в Ubuntu.
mkcp(){
test -d "$2"|| mkdir -p "$2"
cp -r "$1""$2"}
Например, если вы хотите скопировать файл 'test' в целевой каталог 'd' Use,
mkcp test a/b/c/d
Сначала mkcp проверит, существует ли каталог назначения, если нет, то создайте его и скопируйте исходный файл / каталог.
Как другие прокомментировали другой ответ, вам не нужно проверять, существует ли каталог перед вызовом mkdir -p. Это удастся, даже если он уже существует.
Согласен. В man cp:-R If source_file designates a directory, cp copies the directory and the entire subtree connected at that point.
BorracciaBlu
3
Я написал вспомогательный скрипт для cp, называемый CP (примечание заглавными буквами), который предназначен именно для этого. Скрипт проверит наличие ошибок в пути, который вы указали (за исключением последнего, который является пунктом назначения), и, если все хорошо, он сделает шаг mkdir -p, чтобы создать путь назначения перед началом копирования. В этот момент вступает в действие обычная утилита cp, и все переключатели, которые вы используете с CP (например, -r, -p, -rpL, передаются напрямую в cp). Прежде чем использовать мой сценарий, необходимо кое-что понять.
Вся информация здесь может быть доступна с помощью команды CP --help. CP --help-all включает ключи cp.
обычный cp не сделает копию, если не найдет путь назначения. У вас нет такой системы безопасности для опечаток с CP. Ваш пункт назначения будет создан, поэтому, если вы неправильно напишите пункт назначения как / usrr / share / icons или / usr / share / icon, то именно это и будет создано.
обычный cp имеет тенденцию моделировать его поведение на существующем пути: cp / a / b / c / d будет зависеть от того, существует d или нет. если d - существующая папка, cp скопирует в нее b, создав / c / d / b. Если d не существует, b будет скопирован в c и переименован в d. Если d существует, но является файлом, а b является файлом, он будет перезаписан копией b. Если c не существует, cp не делает копию и выходит.
СР не может позволить себе роскошь брать реплики с существующих путей, поэтому у него должны быть очень твердые модели поведения. CP предполагает, что копируемый вами элемент отбрасывается по пути назначения, а не сам пункт назначения (иначе, переименованная копия исходного файла / папки). Смысл:
«CP / a / b / c / d» приведет к / c / d / b, если d - папка
«CP / a / b / c / b» приведет к / c / b / b, если b в / c / b является папкой.
Если оба b и d являются файлами: CP / a / b / c / d приведет к / c / d (где d - копия b). То же самое для CP / a / b / c / b в тех же обстоятельствах.
Это поведение CP по умолчанию можно изменить с помощью ключа «--rename». В этом случае предполагается, что
«CP - имя / a / b / c / d» копирует b в / c и переименовывает копию в d.
Несколько заключительных замечаний: Как и в случае cp, CP может копировать несколько элементов одновременно, причем последний путь в списке считается местом назначения. Он также может обрабатывать пути с пробелами, если вы используете кавычки.
CP проверит введенные вами пути и удостоверится, что они существуют, прежде чем копировать. В строгом режиме (доступно через ключ --strict) все копируемые файлы / папки должны существовать, иначе копирование не производится. В расслабленном режиме (--relaxed) копирование будет продолжено, если хотя бы один из перечисленных вами элементов существует. Режим по умолчанию является расслабленным, вы можете изменить режим временно с помощью переключателей или навсегда, установив переменную easy_going в начале скрипта.
#!/bin/bash#Regular cp works with the assumption that the destination path exists and if it doesn't, it will verify that it's parent directory does.#eg: cp /a/b /c/d will give /c/d/b if folder path /c/d already exists but will give /c/d (where d is renamed copy of b) if /c/d doesn't exists but /c does.#CP works differently, provided that d in /c/d isn't an existing file, it assumes that you're copying item into a folder path called /c/d and will create it if it doesn't exist. so CP /a/b /c/d will always give /c/d/b unless d is an existing file. If you put the --rename switch, it will assume that you're copying into /c and renaming the singl item you're copying from b to d at the destination. Again, if /c doesn't exist, it will be created. So CP --rename /a/b /c/d will give a /c/d and if there already a folder called /c/d, contents of b will be merged into d. #cp+ $source $destination#mkdir -p /foo/bar && cp myfile "$_"
err=0# error count
i=0#item counter, doesn't include destination (starts at 1, ex. item1, item2 etc)
m=0#cp switch counter (starts at 1, switch 1, switch2, etc)
n=1# argument counter (aka the arguments inputed into script, those include both switches and items, aka: $1 $2 $3 $4 $5)
count_s=0
count_i=0
easy_going=true #determines how you deal with bad pathes in your copy, true will allow copy to continue provided one of the items being copied exists, false will exit script for one bad path. this setting can also be changed via the custom switches: --strict and --not-strict
verbal="-v"
help="===============================================================================\
\n CREATIVE COPY SCRIPT (CP) -- written by thebunnyrules\
\n===============================================================================\n
\n This script (CP, note capital letters) is intended to supplement \
\n your system's regular cp command (note uncapped letters). \n
\n Script's function is to check if the destination path exists \
\n before starting the copy. If it doesn't it will be created.\n
\n To make this happen, CP assumes that the item you're copying is \
\n being dropped in the destination path and is not the destination\
\n itself (aka, a renamed copy of the source file/folder). Meaning:\n
\n * \"CP /a/b /c/d\" will result in /c/d/b \
\n * even if you write \"CP /a/b /c/b\", CP will create the path /a/b, \
\n resulting in /c/b/b. \n
\n Of course, if /c/b or /c/d are existing files and /a/b is also a\
\n file, the existing destination file will simply be overwritten. \
\n This behavior can be changed with the \"--rename\" switch. In this\
\n case, it's assumed that \"CP --rename /a/b /c/d\" is copying b into /c \
\n and renaming the copy to d.\n
\n===============================================================================\
\n CP specific help: Switches and their Usages \
\n===============================================================================\n
\
\n --rename\tSee above. Ignored if copying more than one item. \n
\n --quiet\tCP is verbose by default. This quiets it.\n
\n --strict\tIf one+ of your files was not found, CP exits if\
\n\t\tyou use --rename switch with multiple items, CP \
\n\t\texits.\n
\n --relaxed\tIgnores bad paths unless they're all bad but warns\
\n\t\tyou about them. Ignores in-appropriate rename switch\
\n\t\twithout exiting. This is default behavior. You can \
\n\t\tmake strict the default behavior by editing the \
\n\t\tCP script and setting: \n
\n\t\teasy_going=false.\n
\n --help-all\tShows help specific to cp (in addition to CP)."
cp_hlp="\n\nRegular cp command's switches will still work when using CP.\
\nHere is the help out of the original cp command... \
\n\n===============================================================================\
\n cp specific help: \
\n===============================================================================\n"
outro1="\n******************************************************************************\
\n******************************************************************************\
\n******************************************************************************\
\n USE THIS SCRIPT WITH CARE, TYPOS WILL GIVE YOU PROBLEMS...\
\n******************************************************************************\
\n******************************* HIT q TO EXIT ********************************\
\n******************************************************************************"#count and classify arguments that were inputed into script, output help message if neededwhile true;doeval input="\$$n"
in_=${input::1}if[-z "$input"-a $n =1];then input="--help";fiif["$input"="-h"-o "$input"="--help"-o "$input"="-?"-o "$input"="--help-all"];thenif["$input"="--help-all"];then
echo -e "$help"$cp_hlp >/tmp/cp.hlp
cp --help >>/tmp/cp.hlp
echo -e "$outro1">>/tmp/cp.hlp
cat /tmp/cp.hlp|less
cat /tmp/cp.hlp
rm /tmp/cp.hlp
else
echo -e "$help""$outro1"|less
echo -e "$help""$outro1"fi
exit
fiif[-z "$input"];then
count_i=$(expr $count_i -1)# remember, last item is destination and it's not included in coundbreakelif["$in_"="-"];then
count_s=$(expr $count_s +1)else
count_i=$(expr $count_i +1)fi
n=$(expr $n +1)done#error condition: no items to copy or no destinationif[ $count_i -lt 0];then
echo "Error: You haven't listed any items for copying. Exiting."# you didn't put any items for copyingelif[ $count_i -lt 1];then
echo "Error: Copying usually involves a destination. Exiting."# you put one item and no destinationfi#reset the counter and grab content of arguments, aka: switches and item paths
n=1while true;doeval input="\$$n"#input=$1,$2,$3,etc...
in_=${input::1}#first letter of $inputif["$in_"="-"];thenif["$input"="--rename"];then
rename=true #my custom switcheselif["$input"="--strict"];then
easy_going=false #exit script if even one of the non-destinations item is not foundelif["$input"="--relaxed"];then
easy_going=true #continue script if at least one of the non-destination items is foundelif["$input"="--quiet"];then
verbal=""else#m=$(expr $m + 1);eval switch$m="$input" #input is a switch, if it's not one of the above, assume it belongs to cp.
switch_list="$switch_list \"$input\""fielif![-z "$input"];then#if it's not a switch and input is not empty, it's a path
i=$(expr $i +1)if[!-f "$input"-a !-d "$input"-a "$i"-le "$count_i"];then
err=$(expr $err +1); error_list="$error_list\npath does not exit: \"b\""elseif["$i"-le "$count_i"];theneval item$i="$input"
item_list="$item_list \"$input\""else
destination="$input"#destination is last items enteredfifielse
i=0
m=0
n=1breakfi
n=$(expr $n +1)done#error condition: some or all item(s) being copied don't exist. easy_going: continue if at least one item exists, warn about rest, not easy_going: exit.#echo "err=$err count_i=$count_i"if["$easy_going"!= true -a $err -gt 0-a $err != $count_i ];then
echo "Some of the paths you entered are incorrect. Script is running in strict mode and will therefore exit."
echo -e "Bad Paths: $err $error_list"
exit
fiif[ $err = $count_i ];then
echo "ALL THE PATHS you have entered are incorrect! Exiting."
echo -e "Bad Paths: $err $error_list"fi#one item to one destination:#------------------------------#assumes that destination is folder, it does't exist, it will create it. (so copying /a/b/c/d/firefox to /e/f/firefox will result in /e/f/firefox/firefox#if -rename switch is given, will assume that the top element of destination path is the new name for the the item being given.#multi-item to single destination:#------------------------------#assumes destination is a folder, gives error if it exists and it's a file. -rename switch will be ignored.#ERROR CONDITIONS: # - multiple items being sent to a destination and it's a file.# - if -rename switch was given and multiple items are being copied, rename switch will be ignored (easy_going). if not easy_going, exit.# - rename option but source is folder, destination is file, exit.# - rename option but source is file and destination is folder. easy_going: option ignored.if[-f "$destination"];thenif[ $count_i -gt 1];then
echo "Error: You've selected a single file as a destination and are copying multiple items to it. Exiting."; exit
elif[-d "$item1"];then
echo "Error: Your destination is a file but your source is a folder. Exiting."; exit
fifiif["$rename"= true ];thenif[ $count_i -gt 1];thenif[ $easy_going = true ];then
echo "Warning: you choose the rename option but are copying multiple items. Ignoring Rename option. Continuing."else
echo "Error: you choose the rename option but are copying multiple items. Script running in strict mode. Exiting."; exit
fielif[-d "$destination"-a -f "$item1"];then
echo -n "Warning: you choose the rename option but source is a file and destination is a folder with the same name. "if[ $easy_going = true ];then
echo "Ignoring Rename option. Continuing."else
echo "Script running in strict mode. Exiting."; exit
fielse
dest_jr=$(dirname "$destination")if[-d "$destination"];then item_list="$item1/*";fi
mkdir -p "$dest_jr"fielse
mkdir -p "$destination"fieval cp $switch_list $verbal $item_list "$destination"
cp_err="$?"if["$cp_err"!=0];then
echo -e "Something went wrong with the copy operation. \nExit Status: $cp_err"else
echo "Copy operation exited with no errors."fi
exit
Это может быть излишним, но довольно эффективным, работает как положено, спасибо, сэр.
Кседрет
2
cp имеет многократное использование:
$ cp --help
Usage: cp [OPTION]...[-T] SOURCE DEST
or: cp [OPTION]... SOURCE... DIRECTORY
or: cp [OPTION]...-t DIRECTORY SOURCE...Copy SOURCE to DEST, or multiple SOURCE(s) to DIRECTORY.
@ AndyRoss ответ работает на
cp SOURCE DEST
стиль cp, но делает неправильную вещь, если вы используете
cp SOURCE... DIRECTORY/
стиль cp.
Я думаю, что «DEST» неоднозначно без косой черты в этом использовании (то есть, когда целевой каталог еще не существует), поэтому, возможно, поэтому cpникогда не добавлял опцию для этого.
Итак, вот моя версия этой функции, которая вводит косую черту в dest dir:
Как предложено выше в help_asap и spongeman, вы можете использовать команду 'install', чтобы скопировать файлы в существующие каталоги или создать новые целевые каталоги, если они еще не существуют.
Вариант 1
install -D filename some/deep/directory/filename
копирует файл в новый или существующий каталог и дает 755 разрешений по умолчанию для имени файла
Вариант 2
install -D filename -m640 some/deep/directory/filename
согласно варианту 1, но дает имени файла 640 разрешений.
Вариант 3
install -D filename -m640 -t some/deep/directory/
согласно Варианту 2, но он направляет имя файла в целевой каталог, поэтому имя файла не нужно записывать как в исходном, так и в целевом.
Вариант 4
install -D filena* -m640 -t some/deep/directory/
согласно варианту 3, но использует подстановочный знак для нескольких файлов.
Он прекрасно работает в Ubuntu и объединяет два шага (создание каталога, затем копирование файла) в один шаг.
Это очень поздно, но это может помочь новичку где-то. Если вам нужно «автоматически» создавать папки, rsync должен быть вашим лучшим другом. rsync / путь / к / исходный файл / путь / к / tragetdir / thatdoestexist /
cp
(как ответил Пол Уипп), но он не такой общий, как запрошенный OP. Он может копироватьa/foo/bar/file
до ,b/foo/bar/file
но он не может скопироватьfile
вb/foo/bar/file
. (т.е. он работает только для создания родительских каталогов, которые уже существовали в случае с первым файлом, но не может создавать произвольные каталоги)Ответы:
(такой опции нет
cp
).источник
test -d
:mkdir -p
все еще успешно, даже если каталог уже существует.mkdir
является встроенным в BusyBox;)$d
это переменная, которая предположительно содержит соответствующий каталог. Я имею в виду, если вам придется повторять это трижды, вы, скорее всего, сначала назначите его переменной.Если оба из следующих условий верны:
cp
(а не, например, версию Mac), итогда вы можете сделать это с
--parents
флагомcp
. На информационной странице (можно просмотреть по адресу http://www.gnu.org/software/coreutils/manual/html_node/cp-invocation.html#cp-invocation или с помощьюinfo cp
илиman cp
):Пример:
источник
macports
вы можете установить coreutils и егоgcp
.Короткий ответ
Для копирования
myfile.txt
в/foo/bar/myfile.txt
, использования:Как это работает?
Для этого есть несколько компонентов, поэтому я расскажу о синтаксисе шаг за шагом.
MkDir утилита, как указано в стандарте POSIX , делает каталоги.
-p
Аргумент, согласно документации, может привести MkDir кЭто означает , что при вызове
mkdir -p /foo/bar
, MkDir будет создавать/foo
и/foo/bar
, если/foo
еще не существует. (Без-p
этого он вместо этого выдаст ошибку.Оператор
&&
списка, как описано в стандарте POSIX (или, если хотите, в руководстве по Bash ), имеет эффект, которыйcp myfile.txt $_
выполняется только в случаеmkdir -p /foo/bar
успешного выполнения. Это означает , чтоcp
команда не будет пытаться выполнить , еслиmkdir
не удается по одной из многих причин , почему это может произойти сбой .Наконец,
$_
мы передаем в качестве второго аргумента аргументcp
«специальный параметр», который может быть полезен для избежания повторения длинных аргументов (например, путей к файлам) без необходимости их сохранения в переменной. Согласно инструкции Bash , это:В данном случае это то, к
/foo/bar
чему мы перешлиmkdir
. Таким образом,cp
команда расширяется доcp myfile.txt /foo/bar
, которая копируетmyfile.txt
во вновь созданный/foo/bar
каталог.Обратите внимание, что
$_
это не является частью стандарта POSIX , поэтому теоретически вариант Unix может иметь оболочку, которая не поддерживает эту конструкцию. Однако я не знаю никаких современных оболочек, которые не поддерживают$_
; конечно Bash, Dash и Zsh все делают.Последнее замечание: команда, которую я дал в начале этого ответа, предполагает, что в именах ваших каталогов нет пробелов. Если вы имеете дело с именами с пробелами, вам нужно заключить их в кавычки, чтобы разные слова не рассматриваются как разные аргументы
mkdir
илиcp
. Таким образом, ваша команда будет выглядеть так:источник
Такой старый вопрос, но, возможно, я могу предложить альтернативное решение.
Вы можете использовать
install
программу для копирования вашего файла и создания пути назначения «на лету».Однако следует учитывать некоторые аспекты:
Вы можете легко изменить # 2, добавив
-m
опцию для установки разрешений для файла назначения (пример:-m 664
создаст файл назначения с разрешениямиrw-rw-r--
, точно так же как создание нового файла сtouch
).И вот это бесстыдная ссылка на ответ, который меня вдохновил =)
источник
755
(rwxr-xr-x
), что, вероятно, нежелательно. Вы можете указать что-то еще с помощью-m
переключателя, но я не смог найти способ просто сохранить права доступа к файлу :(Функция оболочки, которая делает то, что вы хотите, называя ее «скрытой» копией, потому что она вырывает дыру в файле для хранения:
источник
dirname $2
тоже$2
в качестве аргументаdirname
. Какmkdir -p "$(dirname "$2")"
. Без этой цитаты$2
вcp
это бесполезно;)dirname "$2"
&& cp -r "$ 1" "$ 2"; Обратите внимание, что обратные пометки вокругdirname $2
не отображаются, потому что SO анализирует их как маркеры кода.Вот один из способов сделать это:
dirname
даст вам родителя каталога или файла назначения. Затем mkdir -p `dirname ...` создаст этот каталог, гарантируя, что при вызове cp -r будет создан правильный базовый каталог.Преимущество этого над --parents в том, что он работает для случая, когда последним элементом в пути назначения является имя файла.
И это будет работать на OS X.
источник
install -D file -m 644 -t /path/to/copy/file/to/is/very/deep/there
источник
there
это последний файл, а не последний подкаталог.При всем моем уважении к ответам выше, я предпочитаю использовать rsync следующим образом:
пример:
источник
rsync
более предсказуемо, чемcp
. Например, если я хочу, чтобыcp /from/somedir /to-dir
cp вел себя по-другому, если он/to-dir
уже существует: если/to-dir
существует, тоcp
создаст/to-dir/some-dir
, в противном случае/from/somedir
помещается только содержимое/to-dir
. Тем не менее,rsync
ведет себя так же в случае, то есть,/to-dir/some-dir
будет всегда создаваться.Просто чтобы возобновить и дать полное рабочее решение, в одну строку. Будьте осторожны, если вы хотите переименовать ваш файл, вы должны включить способ предоставить чистый путь к каталогу mkdir. $ fdst может быть файлом или каталогом. Следующий код должен работать в любом случае.
или специфичный для bash
источник
Просто добавьте следующее в ваш .bashrc, настройте, если вам нужно. Работает в Ubuntu.
Например, если вы хотите скопировать файл 'test' в целевой каталог 'd' Use,
Сначала mkcp проверит, существует ли каталог назначения, если нет, то создайте его и скопируйте исходный файл / каталог.
источник
mkdir -p
. Это удастся, даже если он уже существует.Это делает это для меня
источник
man cp
:-R If source_file designates a directory, cp copies the directory and the entire subtree connected at that point.
Я написал вспомогательный скрипт для cp, называемый CP (примечание заглавными буквами), который предназначен именно для этого. Скрипт проверит наличие ошибок в пути, который вы указали (за исключением последнего, который является пунктом назначения), и, если все хорошо, он сделает шаг mkdir -p, чтобы создать путь назначения перед началом копирования. В этот момент вступает в действие обычная утилита cp, и все переключатели, которые вы используете с CP (например, -r, -p, -rpL, передаются напрямую в cp). Прежде чем использовать мой сценарий, необходимо кое-что понять.
СР не может позволить себе роскошь брать реплики с существующих путей, поэтому у него должны быть очень твердые модели поведения. CP предполагает, что копируемый вами элемент отбрасывается по пути назначения, а не сам пункт назначения (иначе, переименованная копия исходного файла / папки). Смысл:
Это поведение CP по умолчанию можно изменить с помощью ключа «--rename». В этом случае предполагается, что
Несколько заключительных замечаний: Как и в случае cp, CP может копировать несколько элементов одновременно, причем последний путь в списке считается местом назначения. Он также может обрабатывать пути с пробелами, если вы используете кавычки.
CP проверит введенные вами пути и удостоверится, что они существуют, прежде чем копировать. В строгом режиме (доступно через ключ --strict) все копируемые файлы / папки должны существовать, иначе копирование не производится. В расслабленном режиме (--relaxed) копирование будет продолжено, если хотя бы один из перечисленных вами элементов существует. Режим по умолчанию является расслабленным, вы можете изменить режим временно с помощью переключателей или навсегда, установив переменную easy_going в начале скрипта.
Вот как это установить:
В терминале без полномочий root выполните:
В gedit вставьте утилиту CP и сохраните:
источник
cp
имеет многократное использование:@ AndyRoss ответ работает на
стиль
cp
, но делает неправильную вещь, если вы используетестиль
cp
.Я думаю, что «DEST» неоднозначно без косой черты в этом использовании (то есть, когда целевой каталог еще не существует), поэтому, возможно, поэтому
cp
никогда не добавлял опцию для этого.Итак, вот моя версия этой функции, которая вводит косую черту в dest dir:
источник
Просто была такая же проблема. Мой подход состоял в том, чтобы просто распаковать файлы в архив следующим образом:
tar cf your_archive.tar file1 /path/to/file2 path/to/even/deeper/file3
tar автоматически сохраняет файлы в соответствующей структуре в архиве. Если вы бежите
tar xf your_archive.tar
файлы извлекаются в желаемую структуру каталогов.
источник
Как предложено выше в help_asap и spongeman, вы можете использовать команду 'install', чтобы скопировать файлы в существующие каталоги или создать новые целевые каталоги, если они еще не существуют.
Вариант 1
install -D filename some/deep/directory/filename
копирует файл в новый или существующий каталог и дает 755 разрешений по умолчанию для имени файла
Вариант 2
install -D filename -m640 some/deep/directory/filename
согласно варианту 1, но дает имени файла 640 разрешений.
Вариант 3
install -D filename -m640 -t some/deep/directory/
согласно Варианту 2, но он направляет имя файла в целевой каталог, поэтому имя файла не нужно записывать как в исходном, так и в целевом.
Вариант 4
install -D filena* -m640 -t some/deep/directory/
согласно варианту 3, но использует подстановочный знак для нескольких файлов.
Он прекрасно работает в Ubuntu и объединяет два шага (создание каталога, затем копирование файла) в один шаг.
источник
Это очень поздно, но это может помочь новичку где-то. Если вам нужно «автоматически» создавать папки, rsync должен быть вашим лучшим другом. rsync / путь / к / исходный файл / путь / к / tragetdir / thatdoestexist /
источник
Это может работать, если у вас есть правильный вид
rsync
.источник
Копировать из источника в несуществующий путь
ПРИМЕЧАНИЕ: эта команда копирует все файлы
cp –r
для копирования всех папок и их содержимого$_
работать как пункт назначения, который создается в последней командеисточник
просто
должен сделать свое дело.
источник
Допустим, вы делаете что-то вроде
где A / B / C / D - каталоги, которые еще не существуют
Возможное решение заключается в следующем
надеюсь, это поможет!
источник