《算法之美》的一处错误

2024-03-07 16:08
文章标签 算法 错误 之美 一处

本文主要是介绍《算法之美》的一处错误,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

今天开始拿到《算法之美》一本书,翻看起来,前阵子在看Charles Petzold大神的《编码-隐匿在计算机软硬件背后的语言》,百看不厌,真的是一本非常有趣的书,觉得左飞哥翻译的不错,了解到他还写了《算法之美》一书,特地买来看看。看到Z字形编码,对其产生了兴趣,于是开始敲代码。

  • Z字形编码是干什么的? 

首先了解一下 JPEG ,它的英文名是 Joint Photographic Experts Group 是一种常见的图像文件格式,也是目前静态图像中压缩比最高的一种图像文件格式,它综合运用了多种压缩技术而达到一种极高的压缩比例。

关于 JPEG 更多详情 请见 维基百科---https://zh.wikipedia.org/wiki/JPEG

在 JPEF 编码过程中,有一个非常重要的步骤,即Z字形编排过程。

 

借用一下本书的一副图,简单说明一下Z字形编码。

下面看一个例子,相信大家就明白了。

上面是 8 × 8 的一个矩阵,将其经过 Z 字形编码得到的结果如果上图。在 《算法之美》这本书中,给的矩阵大小为 8 ,矩阵大小定义为 SIZE ,在本小节末尾,有这么一句话 --- 此外,这个算法不仅对 8 × 8 的矩阵有效,对于 SIZE 取其他值的情况仍然有效,读者不妨试试看。

于是,我就试了将 SIZE 改为其它值,但是 SIZE为偶数时是没有错误的,但是 SIZE 一旦为奇数时就发生了错误。

错误情况如图所示:

 

也就是说没有完成 Z 字形编码,所以我就想把 SIZE 为奇数的情况,也能够进行正确的 Z 字形编码。

经过一番在草稿纸上的演算,只需将书中的两行代码做下小小的改动再加上两个判断即可。

代码如下:

  • SIZE 为奇数的情况:
		if(SIZE%2 == 1){if(i == 0 && j%2 == 0 && j !=SIZE-1 || i == SIZE -1 && j%2 == 1){j++;continue;}if(j == 0 && i%2 == 1 || j == SIZE - 1 && i%2 == 0) {i++;continue;}}
  •  SIZE 为偶数的情况:
		else if(SIZE%2 == 0){if((i == 0 || i == SIZE-1) && j%2 == 0){j++;continue;}if((j == 0 || j == SIZE-1) && i%2 == 1){i++;continue;}	}

完整的判断代码为:

for(x = 0;x< SIZE;x++){for(y = 0;y< SIZE;y++)	{//赋值*(*(a + i) + j) = *(*(matrix + x) +y);if(SIZE%2 == 1){if(i == 0 && j%2 == 0 && j !=SIZE-1 || i == SIZE -1 && j%2 == 1){j++;continue;}if(j == 0 && i%2 == 1 || j == SIZE - 1 && i%2 == 0) {i++;continue;}}else if(SIZE%2 == 0){if((i == 0 || i == SIZE-1) && j%2 == 0){j++;continue;}if((j == 0 || j == SIZE-1) && i%2 == 1){i++;continue;}	}if((i+j)%2 == 0){i--;j++;}else if((i+j)%2 == 1){i++;j--;}}}

 

SIZE 为7时 :

下面来分别解释一下 SIZE 为偶数 和 奇数的情况:(也就是代码怎么来的)

 

 

上图其实也有一处错误,只不过这个错误没有什么太大的影响所以就权当没看到了。

 

 以7 × 7 的矩阵为例:

  • 当 SIZE 为偶数时:
  • 当 SIZE 为奇数时:

 可以发现:

  1. 当 matrix[i][j] 列数 j 是偶数时,并且只有当 i = 0 时,那么遍历路径在矩阵中的走向就是水平向右移动一格。
  2. 当 matrix[i][j] 列数 j 是奇数时,并且只有当 i == SIZE -1 时,那么遍历路径在矩阵中的走向就是水平向右移动一格。
  3. 当 matrix[i][j] 行数 i 是奇数时,并且只有当 j = 0 时,那么遍历路径在矩阵中的走向就是垂直向下移动一格。
  4. 当 matrix[i][j] 行数 i 是偶数时,并且只有当 j == SIZE -1 时,那么遍历路径在矩阵中的走向就是垂直向下移动一格。

以上就是 完整的 Z字形编码 过程。

 附完整代码:

#include <iostream>
#include <iomanip>
using namespace std;
#define SIZE 3
int main(int arc,char** argv)
{int matrix[SIZE][SIZE] = {0};int a[SIZE][SIZE] = {0};int i,j,x,y,value = 0;int *p;p = &matrix[0][0];//初始化矩阵for(i = 0;i<SIZE * SIZE;i++)*p++ = i; //打印原始矩阵cout << "原始矩阵如下:" << endl; for(i = 0; i< SIZE ; i++){for(j = 0;j<SIZE;j++)//setw(int n) 用来控制输出间隔 cout << setw(4) << *(*(matrix + i) + j);cout << endl; }i = 0; j = 0;//进行Z字形编排for(x = 0;x< SIZE;x++){for(y = 0;y< SIZE;y++)	{//赋值*(*(a + i) + j) = *(*(matrix + x) +y);if(SIZE%2 == 1){if(i == 0 && j%2 == 0 && j !=SIZE-1 || i == SIZE -1 && j%2 == 1){j++;continue;}if(j == 0 && i%2 == 1 || j == SIZE - 1 && i%2 == 0) {i++;continue;}}else if(SIZE%2 == 0){if((i == 0 || i == SIZE-1) && j%2 == 0){j++;continue;}if((j == 0 || j == SIZE-1) && i%2 == 1){i++;continue;}	}if((i+j)%2 == 0){i--;j++;}else if((i+j)%2 == 1){i++;j--;}}}cout << endl << "经过Z字形编码后的矩阵如下:" << endl; for(i = 0;i< SIZE;i++){for(j = 0;j<SIZE;j++)cout << setw(4)<<*(*(a+i)+j);cout<<endl;}return 0;
} 

 

这篇关于《算法之美》的一处错误的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/784075

相关文章

SpringBoot+Docker+Graylog 如何让错误自动报警

《SpringBoot+Docker+Graylog如何让错误自动报警》SpringBoot默认使用SLF4J与Logback,支持多日志级别和配置方式,可输出到控制台、文件及远程服务器,集成ELK... 目录01 Spring Boot 默认日志框架解析02 Spring Boot 日志级别详解03 Sp

Java中的雪花算法Snowflake解析与实践技巧

《Java中的雪花算法Snowflake解析与实践技巧》本文解析了雪花算法的原理、Java实现及生产实践,涵盖ID结构、位运算技巧、时钟回拨处理、WorkerId分配等关键点,并探讨了百度UidGen... 目录一、雪花算法核心原理1.1 算法起源1.2 ID结构详解1.3 核心特性二、Java实现解析2.

SpringBoot排查和解决JSON解析错误(400 Bad Request)的方法

《SpringBoot排查和解决JSON解析错误(400BadRequest)的方法》在开发SpringBootRESTfulAPI时,客户端与服务端的数据交互通常使用JSON格式,然而,JSON... 目录问题背景1. 问题描述2. 错误分析解决方案1. 手动重新输入jsON2. 使用工具清理JSON3.

如何解决Druid线程池Cause:java.sql.SQLRecoverableException:IO错误:Socket read timed out的问题

《如何解决Druid线程池Cause:java.sql.SQLRecoverableException:IO错误:Socketreadtimedout的问题》:本文主要介绍解决Druid线程... 目录异常信息触发场景找到版本发布更新的说明从版本更新信息可以看到该默认逻辑已经去除总结异常信息触发场景复

Python struct.unpack() 用法及常见错误详解

《Pythonstruct.unpack()用法及常见错误详解》struct.unpack()是Python中用于将二进制数据(字节序列)解析为Python数据类型的函数,通常与struct.pa... 目录一、函数语法二、格式字符串详解三、使用示例示例 1:解析整数和浮点数示例 2:解析字符串示例 3:解

CentOS 7 YUM源配置错误的解决方法

《CentOS7YUM源配置错误的解决方法》在使用虚拟机安装CentOS7系统时,我们可能会遇到YUM源配置错误的问题,导致无法正常下载软件包,为了解决这个问题,我们可以替换YUM源... 目录一、备份原有的 YUM 源配置文件二、选择并配置新的 YUM 源三、清理旧的缓存并重建新的缓存四、验证 YUM 源

python3 pip终端出现错误解决的方法详解

《python3pip终端出现错误解决的方法详解》这篇文章主要为大家详细介绍了python3pip如果在终端出现错误该如何解决,文中的示例方法讲解详细,感兴趣的小伙伴可以跟随小编一起了解一下... 目录前言一、查看是否已安装pip二、查看是否添加至环境变量1.查看环境变量是http://www.cppcns

python进行while遍历的常见错误解析

《python进行while遍历的常见错误解析》在Python中选择合适的遍历方式需要综合考虑可读性、性能和具体需求,本文就来和大家讲解一下python中while遍历常见错误以及所有遍历方法的优缺点... 目录一、超出数组范围问题分析错误复现解决方法关键区别二、continue使用问题分析正确写法关键点三

Ubuntu上手动安装Go环境并解决“可执行文件格式错误”问题

《Ubuntu上手动安装Go环境并解决“可执行文件格式错误”问题》:本文主要介绍Ubuntu上手动安装Go环境并解决“可执行文件格式错误”问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未... 目录一、前言二、系统架构检测三、卸载旧版 Go四、下载并安装正确版本五、配置环境变量六、验证安装七、常见

正则表达式r前缀使用指南及如何避免常见错误

《正则表达式r前缀使用指南及如何避免常见错误》正则表达式是处理字符串的强大工具,但它常常伴随着转义字符的复杂性,本文将简洁地讲解r的作用、基本原理,以及如何在实际代码中避免常见错误,感兴趣的朋友一... 目录1. 字符串的双重翻译困境2. 为什么需要 r?3. 常见错误和正确用法4. Unicode 转换的