Ответ 1
Да, добавьте параметр -Force
.
copy-item $from $to -Recurse -Force
$from = "\\something\1 XLS\2010_04_22\*"
$to = "c:\out\1 XLS\2010_04_22\"
copy-item $from $to -Recurse
Это работает, если c:\out\1 XLS\2010_04_22\
существует. Можно ли с помощью одной команды создать c:\out\1 XLS\2010_04_22\
если он не существует?
Да, добавьте параметр -Force
.
copy-item $from $to -Recurse -Force
В PowerShell 2.0 по-прежнему невозможно получить командлет Copy-Item для создания папки назначения, вам понадобится следующий код:
$destinationFolder = "C:\My Stuff\Subdir"
if (!(Test-Path -path $destinationFolder)) {New-Item $destinationFolder -Type Directory}
Copy-Item "\\server1\Upgrade.exe" -Destination $destinationFolder
Если вы используете -Recurse в Copy-Item, он создаст все вложенные папки исходной структуры в получателе, но не создаст фактическую папку назначения даже с помощью параметра -Force.
В PowerShell 3 и выше я использую Copy-Item with New-Item.
copy-item -Path $file -Destination (new-item -type directory -force ("C:\Folder\sub\sub\" + $newSub)) -force -ea 0
Я не пробовал это в версии 2.
$filelist | % {
$file = $_
mkdir -force (Split-Path $dest) | Out-Null
cp $file $dest
}
function Copy-File ([System.String] $sourceFile, [System.String] $destinationFile, [Switch] $overWrite) {
if ($sourceFile -notlike "filesystem::*") {
$sourceFile = "filesystem::$sourceFile"
}
if ($destinationFile -notlike "filesystem::*") {
$destinationFile = "filesystem::$destinationFile"
}
$destinationFolder = $destinationFile.Replace($destinationFile.Split("\")[-1],"")
if (!(Test-Path -path $destinationFolder)) {
New-Item $destinationFolder -Type Directory
}
try {
Copy-Item -Path $sourceFile -Destination $destinationFile -Recurse -Force
Return $true
} catch [System.IO.IOException] {
# If overwrite enabled, then delete the item from the destination, and try again:
if ($overWrite) {
try {
Remove-Item -Path $destinationFile -Recurse -Force
Copy-Item -Path $sourceFile -Destination $destinationFile -Recurse -Force
Return $true
} catch {
Write-Error -Message "[Copy-File] Overwrite error occurred!`n$_" -ErrorAction SilentlyContinue
#$PSCmdlet.WriteError($Global:Error[0])
Return $false
}
} else {
Write-Error -Message "[Copy-File] File already exists!" -ErrorAction SilentlyContinue
#$PSCmdlet.WriteError($Global:Error[0])
Return $false
}
} catch {
Write-Error -Message "[Copy-File] File move failed!`n$_" -ErrorAction SilentlyContinue
#$PSCmdlet.WriteError($Global:Error[0])
Return $false
}
}
Моим любимым является использование класса .Net [IO.DirectoryInfo], который позаботится о некоторой логике. Я действительно использую это для множества подобных задач сценариев. Он имеет метод .Create(), который создает каталоги, которые не существуют, без ошибок, если они это делают.
Поскольку это еще проблема с двумя шагами, я использую псевдоним foreach, чтобы он был простым. Для отдельных файлов:
[IO.DirectoryInfo]$to |% {$_.create(); cp $from $_}
Что касается вашего файла с несколькими файлами/каталогами, я бы использовал RoboCopy над xcopy. Удалите "*" из своего и просто используйте:
RoboCopy.exe $from $to *
Вы все равно можете добавить /r (Recurse),/e (Recurse, включая Empty), а также 50 других полезных переключателей.
Изменить: Оглядываясь на это, он краток, но не очень читабельен, если вы часто не используете код. Обычно я разбил его на две части, например:
([IO.DirectoryInfo]$to).Create()
cp $from $to
Кроме того, DirectoryInfo является типом родительского свойства FileInfo, поэтому, если ваш $to является файлом, вы можете использовать их вместе:
([IO.FileInfo]$to).Parent.Create()
cp $from $to
Вот пример, который работал для меня. У меня был список из 500 конкретных файлов в текстовом файле, содержащий около 100 различных папок, которые я должен был скопировать в резервную копию на случай, если эти файлы понадобятся позже. Текстовый файл содержал полный путь и имя файла, по одному на строку. В моем случае я хотел убрать букву диска и имя первой подпапки из каждого имени файла. Я хотел скопировать все эти файлы в похожую структуру папок в другой корневой папке назначения, которую я указал. Я надеюсь, что другие пользователи найдут это полезным.
# Copy list of files (full path + file name) in a txt file to a new destination, creating folder structure for each file before copy
$rootDestFolder = "F:\DestinationFolderName"
$sourceFiles = Get-Content C:\temp\filelist.txt
foreach($sourceFile in $sourceFiles){
$filesplit = $sourceFile.split("\")
$splitcount = $filesplit.count
# This example strips the drive letter & first folder ( ex: E:\Subfolder\ ) but appends the rest of the path to the rootDestFolder
$destFile = $rootDestFolder + "\" + $($($sourceFile.split("\")[2..$splitcount]) -join "\")
# Output List of source and dest
Write-Host ""
Write-Host "===$sourceFile===" -ForegroundColor Green
Write-Host "+++$destFile+++"
# Create path and file, if they do not already exist
$destPath = Split-Path $destFile
If(!(Test-Path $destPath)) { New-Item $destPath -Type Directory }
If(!(Test-Path $destFile)) { Copy-Item $sourceFile $destFile }
}
Я дважды наткнулся на это, и этот последний раз был уникальной ситуацией, и даже если я использую copy-item
, я хотел опубликовать решение, которое использовал.
Имеет список только файлов с полным путем, и в большинстве случаев файлы не имеют расширений. параметр -Recurse -Force
не будет работать для меня, поэтому я отбросил функцию copy-item
и вернулся к чему-то вроде ниже, используя xcopy, поскольку я все еще хотел сохранить его одним лайнером. Первоначально я был связан с Robocopy, но он, по-видимому, ищет расширение файла, и поскольку многие из моих не имели расширения, он считал его каталогом.
$filelist = @("C:\Somepath\test\location\here\file","C:\Somepath\test\location\here2\file2")
$filelist | % { echo f | xcopy $_ $($_.Replace("somepath", "somepath_NEW")) }
Надеюсь, это поможет кому-то.