图像隐写之使用PHP隐藏图像中的文本

2019-04-01 23:23:27

隐写术是一门研究隐藏信息的科学艺术,通过隐写术,可以只让发送者和接收者知道隐藏的信息。

图像隐写术则是研究将数据隐藏在图像中,通过该技术可以防止无关用户发现这些隐藏的信息或数据。

相关的原理

图像都是由很小的点组成的,这些点就是像素。每个像素由三个字节来表示:一个代表红(Red),一个代表绿(Green),一个代表蓝(Blue)。这三种颜色的组合决定了像素的实际颜色。

红色:

二进制:11001001

十进制:201

绿色:


二进制:11111000

十进制:248(原文此处为201,实际应为248)

蓝色:

二进制:00000011

十进制:3

这一组合呈现的像素颜色是橙色。

图像隐写的基本原理就是修改LSB(Least Significant Bit,最不重要的那一bit)并不会被人眼察觉到。所以我们修改RGB的LSB值来存储隐藏信息,同时不影响图像的颜色。

在这个例子中,我们仅仅修改蓝色的LSB,但是如果我们想的话,也可以修改所有的红,绿和蓝的成分。不过只修改蓝色的LSB,我们可以隐藏的信息更少一些。我们也可以将要隐藏信息的长度存储到图像中,不过本教程中还没有做这一点。

假设我们现在要将1101藏到图片中。

首先我们要得到图片中每个像素的RGB值,因为我们要隐藏4bit的数据,而且我们要修改蓝色的部分,所以我们需要4个像素。

假设图片前4个像素的RGB值如下:

11001100 10010001 00101011

00011000 11110000 11111110

11100010 00100101 01010101

11111101 00001010 01000011

现在,我们将这4个RGB值最后的bit替换成1101,

现在新的RGB值变成了:

11001100 10010001 00101011

00011000 11110000 11111111

11100010 00100101 01010100

11111101 00001010 01000011

加粗的最后一bit,就是我们要隐藏到图片的信息。

现在将新的RGB值应用到像素,因为人眼并不会注意到这种改变,所以图片看起来还是一样的。

加密的部分到现在就完成了,接下来我们将解密隐藏在图片的信息。

为了达到这一点,我们需要取出每个像素的RGB值,然后合并LSB来得到我们隐藏的信息。

用PHP实现

我们需要一张图片和3个php脚本来实现基本的图像隐写。

encrypt.php

decrypt.php

functions.php

1.jpg


encrypt.php

include('functions.php');

$message_to_hide = 'hello';

$binary_message = toBin($message_to_hide);

$message_length = strlen($binary_message);

$src = '1.jpg';

$im = imagecreatefromjpeg($src);

for($x=0;$x<$message_length;$x++){

$y = $x;

$rgb = imagecolorat($im,$x,$y);

$r = ($rgb >>16) & 0xFF;

$g = ($rgb >>8) & 0xFF;

$b = $rgb & 0xFF;

$newR = $r;

$newG = $g;

$newB = toBin($b);

$newB[strlen($newB)-1] = $binary_message[$x];

$newB = toString($newB);

$new_color = imagecolorallocate($im,$newR,$newG,$newB);

imagesetpixel($im,$x,$y,$new_color);

}

echo $x;

imagepng($im,'simple.png');

imagedestroy($im);

?>

decrypt.php

include('functions.php');

$src = 'simple.png';

$im = imagecreatefrompng($src);

$real_message = '';

for($x=0;$x<40;$x++){

$y = $x;

$rgb = imagecolorat($im,$x,$y);

$r = ($rgb >>16) & 0xFF;

$g = ($rgb >>8) & 0xFF;

$b = $rgb & 0xFF;

$blue = toBin($b);

$real_message .= $blue[strlen($blue)-1];

}

$real_message = toString($real_message);

echo $real_message;

die;

?>


functions.php

function toBin($str){

$str = (string)$str;

$l = strlen($str);

$result = '';

while($l--){

$result = str_pad(decbin(ord($str[$l])),8,"0",STR_PAD_LEFT).$result;

}

return $result;

}

function toString($binary){

return pack('H*',base_convert($binary,2,16));

}

?>

更多的信息可以查看这个视频:

视频地址:yotube: https://youtu.be/qH0wLIwZa0U (自备梯子)


  • 2018-12-11 10:20:40

    Android下载图片到相册

    调用以上系统自带的方法会把bitmap对象保存到系统图库中,但是这种方法无法指定保存的路径和名称,上述方法的title、description参数只是插入数据库中的字段,真实的图片名称系统会自动分配。 或者

  • 2018-12-11 15:45:00

    Laravel中七个非常有用但很少人知道的Carbon方法

    在编写PHP应用时经常需要处理日期和时间,Carbon继承自 PHP DateTime 类的 API 扩展,它使得处理日期和时间更加简单,这篇文章主要给大家分享了Laravel中七个非常有用但很少人知道的Carbon方法,需要的朋友可以参考下。

  • 2018-12-13 11:41:23

    Android drawable微技巧,你所不知道的drawable的那些细节

    好像有挺久时间没更新博客了,最近我为了准备下一个系列的博客,也是花了很长的时间研读源码。很遗憾的是,下一个系列的博客我可能还要再过一段时间才能写出来,那么为了不至于让大家等太久,今天就给大家更新一篇单篇的文章,讲一讲Android drawable方面的微技巧。

  • 2018-12-13 17:14:41

    Android安全开发之浅谈密钥硬编码

    在阿里聚安全的漏洞扫描器中和人工APP安全审计中,经常发现有开发者将密钥硬编码在Java代码、文件中,这样做会引起很大风险。信息安全的基础在于密码学,而常用的密码学算法都是公开的,加密内容的保密依靠的是密钥的保密,密钥如果泄露,对于对称密码算法,根据用到的密钥算法和加密后的密文,很容易得到加密前的明文;对于非对称密码算法或者签名算法,根据密钥和要加密的明文,很容易获得计算出签名值,从而伪造签名。

  • 2018-12-13 17:17:02

    轻松实现动态获取Android手机CPU架构类型

    .so文件是unix的动态连接库,是二进制文件,作用相当于windows下的.dll文件。 他使用了C/C++代码编写的可以操作硬件比java更高级的 底层代码,执行速度和效率比其他语言要高。 在Android中调用动态库文件(*.so)都是通过jni的方式。

  • 2018-12-13 22:48:48

    Android MultiDex实践:如何绕过那些坑?

    MultiDex, 顾名思义,是指多dex实现,大多数App,解压其apk后,一般只有一个classes.dex文件,采用MultiDex的App解压后可以看到有classes.dex,classes2.dex,… classes(N).dex,这样每个dex都可以最大承载65k个方法,很大限度地缓解了单dex方法数限制。