《算法之美》的一处错误

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

相关文章

Java NoClassDefFoundError运行时错误分析解决

《JavaNoClassDefFoundError运行时错误分析解决》在Java开发中,NoClassDefFoundError是一种常见的运行时错误,它通常表明Java虚拟机在尝试加载一个类时未能... 目录前言一、问题分析二、报错原因三、解决思路检查类路径配置检查依赖库检查类文件调试类加载器问题四、常见

Windows Docker端口占用错误及解决方案总结

《WindowsDocker端口占用错误及解决方案总结》在Windows环境下使用Docker容器时,端口占用错误是开发和运维中常见且棘手的问题,本文将深入剖析该问题的成因,介绍如何通过查看端口分配... 目录引言Windows docker 端口占用错误及解决方案汇总端口冲突形成原因解析诊断当前端口情况解

openCV中KNN算法的实现

《openCV中KNN算法的实现》KNN算法是一种简单且常用的分类算法,本文主要介绍了openCV中KNN算法的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的... 目录KNN算法流程使用OpenCV实现KNNOpenCV 是一个开源的跨平台计算机视觉库,它提供了各

springboot+dubbo实现时间轮算法

《springboot+dubbo实现时间轮算法》时间轮是一种高效利用线程资源进行批量化调度的算法,本文主要介绍了springboot+dubbo实现时间轮算法,文中通过示例代码介绍的非常详细,对大家... 目录前言一、参数说明二、具体实现1、HashedwheelTimer2、createWheel3、n

C/C++错误信息处理的常见方法及函数

《C/C++错误信息处理的常见方法及函数》C/C++是两种广泛使用的编程语言,特别是在系统编程、嵌入式开发以及高性能计算领域,:本文主要介绍C/C++错误信息处理的常见方法及函数,文中通过代码介绍... 目录前言1. errno 和 perror()示例:2. strerror()示例:3. perror(

Go标准库常见错误分析和解决办法

《Go标准库常见错误分析和解决办法》Go语言的标准库为开发者提供了丰富且高效的工具,涵盖了从网络编程到文件操作等各个方面,然而,标准库虽好,使用不当却可能适得其反,正所谓工欲善其事,必先利其器,本文将... 目录1. 使用了错误的time.Duration2. time.After导致的内存泄漏3. jsO

SpringBoot实现MD5加盐算法的示例代码

《SpringBoot实现MD5加盐算法的示例代码》加盐算法是一种用于增强密码安全性的技术,本文主要介绍了SpringBoot实现MD5加盐算法的示例代码,文中通过示例代码介绍的非常详细,对大家的学习... 目录一、什么是加盐算法二、如何实现加盐算法2.1 加盐算法代码实现2.2 注册页面中进行密码加盐2.

Java时间轮调度算法的代码实现

《Java时间轮调度算法的代码实现》时间轮是一种高效的定时调度算法,主要用于管理延时任务或周期性任务,它通过一个环形数组(时间轮)和指针来实现,将大量定时任务分摊到固定的时间槽中,极大地降低了时间复杂... 目录1、简述2、时间轮的原理3. 时间轮的实现步骤3.1 定义时间槽3.2 定义时间轮3.3 使用时

Python中ModuleNotFoundError: No module named ‘timm’的错误解决

《Python中ModuleNotFoundError:Nomodulenamed‘timm’的错误解决》本文主要介绍了Python中ModuleNotFoundError:Nomodulen... 目录一、引言二、错误原因分析三、解决办法1.安装timm模块2. 检查python环境3. 解决安装路径问题

如何解决mysql出现Incorrect string value for column ‘表项‘ at row 1错误问题

《如何解决mysql出现Incorrectstringvalueforcolumn‘表项‘atrow1错误问题》:本文主要介绍如何解决mysql出现Incorrectstringv... 目录mysql出现Incorrect string value for column ‘表项‘ at row 1错误报错