海明码一篇文章彻底搞懂

海明码学习前提

记住几个要点:

  1. 不要用异或套用公式!!!!题目随便变几个变死你!
  2. 看完这篇博客不要看别的博客!!!!别的人瞎写的坑死你

学习海明码之前,我们要约定3个原则:

  1. 海明码只能检测出2位错,纠1位错(因此不要问如果3位错怎么办等幼稚问题)。
  2. 海明码默认进行偶校验(除非特殊说明使用奇校验)。
  3. 海明码是一串由0和1组成的序列(除01外没有其他的值,记住了!这是重点)

如果下面有任何无法理解的问题,反复看上面三个原则,下面再也不赘述。

前提:奇偶校验

奇校验:这串序列1的个数如果为偶数则在前面加个1,使1的个数变成奇数,否则加0。
偶校验:这串序列1的个数如果为奇数则在前面加个1,使1的个数变成偶数,否则加0。

例子:1111 奇校验就是 11111 偶校验就是 01111
1110 奇校验就是 01110 偶校验就是 11110

特性是检测一位错,无法纠错。

概述:海明码的构成

例如如下序列:
1100
我们想要让其变成海明码只需如下操作

1.算出校验位数k

正常情况下我们需要如下此操作:

2^k >= k + 数据位数 + 1

这里等于3

2.确定校验位在海明码中的位置

这里按2^k次幂留出来,就像1,2,4,8,16,32。(如果问有5位等其他烦人的数据位怎么办后面我会说,先按4位数做)

H7 H6 H5 H4 H3 H2 H1
1 1 0 0

3.分组(重点,很多人蒙圈就在此)

我们需要确认H1,H2,H4这三个校验位都来校验哪些位置。
我们按这个规则进行分配。

将1,2,4(海明码下标为1,2,4)

的二进制码写出来,并且最高位补到3位(前面算的K数)
如下所示:

1 2 4
001 010 100

然后我们将0替换为*,作为通配表。

1 2 4
**1 *1* 1**

我们将1到7的二进制序列,列出来如下表

7 6 5 4 3 2 1
111 110 101 100 011 010 001

!!!重点!!!!

我们将7->1依次与上面的通配表进行匹配

1 2 4
**1 *1* 1**
001(1) 010(2) 100(4)
011(3) 011(3) 101(5)
101(5) 110(6) 110(6)
111(7) 111(7) 111(7)

因此我们可以确定
H1 负责 1 3 5 7 位数的校验
H2 负责 2 3 6 7 位数的校验
H4 负责 4 5 6 7 位数的校验

4.求出校验位是0还是1

因为上面我们得出以下结论:

H1 负责 1 3 5 7 位数的校验
H2 负责 2 3 6 7 位数的校验
H4 负责 4 5 6 7 位数的校验

那 根据

H7 H6 H5 H4 H3 H2 H1
1 1 0 0

这张表,我们根据偶校验很容易就求出以下结论:
H3,H5,H7 1的个数为奇数 因此H1=1
H3,H6,H7 1的个数为偶数 因此H2=0
H5,H6,H7 1的个数为偶数 因此H4=0
至此我们得出了完整的汉明码

H7 H6 H5 H4 H3 H2 H1
1 1 0 0 0 0 1

5.查错

查错比较简单,如果以下三组

H1,H3,H5,H7
或者
H2,H3,H6,H7
或者
H4,H5,H6,H7
偶校验出错,则出错。

比方说 如果 H1,H3,H5,H7由1100 变成了 1110 (1的个数为偶数)就是出错了

这里该不赘述

6.纠错

首先我们先理解以下为什么海明码能纠错。
首先我们先画个圆。然后按如下形式做交叉

在每个相邻部位,我们做相加处理

变成了如下形式

当我们如果发现偶校验出错,
比方说在 1 3 7 5 这个区域出错。

如果这个位置出错了,那么一定是 1 3 7 5 这四个位置中的一个位置出错(如果俩位出错则无法纠错,这个点一定要记住)
如果此时其他的俩个组 即:2,3,6,7 和 4,5,6,7偶校验都通过了的话。
也就证明只可能是1出错
所以我们可以将 1 的位数 做修改。如果是0变为1,如果是1变为。来达到纠错的目的。

但是如果2,3,5,7这个位置也出错了,4,5,6,7这个位置没有出错。
我们很容易就推导出,是 3 这个位置出错了。

我们就可以修改3的值,如果是0变为1或者如果是1变为0.

在此时我们会发现一个巧妙的规则!
当我们把1,3,5,7 设为P1,
2,3,6,7设为P2
4,5,6,7设为P3时

当如果哪组校验失败就为1

P3 P2 P1 出错(第几)位数
0 0 1 1
0 1 0 2
0 1 1 3
1 0 0 4
1 0 1 5
1 1 0 6
1 1 1 7

刚好是对应的二进制编码。就是这么绝。

其实学完海明码之后,我真的觉得人家实在是太聪明了。

5位数数据

至此,其实如果认真看上面的部分,大家已经可以理解海明码是如何实现的了。
但是我还是再带大家写一次。这种5位数的。关键在于如何分组!!!!!

比方说
10001

先求出校验位数:

2 ^ k > = k + 5 + 1
则 k = 4

画出表格

将1,2,4,8位置空出来,再将数据位填进去

H9 H8 H7 H6 H5 H4 H3 H2 H1
1 0 0 0 1

分组(*为通配符)

8 4 2 1
1*** *1** **1* ***1
8,9 4,5,6,7 2,3,6,7 1,3,5,7,9

偶校验每个分组得出结果

H9 H8 H7 H6 H5 H4 H3 H2 H1
1 1 0 0 0 0 1 1 0

总结

如果还有人不懂,请仔仔细细跟着我走一遍。本人保证此思路是绝对正确的。

给我讲懂的老师是哈工大刘宏伟(MOOC课上有)
所教的是计算机组成原理
谢谢您老师

原文地址:https://www.cnblogs.com/godoforange/p/12003676.html

时间: 2024-07-30 20:22:52

海明码一篇文章彻底搞懂的相关文章

一篇文章彻底搞懂Java虚拟机

概念 虚拟机:指以软件的方式模拟具有完整硬件系统功能.运行在一个完全隔离环境中的完整计算机系统 ,是物理机的软件实现.常用的虚拟机有VMWare,Visual Box,Java Virtual Machine(Java虚拟机,简称JVM). Java虚拟机阵营:Sun HotSpot VM.BEA JRockit VM.IBM J9 VM.Azul VM.Apache Harmony.Google Dalvik VM.Microsoft JVM- 启动流程 基本架构 Java运行时编译源码(.j

一篇文章彻底搞懂snowflake算法及百度美团的最佳实践

写在前面的话 一提到分布式ID自动生成方案,大家肯定都非常熟悉,并且立即能说出自家拿手的几种方案,确实,ID作为系统数据的重要标识,重要性不言而喻,而各种方案也是历经多代优化,请允许我用这个视角对分布式ID自动生成方案进行分类: 实现方式 完全依赖数据源方式 ID的生成规则,读取控制完全由数据源控制,常见的如数据库的自增长ID,序列号等,或Redis的INCR/INCRBY原子操作产生顺序号等. 半依赖数据源方式 ID的生成规则,有部分生成因子需要由数据源(或配置信息)控制,如snowflake

一篇文章彻底搞懂es6 Promise

前言 Promise,用于解决回调地狱带来的问题,将异步操作以同步的操作编程表达出来,避免了层层嵌套的回调函数. 既然是用来解决回调地狱的问题,那首先来看下什么是回调地狱 var sayhello = function(callback){ setTimeout(function(){ console.log("hello"); return callback(null); },1000); } sayhello(function(err){ console.log("xia

【Networkk】一篇文章完全搞清楚 scoket read/write 返回码、阻塞与非阻塞、异常处理 等让你头疼已久的问题

浅谈TCP/IP网络编程中socket的行为 我认为,想要熟练掌握Linux下的TCP/IP网络编程,至少有三个层面的知识需要熟悉: 1. TCP/IP协议(如连接的建立和终止.重传和确认.滑动窗口和拥塞控制等等) 2. Socket I/O系统调用(重点如read/write),这是TCP/IP协议在应用层表现出来的行为. 3. 编写Performant, Scalable的服务器程序.包括多线程.IO Multiplexing.非阻塞.异步等各种技术. 关于TCP/IP协议,建议参考Rich

一篇文章彻底弄懂Base64编码原理

在互联网中的每一刻,你可能都在享受着Base64带来的便捷,但对于Base64的基础原理又了解多少?今天这篇博文带领大家了解一下Base64的底层实现. Base64的由来 目前Base64已经成为网络上常见的传输8Bit字节代码的编码方式之一.在做支付系统时,系统之间的报文交互都需要使用Base64对明文进行转码,然后再进行签名或加密,之后再进行(或再次Base64)传输.那么,Base64到底起到什么作用呢? 在参数传输的过程中经常遇到的一种情况:使用全英文的没问题,但一旦涉及到中文就会出现

知识扩展——(转)一篇文章彻底弄懂Base64编码原理

在互联网中的每一刻,你可能都在享受着Base64带来的便捷,但对于Base64的基础原理又了解多少?今天这篇博文带领大家了解一下Base64的底层实现. 一.Base64的由来 目前Base64已经成为网络上常见的传输8Bit字节代码的编码方式之一.在做支付系统时,系统之间的报文交互都需要使用Base64对明文进行转码,然后再进行签名或加密,之后再进行(或再次Base64)传输.那么,Base64到底起到什么作用呢? 在参数传输的过程中经常遇到的一种情况:使用全英文的没问题,但一旦涉及到中文就会

一篇文章搞懂python2、3编码

说在前边: 编码问题一直困扰着每一个程序员的编程之路,如果不将它彻底搞清楚,那么你的的这条路一定会走的格外艰辛,尤其是针对使用python的程序员来说,这一问题更加显著, 因为python有两个版本,这两个版本编码格式却完全不同,但我们却经常需要兼顾这两个版本,所以出现各种问题的几率就大了很多. 所以在这里我试图用一篇文章来彻底梳理整个python语言的编码问题,尽量降低以后在这方面举到问题的可能性. ps 此文一定程度上参考和引用了alex的博客:“https://www.cnblogs.co

一篇文章带你搞懂spring全家桶套餐

spring全家桶里都有哪些食物 上期我们讲了spring和springmvc两个框架的基础知识和学习路线,而这期内容,我们将围绕着spring全家桶展开来讨论. 大家应该都知道,按照出现的顺序,spring全家桶大概包含了spring.springmvc.springboot以及springcloud,从开胃小菜spring到满汉全席springcloud,spring全家桶可谓Java工程师的必备大餐,那么,我们不妨先来看看,spring全家桶是如何从光杆司令spring发展到如今的庞大家族

一篇文章看懂Android学习最佳路线

为什么中高级Android程序员不多呢?这是一个问题,我不好回答,但是我想写一篇文章来描述下Android的学习路线,期望可以帮助更多的Android程序员提升自己. 作者:来源:Android开发中文站|2015-11-12 10:40 收藏 分享 前言 看到一篇文章中提到"最近几年国内的初级Android程序员已经很多了,但是中高级的Android技术人才仍然稀缺",这的确不假,从我在百度所进行的一些面试来看,找一个适合的高级Android工程师的确不容易,一般需要进行大量的面试才