특정 폴더에 포함된 파일 목록과 파일 생성 정보를 얻어온 후에 zlib를 활용하면 폴더를 쉽게 압축할 수 있습니다.
FolderList.h 파일
#pragma once
#include <Windows.h>
#include <vector>
#include <string>
class FolderList
{
public:
struct FLIST{
std::string filepath_;
SYSTEMTIME time_;
};
typedef std::vector<FLIST> ZIP_FLIST;
public:
FolderList(void);
~FolderList(void);
void operator() ( std::string foldername, ZIP_FLIST& mlist);
};
FolderList.cpp 파일
#include "FolderList.h"
FolderList::FolderList(void)
{
}
FolderList::~FolderList(void)
{
}
void
FolderList::operator() ( std::string foldername, ZIP_FLIST& mlist)
{
std::string sfolder= foldername+"\\*.*";
WIN32_FIND_DATAA ffd;
HANDLE hFind = FindFirstFileA(sfolder.c_str(),&ffd);
if( INVALID_HANDLE_VALUE == hFind) return;
FLIST flist;
FILETIME ltime;
do{
if( !_strcmpi(ffd.cFileName, ".") || !_strcmpi(ffd.cFileName, "..") ) continue;
flist.filepath_ = foldername+"\\"+ffd.cFileName;
if( ffd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY){
operator() ( flist.filepath_,mlist);
continue;
}
FileTimeToLocalFileTime(&ffd.ftCreationTime,<ime);
FileTimeToSystemTime(<ime,&flist.time_);
mlist.push_back(flist);
} while( FindNextFileA(hFind,&ffd));
}
ZipCTL.h 파일
#pragma once
#include <LibZ/zip.h>
#include "FolderList.h"
class ZipCTL
{
private:
zipFile zf_;
public:
explicit ZipCTL(const char* zipfilename, int append = APPEND_STATUS_CREATE);
~ZipCTL(void);
void make(const char* root_folder, FolderList::ZIP_FLIST& mlist);
};
ZipCTL.cpp 파일
#include "ZipCTL.h"
#include <fstream>
#include <iostream>
ZipCTL::ZipCTL(const char* zipfilename, int append)
:zf_( zipOpen(zipfilename, append) )
{
}
ZipCTL::~ZipCTL(void)
{
zipClose(zf_,"");
}
void
ZipCTL::make(const char* root_folder, FolderList::ZIP_FLIST& mlist){
zip_fileinfo info;
memset(&info,0,sizeof(zip_fileinfo));
size_t len = strlen(root_folder);
const int BUF = 1024;
char in[BUF];
size_t readsize(0);
for( auto& item:mlist){
SYSTEMTIME& mtime = item.time_;
info.tmz_date.tm_hour = mtime.wHour;
info.tmz_date.tm_mday = mtime.wDay;
info.tmz_date.tm_min = mtime.wMinute;
info.tmz_date.tm_mon = mtime.wMonth-1;
info.tmz_date.tm_sec = mtime.wSecond;
info.tmz_date.tm_year= mtime.wYear;
const char* m = item.filepath_.c_str();
zipOpenNewFileInZip(zf_,m+len,&info, NULL,0,NULL,0,NULL,Z_DEFLATED, Z_BEST_SPEED);
std::ifstream fp(m, std::ios_base::binary);
do{
fp.read(in,BUF);
readsize = (size_t)fp.gcount();
zipWriteInFileInZip(zf_, in,readsize);
} while( !fp.eof() );
fp.close();
zipCloseFileInZip(zf_);
std::cout<<std::endl<<m<<" Complete Compress!!";
}
}
zip_folder.cpp 파일
// zip_folder.cpp : Defines the entry point for the console application.
//
#include "stdafx.h"
#include "FolderList.h"
#include "ZipCTL.h"
#include <iostream>
int _tmain(int argc, _TCHAR* argv[])
{
{
// 폴더내 파일 목록
FolderList::ZIP_FLIST flist;
FolderList fn;
fn("C:\\Python27", flist);
for( auto& item:flist){
std::cout<<std::endl<< item.filepath_;
}
// 압축
ZipCTL zc("test.zip");
zc.make("C:\\Python27", flist);
}
std::cout<<std::endl;
system("pause");
return 0;
}
thanks you very!
안녕하세요 좋은 영상과 글을 잘 봤습니다.
글에 보면 “입력 버퍼와 출력 버퍼를 고정으로한 압축 함수”에 대해서 어떤 함수를 봐야 하는지 알수 있을까요?
좋은 글 감사드리며, 오래된 글에 문의를 남깁니다.
감사합니다.
[compress,uncompress 함수 사용하기] 동영상입니다.
동영상 잘 보았습니다. 궁금한것이 있는데요, Z_SYNC_FLUSH 를 사용할 경우에, 만약 next_out buffer 크기가 압축한 data 를 담기에 충분하지 않는다면, 어떻게 해야 하는지요?
Z_SYNC_FLUSH는 쓰고 있는 비트를 모두 바이트 단위로 쓰라는 옵션임으로 일반적인 쓰기 과정에서 발생한 에러 코드를 반환하게 되어 있습니다. [Deflate 샘플] 동영상을 참고하세요.
C로 압축하고 c#으로 압축을 푸는 방법이 있을까요?
텍스트 파일을 작성시
gzwritre(파일 포인터, 2019, sizeof(int));
gzwritre(파일 포인터, 09, sizeof(int));
gzwritre(파일 포인터, 17, sizeof(int));
gzwritre(파일 포인터, 0.1, sizeof(float)); 결과값
gzwritre(파일 포인터, 0.2, sizeof(float));
gzwritre(파일 포인터, 0.3, sizeof(float));
…
gzwritre(파일 포인터, 100.0, sizeof(float));
이렇게 텍스트 파일을 저장(압축)했는데 C#에서 이 파일을 읽어서
연 / 월 / 일 을 정수 변수에
결과값을 실수배열에 저장하고 싶습니다.
C#으로 포팅된 zip라이브러리를 사용하시면 문제없이 될겁니다. 예를 들어 SharpZipLib과 같은~
http://icsharpcode.github.io/SharpZipLib/
https://github.com/icsharpcode/SharpZipLib/wiki/Zip-Samples
https://icsharpcode.github.io/SharpZipLib/help/api/ICSharpCode.SharpZipLib.GZip.GZip.html
분할 압축파일일 경우 zilb로 압축 해제가 가능 한가요?
아니면 다른 라이브러리를 사용해야 할까요 ?
별도로 지원하는 함수는 없지만 그냥 저장하는 과정에서 파일만 나눠 저장하는 것이 아닐지요.
안녕하세요
c++에서 zlib uncompress로 .emf image 파일을 압축해제 하려고 합니다.
그런데 손상된 파일이을 대상으로 진행할 경우 Z_DATA_ERROR을 반환하면서 자동종료됩니다.
압축된 파일이 Z_DATA_ERROR를 반환하면(손상된 파일이면) 빈 .emf image 파일로 압축해제 되도록 하려면 어떻게 해야할까요?
Z_DATA_ERROR이면 그냥 빈.emf image 파일을 생성해주면 되는 것 아닌가요?
리눅스에서 C언어를 사용해서 zlib으로 위와 같이 폴더(디렉토리)를 압축할수있나요?
libzip에 포함된 contrib/minizip를 함께 사용하시면 가능합니다.