Linux:如果目标目录不存在,则复制并创建目标目录


347

我想要创建目标目录(如果不存在)的命令(或cp的选项)。

例:

cp -? file /path/to/copy/file/to/is/very/deep/there


5
接受的答案是错误的(cp有这样的选项)。参见Paul Whipp的第二个答案。
Ronenz 2014年

1
@Ronenz,我同意GNU中有一个值得一提的选项cp(由Paul Whipp回答),但是它不如OP所要求的那么笼统。它可以复制a/foo/bar/file到,b/foo/bar/file但不能复制fileb/foo/bar/file。(即,它仅可用于创建第一个文件的情况下已经存在的父目录,但不能创建任意目录)
Sudo Bash

这看起来像是对cp的不错的功能请求。我们是否有可能在不久的将来获得这样的命令行开关?
Arnaud

Answers:


327
mkdir -p "$d" && cp file "$d"

()没有此类选项cp


62
我认为您不需要test -dmkdir -p即使目录已经存在也仍然可以成功。
迅速

哎呀,对 但是,然后,测试可能是内置的bash(尤其是如果写为[[-d“ $ d”]])并且mkdir不能;-)
Michael Krelin-黑客

1
mkdir是在BusyBox中内置的;)
短暂

7
@holms $d是一个变量,可能包含有问题的目录。我的意思是,如果必须重复三次,则可能首先将其分配给变量。
Michael Krelin-黑客

237

如果以下两个条件都成立:

  1. 您使用的是GNU版本cp(而不是Mac版本),并且
  2. 您正在从一些现有的目录结构中复制,只需要重新创建它

那么您可以使用--parents标记来执行此操作cp。从信息页面(可视范围在http://www.gnu.org/software/coreutils/manual/html_node/cp-invocation.html#cp-invocationinfo cpman cp):

--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

3
这在Mac OS X上不起作用,所以我猜它是特定于Linux的。
olt 2012年

6
我会说特定于gnu。如果有,macports您可以安装coreutils及其gcp
Michael Krelin-黑客

16
伙计,Linux拥有一切
Ziggy

19
我觉得这是一个稍有不同的问题的答案,即使这是我一直在寻找的答案(因此也是我赞成的答案)。我想这是假设你想要的目的地父目录是相同的起源父目录,这可能是使用的情况下,大多数人读这是有趣的解决方案。
大卫Winiecki

7
这假定目录“ bar”已经存在,但是原始问题暗示了当提供“ bar”作为目的地并且尚不存在时如何自动创建“ bar”。对此的答案由@ MichaelKrelin-hacker提供。
thdoan's

69

简短答案

要复制myfile.txt/foo/bar/myfile.txt,请使用:

mkdir -p /foo/bar && cp myfile.txt $_

这是如何运作的?

这有几个组成部分,因此我将逐步介绍所有语法。

POSIX标准中指定mkdir实用程序创建目录。的说法,每文档,会造成MKDIR-p

创建任何缺少的中间路径名组件

意味着在调用时mkdir -p /foo/barmkdir将创建/foo /foo/bar如果/foo尚不存在。(如果没有-p,它将抛出错误。

POSIX标准(或Bash手册,如果您愿意)中&&记录的list运算符具有仅在成功执行后才执行的效果。这意味着如果由于多种原因之一而失败,该命令将不会尝试执行cp myfile.txt $_mkdir -p /foo/barcpmkdir

最后,$_作为第二个参数,我们传递的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" "$_"

16
另外:我通常对在Stack Overflow上有关Bash或Unix shell命令的问题缺乏详细信息感到失望,这些问题经常抛出一个复杂且极其密集的单行代码,如果您还不是Unix向导的话,很难取消选择。我在这里尝试了另一种极端的做法,它们都解释了语法的每个细节,并链接到适当的位置以查找有关该工具如何工作的文档。我希望这对至少某些人有所帮助。另外,应归功于我:我从这个答案中剔除了这种方法,重复了一个问题:stackoverflow.com/a/14085147/1709587
Mark Amery 2015年

我从未见过$ _。它指的是什么?最后一个命令中使用的文件夹?
thebunnyrules

9
@thebunnyrules但是...但是...有一个“这如何工作的?” 在我的答案中,该部分实际上包含了多个专门针对您刚刚提出的问题的段落。感觉被忽略和不喜欢。:(
马克·阿默里'18

对于那个很抱歉。你是绝对正确的。我已经知道您答案中的其他所有内容,因此我快速浏览了一下。我应该更仔细地阅读它。
thebunnyrules '18

哇,您确实在答案中做了很多工作...谢谢。很高兴了解:$ _。从现在开始,我可以看到自己使用了很多东西。我编写了一个脚本来自动化整个过程,并放入了该线程中。试试吧,也许您会喜欢:stackoverflow.com/questions/1529946/…–
thebunnyrules

39

这么老的问题,但是也许我可以提出一个替代解决方案。

您可以使用该install程序复制文件并“即时”创建目标路径。

install -D file /path/to/copy/file/to/is/very/deep/there/file

但是,有一些方面需要考虑:

  1. 您还需要指定目标文件名,而不仅仅是目标路径
  2. 目标文件将是可执行文件(至少,据我从测试中看到的那样)

您可以通过添加-m用于在目标文件上设置权限的选项来轻松地修改#2 (例如:-m 664将创建具有权限的目标文件rw-rw-r--,就像使用来创建新文件一样touch)。


这是我受 = 启发的答案无耻链接


并不是真的适合我的用法,但是很酷!我会记住的。
thebunnyrules

请注意,此命令使用755rwxr-xr-x)权限创建目标文件,这可能是不希望的。你可以指定与别的-m开关,但我无法找到一种方法,只保留文件的权限:(
törzsmókus

19

可以执行您想要的操作的Shell函数,称其为“埋藏”副本,因为它为文件存在了一个洞:

bury_copy() { mkdir -p `dirname $2` && cp "$1" "$2"; }

1
您应该对周围的引号dirname $2
Tarnay卡尔曼

4
@Kalmi,为了获得正确的报价,您还想引用$2作为的参数dirname。喜欢mkdir -p "$(dirname "$2")"。如果没有这个报价$2cp是没有用的;)
迈克尔Krelin -黑客

3
请注意,此答案假设$ 2尚未成为目标目录,否则,您只希望将“ mkdir -p” $ 2“ && cp” $ 1“” $ 2“作为函数体
user467257 2014年

我认为这有一个错误,如@ user467257所指出。我认为这不是可以解决的,因为在这种情况下,$ 2在目标目录和目标文件之间是不明确的。我已经发布了解决此问题的替代答案。
丰富

@Rich,我不同意这是不可修复的。以下对文件和目录都适用:mkdir -p dirname "$2"&& cp -r“ $ 1”“ $ 2”; 注意,周围的反引号dirname $2不会出现,因为SO会将其解析为代码标记。
尤里

11

这是一种实现方法:

mkdir -p `dirname /path/to/copy/file/to/is/very/deep/there` \
   && cp -r file /path/to/copy/file/to/is/very/deep/there

dirname将为您提供目标目录或文件的父目录。然后,mkdir -p`dirname ...`将创建该目录,以确保当您调用cp -r时,正确的基本目录就位。

与--parents相比,此方法的优点在于,它适用于目标路径中最后一个元素是文件名的情况。

它将在OS X上运行。


6

install -D file -m 644 -t /path/to/copy/file/to/is/very/deep/there


1
实际上,根本不一样。他要求您两次传递文件名(因此带有“ -t”标志),并设置文件上的执行位(因此带有“ -m”示例)。他应该是最佳答案,因为它实际上并不能回答问题-而我的却可以。
海绵宝宝

1
这适用于单个文件。只需考虑这there是最终文件,而不是最终子目录。
kvantour

6

考虑到以上所有答案,我更喜欢按以下方式使用rsync:

$  rsync -a directory_name /path_where_to_inject_your_directory/

例:

$ rsync -a test /usr/local/lib/

我尝试了一下并得到以下结果:rsync -a bull / some / hoop / ti / doop / ploop结果rsync:mkdir“ / some / hoop / ti / doop / ploop”失败:没有这样的文件或目录(2)rsync错误:在main.c(675)[Receiver = 3.1.2]上的文件IO(代码11)中
出错-thebunnyrules

@thebunnyrules,您必须使用pwd 命令检查路径 (查看:[link](access.redhat.com/documentation/en-US/Red_Hat_Enterprise_Linux/…)),并将其正确插入指令中
marcdahan

如果我正确理解了您的建议,则表明我在命令中指定了源的完整路径,即使用:“ $(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可以是文件或目录。在任何情况下,下一个代码都应该起作用。

fsrc=/tmp/myfile.unk
fdst=/tmp/dir1/dir2/dir3/myfile.txt
mkdir -p $(dirname ${fdst}) && cp -p ${fsrc} ${fdst}

或特定于bash

fsrc=/tmp/myfile.unk
fdst=/tmp/dir1/dir2/dir3/myfile.txt
mkdir -p ${fdst%/*} && cp -p ${fsrc} ${fdst}

谢谢!这就是我所需要的。在我的情况下,我不知道目标位置是否具有目录以及该目录是否存在,因此此代码为我提供了父目录,然后我可以安全地创建它(如果它不存在)
xbmono

4

只需在.bashrc中添加以下内容,即可根据需要进行调整。在Ubuntu中工作。

mkcp() {
    test -d "$2" || mkdir -p "$2"
    cp -r "$1" "$2"
}

例如,如果您要将“测试”文件复制到目标目录“ d”使用,

mkcp test a/b/c/d

mkcp将首先检查目标目录是否存在,如果不存在,则将其复制并复制源文件/目录。


正如其他人在另一个答案中提到的那样,您无需在调用之前测试目录是否存在mkdir -p。即使它已经存在,它也会成功。
bfontaine


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假定您要复制的项目被放置在目标路径中,而不是目标本身(也就是源文件/文件夹的重命名副本)。含义:

  • 如果d是文件夹,则“ CP / a / b / c / d”将导致/ c / d / b
  • 如果/ c / b中的b是文件夹,则“ CP / a / b / c / b”将导致/ c / b / b。
  • 如果b和d均为文件:CP / a / b / c / d将生成/ c / d(其中d是b的副本)。在相同情况下,CP / a / b / c / b相同。

可以使用“ --rename”开关更改默认的CP行为。在这种情况下,假设

  • “ CP --rename / a / b / c / d”将b复制到/ c并将副本重命名为d。

结束语:与cp一样,CP一次可以复制多个项目,并且列出的最后一个路径被假定为目的地。只要您使用引号,它也可以处理带有空格的路径。

CP将检查您放入的路径,并确保它们在复制之前存在。在严格模式下(可通过--strict开关使用),所有要复制的文件/文件夹必须存在或不进行复制。在放松模式(松弛)下,如果您列出的至少一项存在,则复制将继续。宽松模式是默认模式,您可以通过开关临时更改模式,也可以通过在脚本开头设置变量easy_going永久更改模式。

安装方法如下:

在非root用户终端中,执行以下操作:

sudo echo > /usr/bin/CP; sudo chmod +x /usr/bin/CP; sudo touch /usr/bin/CP
gedit admin:///usr/bin/CP 

在gedit中,粘贴CP实用程序并保存:

#!/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 needed
while true; do
    eval input="\$$n"
    in_=${input::1}

    if [ -z "$input" -a $n = 1 ]; then input="--help"; fi 

    if [ "$input" = "-h" -o "$input" = "--help" -o "$input" = "-?" -o "$input" = "--help-all" ]; then
        if [ "$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
    fi

    if [ -z "$input" ]; then
        count_i=$(expr $count_i - 1 ) # remember, last item is destination and it's not included in cound
        break 
    elif [ "$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 destination
    if [ $count_i -lt 0 ]; then 
            echo "Error: You haven't listed any items for copying. Exiting." # you didn't put any items for copying
    elif [ $count_i -lt 1 ]; then
            echo "Error: Copying usually involves a destination. Exiting." # you put one item and no destination
    fi

#reset the counter and grab content of arguments, aka: switches and item paths
n=1
while true; do
        eval input="\$$n" #input=$1,$2,$3,etc...
        in_=${input::1} #first letter of $input

        if [ "$in_" = "-" ]; then
            if [ "$input" = "--rename" ]; then 
                rename=true #my custom switches
            elif [ "$input" = "--strict" ]; then 
                easy_going=false #exit script if even one of the non-destinations item is not found
            elif [ "$input" = "--relaxed" ]; then 
                easy_going=true #continue script if at least one of the non-destination items is found
            elif [ "$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\""
            fi                                  
        elif ! [ -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\""
                else
                    if [ "$i" -le "$count_i" ]; then 
                        eval item$i="$input" 
                        item_list="$item_list \"$input\""
                    else
                        destination="$input" #destination is last items entered
                    fi
                fi
        else
            i=0
            m=0
            n=1                     
            break
        fi      
        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
fi

if [ $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" ]; then
    if [ $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
    fi
fi
if [ "$rename" = true ]; then
    if [ $count_i -gt 1 ]; then
        if [ $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
        fi
    elif [ -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
        fi
    else
        dest_jr=$(dirname "$destination")
        if [ -d "$destination" ]; then item_list="$item1/*";fi
        mkdir -p "$dest_jr"
    fi
else
    mkdir -p "$destination"
fi

eval 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

这也许是一个矫over过正,但效率很高,按预期工作,谢谢您,先生。
Xedret

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上强制使用斜杠:

cp-p() {
  last=${@: -1}

  if [[ $# -ge 2 && "$last" == */ ]] ; then
    # cp SOURCE... DEST/
    mkdir -p "$last" && cp "$@"
  else
    echo "cp-p: (copy, creating parent dirs)"
    echo "cp-p: Usage: cp-p SOURCE... DEST/"
  fi
}

2

只是有同样的问题。我的方法是将文件像这样存档:

tar cf your_archive.tar file1 /path/to/file2 path/to/even/deeper/file3

tar会自动将文件存储在存档中的适当结构中。如果你跑

tar xf your_archive.tar

文件被提取到所需的目录结构中。


1

如上面的help_asap和spongeman所建议的,您可以使用“安装”命令将文件复制到现有目录,或者如果尚不存在则创建新的目标目录。

选项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中运行良好,将两个步骤(目录创建然后是文件复制)组合为一个步骤。


1

这已经很晚了,但可能对新手有所帮助。如果您需要“自动”创建文件夹,则rsync应该是您最好的朋友。rsync / path / to / sourcefile / path / to / tragetdir / thatdoestexist /


0
rsync file /path/to/copy/file/to/is/very/deep/there

如果您使用的是正确的,这可能会起作用rsync


这对我不起作用-目标目录上显示“无此文件或目录”
Rich

0

从源复制到不存在的路径

mkdir p /destination && cp r /source/ $_

注意:此命令复制所有文件

cp –r 用于复制所有文件夹及其内容

$_ 作为在最后一个命令中创建的目标



-1

假设您正在做类似的事情

cp file1.txt A / B / C / D / file.txt

其中A / B / C / D是尚不存在的目录

可能的解决方案如下

DIR=$(dirname A/B/C/D/file.txt)
# DIR= "A/B/C/D"
mkdir -p $DIR
cp file1.txt A/B/C/D/file.txt

希望有帮助!

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.