Использование SharpZipLib для распаковки определенных файлов?
Я пытаюсь использовать SharpZipLib для извлечения указанных файлов из zip-архива. Все примеры, которые я видел, всегда ожидают, что вы хотите распаковать всю молнию и сделать что-то вроде:
FileStream fileStreamIn = new FileStream (sourcePath, FileMode.Open, FileAccess.Read);
ZipInputStream zipInStream = new ZipInputStream(fileStreamIn);
ZipEntry entry;
while (entry = zipInStream.GetNextEntry() != null)
{
// Unzip file
}
то, что я хочу сделать, это что-то вроде:
ZipEntry entry = zipInStream.SeekToFile("FileName");
поскольку мои потребности включают использование zip в качестве пакета и только захват файлов в память по мере необходимости.
кто-нибудь знаком с SharpZipLib? Кто-нибудь знает, могу ли я сделать это без бега через всю молнию вручную?
4 ответов
zip-файл.GetEntry должен сделать трюк:
using (var fs = new FileStream(sourcePath, FileMode.Open, FileAccess.Read))
using (var zf = new ZipFile(fs)) {
var ze = zf.GetEntry(fileName);
if (ze == null) {
throw new ArgumentException(fileName, "not found in Zip");
}
using (var s = zf.GetInputStream(ze)) {
// do something with ZipInputStream
}
}
FastZip.ExtractZip (string zipFileName, string targetDirectory, string fileFilter)
можно извлечь один или несколько файлов на основе фильтра файлов (т. е. обычная строка expressoin)
вот документ, касающийся фильтра файлов:
// A filter is a sequence of independant <see cref="Regex">regular expressions</see> separated by semi-colons ';'
// Each expression can be prefixed by a plus '+' sign or a minus '-' sign to denote the expression
// is intended to include or exclude names. If neither a plus or minus sign is found include is the default
// A given name is tested for inclusion before checking exclusions. Only names matching an include spec
// and not matching an exclude spec are deemed to match the filter.
// An empty filter matches any name.
// </summary>
// <example>The following expression includes all name ending in '.dat' with the exception of 'dummy.dat'
// "+\.dat$;-^dummy\.dat$"
Итак, для файла с именем myfile.dat вы можете использовать"+.*файл myfile\.dat$ " в качестве фильтра файлов.
DotNetZip имеет индексатор строк в классе ZipFile, чтобы сделать это очень легко.
using (ZipFile zip = ZipFile.Read(sourcePath)
{
zip["NameOfFileToUnzip.txt"].Extract();
}
вам не нужно возиться с inputstreams и outputstreams и так далее, просто чтобы извлечь файл. С другой стороны, если вы хотите поток, вы можете получить его:
using (ZipFile zip = ZipFile.Read(sourcePath)
{
Stream s = zip["NameOfFileToUnzip.txt"].OpenReader();
// fiddle with stream here
}
вы также можете сделать по шаблону вырезать.
using (ZipFile zip = ZipFile.Read(sourcePath)
{
// extract all XML files in the archive
zip.ExtractSelectedEntries("*.xml");
}
существуют перегрузки для указания перезаписи / no-overwrite, различных целевых каталогов и т. д. Вы также можете извлечь на основе критерии, отличные от имени файла. Например, извлеките все файлы новее 15 января 2009 года:
// extract all files modified after 15 Jan 2009
zip.ExtractSelectedEntries("mtime > 2009-01-15");
и вы можете комбинировать критерии:
// extract all files that are modified after 15 Jan 2009) AND larger than 1mb
zip.ExtractSelectedEntries("mtime > 2009-01-15 and size > 1mb");
// extract all XML files that are modified after 15 Jan 2009) AND larger than 1mb
zip.ExtractSelectedEntries("name = *.xml and mtime > 2009-01-15 and size > 1mb");
вам не нужно извлекать файлы, которые вы выбираете. Вы можете просто выбрать их, а затем принять решение о том, следует ли извлекать или нет.
using (ZipFile zip1 = ZipFile.Read(ZipFileName))
{
var PhotoShopFiles = zip1.SelectEntries("*.psd");
// the selection is just an ICollection<ZipEntry>
foreach (ZipEntry e in PhotoShopFiles)
{
// examine metadata here, make decision on extraction
e.Extract();
}
}
Я получаю возможность VB.NET, для извлечения определенных файлов из архиваторов Zip. Комментарии на испанском языке. Возьмите путь к Zip-файлу, имя файла, которое вы хотите извлечь, пароль, если это необходимо. Извлечь в исходный путь, если OriginalPath имеет значение 1, или использовать DestPath, если OriginalPath=0. Следите за " ICSharpCode.SharpZipLib.Застежка-молния.Компрессия.Потоки.InflaterInputStream" вещи...
это выглядит следующим образом:
''' <summary>
''' Extrae un archivo específico comprimido dentro de un archivo zip
''' </summary>
''' <param name="SourceZipPath"></param>
''' <param name="FileName">Nombre del archivo buscado. Debe incluir ruta, si se comprimió usando guardar con FullPath</param>
''' <param name="DestPath">Ruta de destino del archivo. Ver parámetro OriginalPath.</param>
''' <param name="password">Si el archivador no tiene contraseña, puede quedar en blanco</param>
''' <param name="OriginalPath">OriginalPath=1, extraer en la RUTA ORIGINAL. OriginalPath=0, extraer en DestPath</param>
''' <returns></returns>
''' <remarks></remarks>
Public Function ExtractSpecificZipFile(ByVal SourceZipPath As String, ByVal FileName As String, _
ByVal DestPath As String, ByVal password As String, ByVal OriginalPath As Integer) As Boolean
Try
Dim fileStreamIn As FileStream = New FileStream(SourceZipPath, FileMode.Open, FileAccess.Read)
Dim fileStreamOut As FileStream
Dim zf As ZipFile = New ZipFile(fileStreamIn)
Dim Size As Integer
Dim buffer(4096) As Byte
zf.Password = password
Dim Zentry As ZipEntry = zf.GetEntry(FileName)
If (Zentry Is Nothing) Then
Debug.Print("not found in Zip")
Return False
Exit Function
End If
Dim fstr As ICSharpCode.SharpZipLib.Zip.Compression.Streams.InflaterInputStream
fstr = zf.GetInputStream(Zentry)
If OriginalPath = 1 Then
Dim strFullPath As String = Path.GetDirectoryName(Zentry.Name)
Directory.CreateDirectory(strFullPath)
fileStreamOut = New FileStream(strFullPath & "\" & Path.GetFileName(Zentry.Name), FileMode.Create, FileAccess.Write)
Else
fileStreamOut = New FileStream(DestPath + "\" + Path.GetFileName(Zentry.Name), FileMode.Create, FileAccess.Write)
End If
Do
Size = fstr.Read(buffer, 0, buffer.Length)
fileStreamOut.Write(buffer, 0, Size)
Loop While (Size > 0)
fstr.Close()
fileStreamOut.Close()
fileStreamIn.Close()
Return True
Catch ex As Exception
Return False
End Try
End Function