GZIP 对文件和响应数据 都能压缩

本文介绍了一种使用gzip压缩技术对HTTP响应进行压缩的方法,包括创建自定义的ServletResponse包装类,实现gzip压缩流的封装,以及通过过滤器判断浏览器支持情况并进行响应压缩。适用于希望减少网络传输流量、提升网页加载速度的应用场景。
package com.chinalife.datacompress;

import java.io.IOException;
import java.io.PrintWriter;

import javax.servlet.ServletOutputStream;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpServletResponseWrapper;
/**
 * 描述:servletResponse包装类<br>
 * 那么子类化HttpServletResponseWrapper都需要重写那些方法呢
 * 获取response对象,并将输出存放在自定义的流里面,那么关于输出流(outputStream、writer)的操作都是需要重写的了:
 * 1)以流的方式获取输出——重写getOutputStream()
 * 2)以字符方式获取输出——重写getWriter()
 * 3)刷新流——重写flushBuffer()
 * 4)重置流——重写reset()
 * 然后加入新增的获取输出数据的方法就ok了
 * 作者:万鸿辉 <br>
 * 修改日期:2012-9-19上午09:37:49 <br>
 * E-mail: wanhonghui@126.com <br>
 */
public class CompressWapper extends HttpServletResponseWrapper {
    private ServletOutputStream servletOutputStream;
    private PrintWriter printWriter;
    private GzipOutPutStream gizpStream;

    public CompressWapper(HttpServletResponse response) throws IOException {
        super(response);
        gizpStream = new GzipOutPutStream(response.getOutputStream());
        servletOutputStream = gizpStream;
        printWriter = new PrintWriter(gizpStream);
    }

    /**
     * 方法名称:flushBuffer <br>
     * 描述:刷新流 <br>
     * 作者:Administrator <br>
     * 修改日期:2012-9-19上午09:43:34
     * @see javax.servlet.ServletResponseWrapper#flushBuffer()
     * @throws IOException
     */
    public void flushBuffer() throws IOException {
        if(servletOutputStream != null){
            servletOutputStream.flush();
        }
        if(printWriter != null){
            printWriter.flush();
        }
        gizpStream.close();
    }

    /**
     * 方法名称:getOutputStream <br>
     * 描述:获取字节流 <br>
     * 作者:Administrator <br>
     * 修改日期:2012-9-19上午09:44:01
     * @see javax.servlet.ServletResponseWrapper#getOutputStream()
     * @return
     * @throws IOException
     */
    public ServletOutputStream getOutputStream() throws IOException {
        return servletOutputStream;
    }

    /**
     * 方法名称:getWriter <br>
     * 描述:获取字符流 <br>
     * 作者:Administrator <br>
     * 修改日期:2012-9-19上午09:44:19
     * @see javax.servlet.ServletResponseWrapper#getWriter()
     * @return
     * @throws IOException
     */
    public PrintWriter getWriter() throws IOException {
        return printWriter;
    }

    /**
     * 方法名称:reset <br>
     * 描述:重置流 <br>
     * 作者:Administrator <br>
     * 修改日期:2012-9-19上午09:44:30
     * @see javax.servlet.ServletResponseWrapper#reset()
     */
    public void reset() {
        try {
            gizpStream.reset();
        } catch (IOException e) {
            throw new RuntimeException(e);
        }
    }

    /**
     * 方法名称:resetBuffer <br>
     * 描述:重置流 <br>
     * 作者:Administrator <br>
     * 修改日期:2012-9-19上午09:44:30
     * @see javax.servlet.ServletResponseWrapper#reset()
     */
    public void resetBuffer() {
        try {
            gizpStream.reset();
        } catch (IOException e) {
            throw new RuntimeException(e);
        }
    }

    
    
    
}


package com.chinalife.datacompress;

import java.io.IOException;
import java.util.Enumeration;

import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

/**
 * 描述:通过filter进行文件压缩<br>
 * 作者:万鸿辉 <br>
 * 修改日期:2012-9-19上午09:11:22 <br>
 * E-mail: wanhonghui@126.com <br>
 */
public class FileCompressFilter implements Filter{

    /**
     * 方法名称:destroy <br>
     * 描述:过滤器销毁方法 <br>
     * 作者:万鸿辉 <br>
     * 修改日期:2012-9-19上午09:15:33
     * @see javax.servlet.Filter#destroy()
     */
    public void destroy() {
        
    }
    
    /**
     * 方法名称:doFilter <br>
     * 描述:通过过滤器来拦截请求,压缩对应的文件,通过gzip算法进行压缩处理 <br>
     * 作者:万鸿辉 <br>
     * 修改日期:2012-9-19上午09:21:36
     * @see javax.servlet.Filter#doFilter(javax.servlet.ServletRequest, javax.servlet.ServletResponse, javax.servlet.FilterChain)
     * @param servletRequest
     * @param servletResponse
     * @param filterChain
     * @throws IOException
     * @throws ServletException
     */
    public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse,
            FilterChain filterChain) throws IOException, ServletException {
        HttpServletRequest request = (HttpServletRequest) servletRequest;
        HttpServletResponse response = (HttpServletResponse) servletResponse;
        CompressWapper wapper = new CompressWapper(response);
        Enumeration<String> enumer =request.getHeaders("Accept-Encoding"); //通过名称获取请求头信息
        
        boolean isBrowserCompress = false; //浏览器是否支持gzip压缩
        while(enumer.hasMoreElements()){
            String acceptEncoding = enumer.nextElement();
            if(acceptEncoding.indexOf("gzip") != -1){
                isBrowserCompress = true;
            }
        }
        //如果浏览器支持,则进行文件压缩
        if(isBrowserCompress){
            response.addHeader("Content-Encoding", "gzip");
            filterChain.doFilter(servletRequest, wapper);
            wapper.flushBuffer();
        }
        //如果浏览器不支持,则直接放行
        else{
            filterChain.doFilter(servletRequest, servletResponse);
        }
        
    }

    /**
     * 方法名称:init <br>
     * 描述:过滤器初始化方法 <br>
     * 作者:万鸿辉 <br>
     * 修改日期:2012-9-19上午09:15:07
     * @see javax.servlet.Filter#init(javax.servlet.FilterConfig)
     * @param filterConfig
     * @throws ServletException
     */
    public void init(FilterConfig filterConfig) throws ServletException {
        
    }


}

package com.chinalife.datacompress;

import java.io.IOException;
import java.io.OutputStream;
import java.util.zip.GZIPOutputStream;

import javax.servlet.ServletOutputStream;

/**
 * 描述:对数据压缩流<br>
 * 作者:Administrator <br>
 * 修改日期:2012-9-19上午09:52:11 <br>
 * E-mail:  <br>
 */
public class GzipOutPutStream extends ServletOutputStream {
    private GZIPOutputStream gzipstStream;
    private OutputStream stream ;
    
    
    public GzipOutPutStream(OutputStream stream) throws IOException {
        this.stream = stream;
        reset();
    }

    public void write(int b) throws IOException {
        gzipstStream.write(b);
    }

    public void write(byte[] b) throws IOException {
        gzipstStream.write(b,0,b.length);
    }


    public void write(byte[] b, int off, int len) throws IOException {
        gzipstStream.write(b, off, len);
    }


    public void flush() throws IOException {
        gzipstStream.flush();
    }


    public void close() throws IOException {
        gzipstStream.close();
    }
    
    public void reset() throws IOException{
        gzipstStream = new GZIPOutputStream(stream);
    }
    

}

 <filter>
      <filter-name>fileFilter</filter-name>
      <filter-class>com.chinalife.datacompress.FileCompressFilter</filter-class>
  </filter>
  <filter-mapping>
      <filter-name>fileFilter</filter-name>
      <url-pattern>/*</url-pattern>
  </filter-mapping>


<think> 我们讨论的是LoadRunner中如何检查响应是否被GZIP压缩。这个问题通常出现在性能测试中,因为服务器返回的响应如果被压缩,而我们没有正确处理,可能导致关联函数失败(如Error -26377)或其他问题。 在LoadRunner中,我们可以通过以下方法来检查响应是否被GZIP压缩: 1. **查看响应头信息**: 在脚本回放时,启用详细日志(Extended Log)并勾选“Data returned by server”。回放脚本后,在日志中搜索响应头。如果响应头中包含`Content-Encoding: gzip`,则表明响应GZIP压缩了。 2. **使用网络抓包工具**: 在运行LoadRunner脚本的同时,使用网络抓包工具(如Wireshark、Fiddler)捕获HTTP流量。在抓包工具中查看响应的HTTP头,同样寻找`Content-Encoding: gzip`字段。 3. **在脚本中检查响应头**: 在LoadRunner脚本中,我们可以使用关联函数来捕获响应头中的`Content-Encoding`字段,然后判断其值是否为gzip。 示例代码: ```c web_reg_save_param("ContentEncoding", "LB=Content-Encoding: ", "RB=\r\n", "Search=Headers", LAST); // 发起请求 web_url(...); // 检查捕获的值 if (strstr(lr_eval_string("{ContentEncoding}"), "gzip") != NULL) { lr_output_message("响应GZIP压缩了!"); } else { lr_output_message("响应未被压缩。"); } ``` 4. **手动检查响应体**: 如果响应GZIP压缩,那么通过日志或抓包工具看到的响应体将是二进制数据(乱码)。而正常的文本响应(如HTML、JSON)应该是可读的。 5. **在运行时设置中启用解压缩**: LoadRunner本身支持自动解压GZIP响应。在Runtime Settings中,可以设置: - 进入Runtime Settings -> Preferences -> Options - 设置`Download non-HTML resources`为`Yes` - 设置`Use zlib for Content-Encoding decompression`为`Yes` 这样,LoadRunner会自动解压缩响应,我们在日志中看到的就是解压后的内容。但是,这并不影响我们检查原始的响应头。 如果确认响应GZIP压缩,而我们希望关联函数能够正确处理压缩后的响应,那么需要确保: - 在脚本中启用了LoadRunner的自动解压缩(如上设置)。 - 或者,如果我们自己处理压缩数据,则需要先解压再关联(不常见,因为LoadRunner通常自动处理)。 注意:在关联函数(如`web_reg_save_param`)执行之前,LoadRunner会自动解压响应(如果设置了自动解压缩),因此关联函数处理的是解压后的文本。所以,我们通常不需要手动解压。但是,如果服务器返回了压缩内容而LoadRunner没有自动解压,那么关联函数将无法在二进制数据中查找文本边界,从而可能导致关联失败(如Error -26377)。 因此,当遇到关联失败时,除了检查边界值、参数长度等,也要检查响应是否被压缩而未被正确解压。 最后,如果我们在脚本中启用了自动解压缩,那么关联函数看到的响应就是解压后的文本,此时关联函数的边界值应该按照解压后的文本来设置。 相关问题: 1. 如何在LoadRunner中启用自动解压缩? 2. 如果服务器返回GZIP压缩响应,LoadRunner如何处理? 3. 除了GZIP,还有哪些常见的HTTP压缩方式? 4. 如何通过LoadRunner脚本解压GZIP数据? </think> ### 如何检查数据响应是否被GZIP压缩 在LoadRunner中检查服务器响应是否被GZIP压缩,可以通过以下三种方法实现: --- #### **方法 1:查看响应头(推荐)** 在脚本回放日志中检查响应头信息: 1. 开启详细日志: ```c // 脚本开头添加 web_set_option("LogDetail", "EXTENDED_LOG", "1"); // 启用扩展日志 ``` 2. 回放脚本后,在日志中搜索 `Content-Encoding` 响应头: ```log HTTP/1.1 200 OK Content-Encoding: gzip <-- 存在此字段表示响应压缩[^1] Content-Length: 358 ``` - ✅ **压缩标识**:`gzip`/`deflate`/`br` - ❌ **未压缩**:无此字段或值为 `identity` --- #### **方法 2:使用关联函数捕获响应头** 在脚本中动态检查压缩状态: ```c // 注册关联函数捕获Content-Encoding web_reg_save_param_ex( "ParamName=ContentEncoding", "LB/IC=Content-Encoding: ", "RB/IC=\r\n", "Search=Headers", // 关键:指定搜索响应头 LAST ); // 发起请求 web_url("api.example.com", ...); // 检查结果 if (strstr(lr_eval_string("{ContentEncoding}"), "gzip") != NULL) { lr_output_message("响应已被GZIP压缩"); } else { lr_output_message("响应压缩"); } ``` --- #### **方法 3:分析响应内容特征** 直接检查响应体特征(备选方案): ```c // 保存完整响应文件 web_reg_save_param_ex( "ParamName=FullResponse", "LB=", "RB=", "SEARCH_FILTERS", "Scope=Body", LAST ); web_url("target", ...); // 检查响应内容 if (strstr(lr_eval_string("{FullResponse}"), "\x1F\x8B\x08") != NULL) { lr_output_message("检测到GZIP魔术字"); // GZIP文件头标识 } ``` > **说明**:GZIP压缩数据起始字节为 `1F 8B 08`(十六进制)[^2] --- ### **处理压缩响应的关键步骤** 如果确认响应压缩: 1. **启用自动解压**(LoadRunner默认支持): ```c web_set_option("DecompressResponse", "1", LAST); // 确保启用 ``` 2. **验证解压结果**: ```c web_reg_save_param("DecompressedData", "LB=<body>", "RB=</body>", LAST); if (strlen(lr_eval_string("{DecompressedData}")) > 0) { lr_output_message("解压成功"); } ``` 3. 若仍失败,检查运行时设置: - `Runtime Settings > Preferences > Advanced > Use zlib decompression` 设为 `Yes` --- ### **压缩响应处理流程图** ```mermaid graph TD A[发起请求] --> B{检测Content-Encoding} B -->|含gzip| C[启用web_set_option解压] B -->|无压缩| D[直接关联响应] C --> E{关联成功?} E -->|是| F[正常处理] E -->|否| G[检查解压设置] G --> H[验证zlib配置] --> F ``` --- ### **常见问题排查** - **Q:开启解压后仍关联失败?** **A**:检查边界值是否在解压后文本中有效(解压后数据可能改变结构) - **Q:压缩响应在VuGen中正常但Controller失败?** **A**:确保所有负载生成器安装了zlib库(Windows系统需 `zlib1.dll`)[^3] > 通过以上方法,可100%确认响应压缩状态并正确处理。实际项目中,**方法1+方法2组合验证**是最可靠方案。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值