RLE算法在PS2中的应用

RLE算法介绍

RLE(Run Length Encoding,行程长度编码)算法,是把文件内容用"重复次数x数据"的形式来表示的压缩方法。比如:有AAAAAABBCDDEEEEEF这样一段数据,在字符后面加上重复出现次数,就可以用6A2B1C2D5E1F来表示。可以看到原始数据是17字节,编码后是12字节,因此压缩是成功的。

让我们再看一串数据:ABCDE,如果按照上面的算法,编码后为1A1B1C1D1E,原始数据是5字节,编码后是10字节,毫无疑问这种压缩方式是失败的。

为什么第二种字符串压缩会失败呢?细心的朋友一定看出来是因为它的字符重复出现的次数很少,因此使用"重复次数x数据"反而增加了数据长度。那有没有办法解决这个"缺陷"呢?答案是有的。我们接下来介绍在PS2游戏机中,是如何使用RLE算法来压缩图片的。

RLE算法在PS2中的应用

PS2中,图片文件的前4个字节指示了压缩后文件的大小。接下来的数据按照rle_code + 数据块的格式重复排列。需要注意的是,在PS2这里,rle_code数据块中的每个数据,都是2字节,这点是与其它传统的RLE算法普遍为1字节最大的不同。

rle_code的最高位是标识位,如果这一位是1,则表示后面紧跟着的数据块是"非重复数据",类似于上面的ABCDE。此时将0x8000减去rle_code的后7位,得到的是数据块的长度。此时只需取出后面紧跟的该长度的数据块即可。

如果标识位为0,则表示后面紧跟着的数据块是"重复数据",类似于AAAAA,此时rle_code就是重复次数,只要取出后面紧跟着的一个数据块,重复rle_code次即可。

伪代码如下:

python 复制代码
while rle_offset < compressed_size:
    rle_code = rle_code_struct.unpack_from(self.byte_val, rle_offset)[0]
    rle_offset += 2
    if rle_code & 0x8000:
        next_bytes = 0x8000 - (rle_code ^ 0x8000)
        texture_buf += self.byte_val[rle_offset: rle_offset + next_bytes * 2]
        rle_offset += next_bytes * 2
    else:
        times = rle_code
        if times > 0:
            next_byte = self.byte_val[rle_offset: rle_offset + 2]
            for _ in range(times):
                texture_buf += next_byte
            rle_offset += 2

总结

如果在一个文件中,能连续遇到大量重复的数据,RLE算法可以提供很好的压缩效果。但对于出现连续的"非重复数据",需要使用改良过的算法进行优化。PS2使用的是众多改良算法的一种,比较简单,也很方便初学者对该算法的学习。

本篇文章使用的代码摘自我的一个github项目:ps2mc-browser,欢迎有兴趣的小伙伴一起研究那个古老的PS2游戏机。

相关推荐
papership31 分钟前
【入门级-算法-6、排序算法: 插入排序】
数据结构·算法·排序算法
HAH-HAH42 分钟前
【蓝桥杯 2024 国 Java A】粉刷匠小蓝
c++·学习·数学·算法·职场和发展·蓝桥杯·组合数学
我是是是是是西红柿1 小时前
游戏中的展销系统使用的数据结构
数据结构·游戏
hweiyu002 小时前
C++设计模式,高级开发,算法原理实战,系统设计与实战(视频教程)
c++·算法·设计模式
大千AI助手2 小时前
粒子群优化(PSO)算法详解:从鸟群行为到强大优化工具
人工智能·算法·优化算法·pso·粒子群优化
我叫汪枫3 小时前
C语言深度入门系列:第十一篇 - 动态内存管理与数据结构:程序世界的高效算法大师
c语言·数据结构·算法
Li_7695323 小时前
优选算法100 题 ——1 双指针
算法
77qqqiqi3 小时前
算法——数学基础
算法
啊?啊?3 小时前
7 排序算法通关指南:从 O (n²)(选择 / 冒泡)到 O (nlogn)(快排 / 归并)+ 计数排序
数据结构·算法·排序算法
张较瘦_3 小时前
[论文阅读] 算法 | 抗量子+紧凑!SM3-OTS:基于国产哈希算法的一次签名新方案
论文阅读·算法·哈希算法