zlib 简单封装
下列代码用于压缩和解压字符串,使用标准库string。实现了对zlib的简单封装。
#pragma once
#include <boost/noncopyable.hpp>
#include <zlib.h>
#include <string>
#include <cassert>
#include <strings.h> // for bzero
class ZlibDecompressor : boost::noncopyable {
public:
ZlibDecompressor(const std::string &input)
: zerror_(Z_OK),
z_init_error_(Z_OK),
input_(input),
output_()
{
::bzero(&zstream_, sizeof(zstream_));
z_init_error_ = inflateInit(&zstream_);
if (z_init_error_ == Z_OK)
zerror_ = decompress();
}
const char *zlibErrorMessage() const { return zstream_.msg; }
int zlibErrorCode() const { return zerror_; }
~ZlibDecompressor()
{
if (z_init_error_ == Z_OK)
inflateEnd(&zstream_);
}
std::string output() const
{
if(valid())
return output_;
return output_;
}
bool valid() const { return zerror_ == Z_OK; }
private:
const static int CHUNK = 8192;
int decompress() {
int ret;
size_t begin = 0;
size_t size = input_.size();
unsigned char out[CHUNK];
do {
int chunk = ((size - begin) < CHUNK ? size - begin : CHUNK);
if (chunk == 0)
break;
zstream_.avail_in = static_cast<uint>(chunk);
zstream_.next_in = (Bytef *) &input_[begin];
do
{
zstream_.avail_out = CHUNK;
zstream_.next_out = (Bytef*)out;
ret = inflate(&zstream_, Z_NO_FLUSH);
assert(ret != Z_STREAM_ERROR);
switch (ret)
{
case Z_NEED_DICT:
ret = Z_DATA_ERROR;
case Z_DATA_ERROR:
case Z_MEM_ERROR:
return ret;
}
int have = CHUNK - static_cast<int>(zstream_.avail_out);
output_.append(out, out + have);
} while (zstream_.avail_out == 0);
begin += chunk;
} while (ret != Z_STREAM_END);
return ret;
}
z_stream zstream_;
int zerror_;
int z_init_error_;
std::string input_;
std::string output_;
};
class ZlibCompressor : boost::noncopyable {
public:
explicit ZlibCompressor(const std::string &input)
: zerror_(Z_OK),
z_init_error_(Z_OK),
input_(input),
output_()
{
::bzero(&zstream_, sizeof(zstream_));
z_init_error_ = deflateInit(&zstream_, -1);
if(z_init_error_ == Z_OK)
zerror_ = compress();
}
explicit ZlibCompressor(const std::string& input, int level)
: zerror_(Z_OK),
z_init_error_(Z_OK),
input_(input),
output_()
{
assert(level >= -1 && level <= 9);
::bzero(&zstream_, sizeof(zstream_));
z_init_error_ = deflateInit(&zstream_, level);
if(z_init_error_ == Z_OK)
zerror_ = compress();
}
~ZlibCompressor() {
if(z_init_error_ == Z_OK)
deflateEnd(&zstream_);
}
const char *zlibErrorMessage() const { return zstream_.msg; }
int zlibErrorCode() const { return zerror_; }
std::string output() const
{
if(valid())
return output_;
return "";
}
bool valid() const { return zerror_ == Z_OK; }
private:
const static int CHUNK = 8192;
int compress() {
unsigned char out[CHUNK];
size_t size = input_.size();
size_t begin = 0;
int ret;
int flush;
do
{
int chunk;
if(size - begin <= CHUNK)
{
chunk = size - begin;
flush = Z_FINISH;
}
else
{
chunk = CHUNK;
flush = Z_NO_FLUSH;
}
zstream_.avail_in = chunk;
zstream_.next_in = (Bytef *)&input_[begin];
do
{
zstream_.avail_out = CHUNK;
zstream_.next_out = out;
ret = deflate(&zstream_, flush);
assert(ret != Z_STREAM_ERROR);
int have = CHUNK - static_cast<int>(zstream_.avail_out);
output_.append(out, out + have);
}while(zstream_.avail_out == 0);
assert(zstream_.avail_in == 0);
begin += chunk;
}while(flush != Z_FINISH);
assert(ret == Z_STREAM_END);
return Z_OK;
}
z_stream zstream_;
int zerror_;
int z_init_error_;
std::string input_;
std::string output_;
};
zlib 简单封装的更多相关文章
- Android AsyncTask 深度理解、简单封装、任务队列分析、自定义线程池
前言:由于最近在做SDK的功能,需要设计线程池.看了很多资料不知道从何开始着手,突然发现了AsyncTask有对线程池的封装,so,就拿它开刀,本文将从AsyncTask的基本用法,到简单的封装,再到 ...
- FMDB简单封装和使用
工具:火狐浏览器+SQLite Manager插件 ; Xcode; FMDB库; 效果: 项目地址: https://github.com/sven713/PackFMDB 主要参考这两篇博客: 1 ...
- Android--Retrofit+RxJava的简单封装(三)
1,继续接着上一篇的讲讲,话说如果像上一篇这样的话,那么我们每一次请求一个结构都要创建一堆的Retrofit对象,而且代码都是相同的,我们可以试试封装一下 先创建一个HttpMethods类,将Ret ...
- okhttp3 get post 简单封装
最近打算在新项目中使用 okhttp3, 简单封装了一下异步 get post 因为 CallBack 也是在子线程中执行,所以用到了 Handler public class MyOkHttpCli ...
- python网页请求urllib2模块简单封装代码
这篇文章主要分享一个python网页请求模块urllib2模块的简单封装代码. 原文转自:http://www.jbxue.com/article/16585.html 对python网页请求模块ur ...
- 对pymysql的简单封装
#coding=utf-8 #!/usr/bin/python import pymysql class MYSQL: """ 对pymysql的简单封装 "& ...
- iOS开发——UI篇OC篇&UITableView简单封装
UITableView简单封装 UITableView时iOS开发中使用最多也是最重的一个UI空间,其实在App Store里面的%80以上的应用都用到了这个控件,所以就给大家介绍一下,前面的文章中也 ...
- iOS sqlite 增删改查 简单封装(基于 FMDB)
/** * 对 sqlite 的使用进行简单封装,仅涉及简单的单表 增删改查 * * 基于 FMDB * * 操作基于 model ,数据库表字段与 model 属性一一对应,对 model 整 ...
- ADO简单封装(MFC)
简单封装了一下,不是很严谨. /************************************************************************/ /* INSTRUC ...
随机推荐
- MATLAB——BP神经网络
1.使用误差反向传播(error back propagation )的网络就叫BP神经网络 2.BP网络的特点: 1)网络由多层构成,层与层之间全连接,同一层之间的神经元无连接 . 2)BP网络的传 ...
- PAT A1104 Sum of Number Segments (20 分)——数学规律,long long
Given a sequence of positive numbers, a segment is defined to be a consecutive subsequence. For exam ...
- Selenium:三种等待方式
UI自动化测试,大多都是通过定位页面元素来模拟实际的生产场景操作.但在编写自动化测试脚本中,经常出现元素定位不到的情况,究其原因,无非两种情况:1.有frame:2.没有设置等待. 因为代码运行速度和 ...
- JAVA中map的分类和各自的特性
java为数据结构中的映射定义了一个接口java.util.Map,他实现了四个类,分别是:HashMap,HashTable,LinkedHashMapTreeMap Map不允许键重复,但允许值重 ...
- php中按值传递和按引用传递的一个问题
php中传递变量默认是按照值传递. 简单举个例子: <?php function testArray($arr){// &$arr $arr = array(1,2,3,); } $ar ...
- 搭建mysql cluster
虚拟机搭建Mysql Cluster 参考文档:http://www.cnblogs.com/jackluo/archive/2013/01/19/2868152.html http://www.cn ...
- C# 判断一个文本文件的编码格式(转载)
文件的字符集在Windows下有两种,一种是ANSI,一种Unicode.对于Unicode,Windows支持了它的三种编码方式,一种是小尾编码(Unicode),一种是大尾编码(BigEndian ...
- Caffe源码中syncedmem文件分析
Caffe源码(caffe version:09868ac , date: 2015.08.15)中有一些重要文件,这里介绍下syncedmem文件. 1. include文件: (1).& ...
- Python 学习 第九篇:模块
模块是把程序代码和数据封装的Python文件,也就是说,每一个以扩展名py结尾的Python源代码文件都是一个模块.每一个模块文件就是一个独立的命名空间,用于封装顶层变量名:在一个模块文件的顶层定义的 ...
- Crackme006 - 全新160个CrackMe学习系列(图文|视频|注册机源码)
知乎:逆向驿站 原文链接 CrackMe006 | 难度适中适合练手 |160个CrackMe深度解析(图文+视频+注册机源码) crackme006,依然是delphi的,而且没壳子,条线比较清晰, ...