`

图片压缩

    博客分类:
  • JAVA
 
阅读更多
原文地址:http://hi.baidu.com/su602/blog/item/ecaa354f4282be31aec3ab2c.html

import java.io.DataInputStream;
import javax.microedition.lcdui.Image;

public class Tools {

private static final int FLAG_16BIT_4_LEN = 0;

private static final int FLAG_REBUILD_SIZE = 0;

private static final int FLAG_REBUILD_MODULE = 0;


/***********************************************************************************************************
* 1 压缩原理 要清楚 USI 的压缩原理,首先需要对图像的存储方式有一个基本的了解。USI 压缩是建立在索引色的基础上进行的。
* 
* 1.1 索引图与RGB图
* 对于PNG图像,可以分为索引(Index)图和RGB图两种,索引图只包含固定数量的颜色,而RGB图的颜色数量是不受限制的。
* RGB图的每一个象素都保存一个RGB值,代表这个象素的颜色,因此,一张RGB图有多少个象素,文件中就保存多少个RGB值。
* 而索引图会将其固定数量的颜色,按照顺序排列起来,作为颜色的索引保存在文件头中,被称为调色板(palette)。每一个
* 象素只保存其颜色在调色板中的索引。如一个32色的索引图,在文件头中保存了32个颜色,索引值从0到31。图中每一个象
* 素只记录其颜色的索引。因此,对于一般的PNG图,索引图文件的大小总是小于RGB图的。
* 
* 1.2 行程压缩原理
* 当我们把一张索引图的所有象素(N个),按照从上到下,从左至右,即按行扫描的顺序排列起来的时候,我们得到一个队列。
* 如果我们用1个字节来存储一个象素的索引值(调色板颜色不超过256),那么数据的大小为N字节。这段数据的格式我们表示为
* [I1][I2]…[In] 共 N 个。在上面的队列中,可能会出现很多连续相同的索引值,最多的就是透明色。如果我们在每个索引值
* 前用1个字节保存这个值连续出现的数量(最多可以表示256个),那数据的格式变为[C1][I1][C2][I2]…[Cm][Im] 共 M个。
* 那么一张256个象素的单色图的所有数据,只需要2个字节来保存。通常,我们所需的图中总是有大片连续的颜色,包括透明色,
* 因此按照这个格式保存的图像,其文件大小可以大大降低,这就是行程的压缩原理。
* 
* 1.3 USI压缩原理 如果一张索引图的颜色数为32,那么在[C1][I1][C2][I2]…[Cm][Im]
* 格式中,I的数值都小于32,那么每个字节前3 bits 始终为0。为了充分利用这 3bits,我们可以将 C 的值保存在这 3bits中,
* 这样我们的格式变为 [G1][G2]….[Gk] 共 K个(G的高位为数量,低位为颜色索引)。这样,对于32色的图,
* 每个字节最多可以保存8个象素的信息,对于64色的图,每个字节最多可以保存4个象素的信息,对于16色的图,每个字节最多
* 可以保存16个象素的信息。 在[G1][G2]….[Gk] 这K个字节前,再加上调色板数据和其它本图的必要信息,就得到了USI格式的文件。
*****************************************************************************************************************/

int m_flags ,m_count ,m_mask ,m_modelCount ,m_dataSize ;
int m_rebuildWidth,m_rebuildHeight;
int[][] m_pal ;
int []m_dataOffset;
byte[] m_models ,m_data ;
private void load(String file) {
   try {
    DataInputStream din = new DataInputStream(getClass()
      .getResourceAsStream(file));

    m_flags = din.readInt(); // 格式标志

    /** 读取调色板信息 */
    m_count = din.readByte() & 0xff; // 调色板位数
    m_mask = 0xff >> (8 - m_count); // 计算 取色板索引的掩码
    int pal_count = din.readByte() & 0xff; // 调色板数量
    int pal_len = din.readByte() & 0xff; // 调色板长度 即颜色数
    m_pal = new int[pal_count][pal_len]; // 初始化调色板容器
    int pal;
    // 读取调色板信息
    for (int i = 0; i < pal_count; i++) {
     for (int j = 0; j < pal_len; j++) {
      pal = din.readShort() & 0xffff;
      m_pal[i][j] = (((((pal & 0xF000) >>> 12) * (17 << 24)) & 0xFF000000)
        | ((((pal & 0x0F00) >>> 8) * (17 << 16)) & 0x00FF0000)
        | ((((pal & 0x00F0) >>> 4) * (17 << 8)) & 0x0000FF00) | ((((pal & 0x000F) * 17))));
     }
    }

    /** 读取图块信息 */
    m_modelCount = din.readShort() & 0xffff; // 图块数量
    // 读取图块尺寸
    if ((m_flags & FLAG_REBUILD_SIZE) != 0) {
     // 基于尺寸的转换方式
     m_rebuildWidth = din.readByte() & 0xff;
     m_rebuildHeight = din.readByte() & 0xff;
    } else if ((m_flags & FLAG_REBUILD_MODULE) != 0) {
     // 基于动画model的转换方式
     m_models = new byte[m_modelCount * 2];
     din.read(m_models);
    }

    /** 读取像素数据 */
    m_dataSize = din.readInt(); // 像素数据大小(压缩数据)
    m_data = new byte[m_dataSize];
    din.read(m_data); // 读取像素数据(压缩数据)

    // 读取每个图块数据的起始偏移量
    int offset = 0;
    m_dataOffset = new int[m_modelCount];
    for (int i = 0; i < m_modelCount; i++) {
     m_dataOffset[i] = offset;
     if ((m_flags & FLAG_16BIT_4_LEN) != 0) {
      offset += din.readShort();
     } else {
      offset += din.readByte() & 0xff;
     }
    }
   } catch (Exception ex) {
   }
}

/***************************************************************************
* 解压缩指定图块像素数据
* 
* @param model_id
*            int 图块号
* @param pal_id
*            int 调色板号
* @return int[] 解压缩图块像素数据(ARPG值)
**************************************************************************/
private int[] BuildRle8bFrm(int model_id, int pal_id) {

   // 计算解压后,像素数据的大小(图块W*图块H)
   int size;
   if ((m_flags & FLAG_REBUILD_SIZE) != 0) {
    size = m_rebuildWidth * m_rebuildHeight;
   } else {
    size = (m_models[model_id * 2] & 0xff)
      * (m_models[model_id * 2 + 1] & 0xff);
   }

   // 初始化像素buf
   int[] m_bufB = new int[size];

   int pal[] = m_pal[pal_id]; // 获取当前调色板
   int offset = m_dataOffset[model_id]; // 获取压缩数据起点

   // 解压缩
   int count, index, pos = 0;
   while (pos < size) {
    count = ((m_data[offset] & 0xFF) >> m_count) + 1;
    index = pal[m_data[offset] & m_mask];
    offset++;
    while (--count >= 0) {
     m_bufB[pos++] = index;
    }
   }
   return m_bufB;
}

/***************************************************************************
* 获取指定图块Image
* 
* @param model_id
*            int 图块号
* @param pal_id
*            int 调色板号
* @return Image 图块Image对象
**************************************************************************/
public Image GetImage(int model_id, int pal_id) {

   // 获得指定图块解压数据(ARPG颜色数据)
   int[] m_bufB = BuildRle8bFrm(model_id, pal_id);

   // 计算图块尺寸
   int w, h;
   if ((m_flags & FLAG_REBUILD_SIZE) != 0) {
    w = m_rebuildWidth;
    h = m_rebuildHeight;
   } else {
    w = m_models[model_id * 2] & 0xff;
    h = m_models[model_id * 2 + 1] & 0xff;
   }
   // 生成Image图片
   Image m_image = Image.createRGBImage(m_bufB, w, h, true);
   m_bufB = null;
   return m_image;
}

}


分享到:
评论

相关推荐

    C# core 图片压缩 图片无损压缩 图片无损剪切 无损图片压缩 无损图片剪切

    在C# Core中进行图片处理是一项常见的任务,其中包括图片压缩和剪切操作。无损压缩和剪切技术在保持原始图像质量的同时,可以减小文件大小或改变图像的形状。以下将详细介绍C# Core中如何实现这些功能。 首先,我们...

    图片压缩小工具

    《ImageResizer.exe:一款轻巧实用的图片压缩神器》 在数字时代,我们经常会遇到大量图片需要处理的情况,无论是为了存储空间的节省,还是为了网络传输的便捷,图片压缩都成为了必不可少的操作。今天,我们要介绍的...

    图片压缩到最小.rar

    在IT领域,图片压缩是一项重要的技术,特别是在网络传输、存储和显示方面。"图片压缩到最小.rar"这个压缩包文件的标题和描述直接指向了这一主题。本文将深入探讨图片压缩的基本原理、常见方法以及易语言...

    C#图片压缩工具源代码

    Grearo图片压缩工具 功能介绍【必读】: 1 图片批量压缩(30以内,依个人机器以及被压缩图片大小而定); 2 图片限定宽度,等比例压缩; 3 图片限定高度,等比例压缩; 4 图片限定高度,宽度压缩;...

    易语言图片压缩到最小

    在IT行业中,图片压缩是一项非常重要的技术,尤其是在网络传输、存储和显示图像时。"易语言图片压缩到最小"这个主题涉及到的是如何使用易语言这一编程工具来实现图片的高效压缩,以减小其占用的存储空间。易语言是...

    java图片压缩处理(可以压缩为任意大小

    在Java编程语言中,处理图片压缩是一项常见的任务,特别是在存储、传输或展示大量图像资源时。本主题将深入探讨如何使用Java实现图片压缩,并能够将其调整到任意大小,同时保持图片质量并避免变形。 首先,我们需要...

    图片压缩节约内存

    在IT行业中,图片压缩是一个非常重要的技术,尤其是在内存管理和移动应用开发中。标题"图片压缩节约内存"直接指向了这个核心目标:通过压缩图片来减少应用程序对内存的占用,从而提高性能并避免内存溢出的问题。这在...

    ASP.NET(C#) 图片压缩类

    ### ASP.NET(C#) 图片压缩类:深入解析与应用 在现代Web开发中,图片是网站内容的重要组成部分,但过大的图片文件会严重影响网页加载速度,降低用户体验。因此,图片压缩技术成为优化网站性能的关键手段之一。本文...

    小巧的图片压缩工具,压缩比例可调。

    在IT领域,图片压缩是一个非常实用的技术,尤其在网页设计、社交媒体分享、存储空间有限的设备上。本文将深入探讨一款被描述为“小巧的图片压缩工具,压缩比例可调”的软件,它允许用户批量处理图片,降低文件大小而...

    C# 图片压缩 多文件压缩

    在IT行业中,图片压缩和文件压缩是常见的操作,特别是在存储和传输大量数据时。本文将深入探讨使用C#编程语言实现图片压缩以及多文件压缩成ZIP格式的压缩包的技术细节和自定义设置。 图片压缩主要目标是减小图像...

    VB6图片压缩处理源码

    总的来说,VB6图片压缩处理源码可能涵盖了图像读取、压缩算法应用、图像尺寸调整、文件I/O以及用户交互等多个方面,体现了VB6结合外部库进行图像处理的能力。通过理解这些知识点,可以对源码进行深入研究和扩展,以...

    事业单位图片压缩工具

    为了解决这一问题,出现了诸多针对事业单位需求的图片压缩工具,其中,“事业单位图片压缩工具”脱颖而出,成为了一款深受用户欢迎的软件。 “事业单位图片压缩工具”是一款设计精良的图片压缩软件,具备多种先进...

    移动端H5图片压缩上传

    在移动端H5应用中,图片压缩上传是一项常见的需求,它涉及到前端图像处理、文件上传以及与服务器的交互。本文将详细讲解如何实现这一功能,主要关注JavaScript开发中的图片展示处理。 首先,我们要理解图片压缩的...

    微信小程序实现图片压缩

    本文实例为大家分享了微信小程序图片压缩的具体代码,供大家参考,具体内容如下 设计思路: 选择图片后调用微信压缩图片接口,压缩后接收压缩图片的临时地址,调用微信储存接口保存图片至本地。 参数: imagesrc:...

    图片压缩flex demo

    在IT行业中,图片压缩是一个非常重要的领域,尤其是在网络传输、存储和显示方面。"图片压缩flex demo"这个项目显然关注的是使用Flex技术进行图片压缩的示例。Flex是一种基于Adobe Flash Player或Adobe AIR运行时的...

    蜂鸟图片压缩软件 JPG PNG 压缩

    【蜂鸟图片压缩软件 JPG PNG 压缩】 在数字媒体和互联网领域,图片的大小对网站加载速度和存储空间有着直接影响。为了优化用户体验和节省资源,图片压缩变得至关重要。"蜂鸟图片压缩软件"(Hummingbird)是专为此...

    pb 图片压缩、解压技术结合XML

    在本项目中,"pb 图片压缩、解压技术结合XML" 提到了如何使用PB进行图片处理,具体涉及图片的压缩和解压,并结合XML文件进行数据存储。以下是关于这个主题的详细知识点: 1. **图片压缩**:图片压缩通常是为了减小...

    html5 canvas 图片压缩

    HTML5 Canvas是Web开发中的一个强大工具,它允许开发者在网页上进行动态...通过理解并运用上述技术,你可以创建一个高效且兼容性强的图片压缩解决方案,确保在微信、Chrome、Firefox等不同环境中都能得到良好的效果。

    批量图片压缩处理工具

    在IT行业中,图片压缩是一个非常重要的领域,尤其是在网络传输、存储和显示图像时,为了减少带宽需求和存储空间,通常需要对图片进行压缩。批量图片压缩处理工具就是专门针对这种情况设计的,它能帮助用户一次性处理...

    图片压缩工具PNG图片压缩工具压缩率高不失真

    PNG图片压缩工具是一种高效优化图像文件大小的软件,尤其适用于需要高质量、透明度支持的图像。PNG(Portable Network Graphics)格式因其无损压缩和广泛的颜色范围而被广泛使用,但通常其文件大小比其他格式如JPEG...

Global site tag (gtag.js) - Google Analytics