Python实践项目——LSB隐写术
此为北京理工大学某专业某学期某课程的某次作业
一、项目背景
1、隐写术
隐写术是一门关于信息隐藏的技巧与科学,所谓信息隐藏指的是不让除预期的接收者之外的任何人知晓信息的传递事件或者信息的内容。
2.LSB 隐写术
LSB 隐写术是一种图像隐写术技术,其中通过将每个像素的最低有效位替换为要隐藏的消息位来将消息隐藏在图像中。
3.实现原理
为了更好地理解,让我们将数字图像视为像素的二维阵列,每个像素包含取决于其类型和深度的值,使用最广泛的颜色模式RGB,这些值的范围为0–255之间。
可以使用 ASCII Table 将消息转换为十进制值,然后再转换为二进制。然后,我们逐个迭代像素值,将它们转换为二进制后,我们将每个最低有效位替换为序列中的消息位。
要解码编码图像,我们只需反转该过程,收集并存储每个像素的最后一位,然后将它们分成 8 组,并将其转换回 ASCII 字符以获取隐藏消息。
二、项目目标
1.主要目标
编写LSB图像隐写程序,包括:加密程序和解密程序。
2.目标分解
a)实现文本信息加密到图像
b)实现图像文件解密到文本
三、技术选型
1.问题:如何以二进制方式读写图像文件?
首先安装pillow库,win+R输入cmd快速打开控制台,直接输入以下代码即可自动安装
pip install pillow
然后读取图片
from PIL import Image #从pillow库(即PIL)中导入Image类
img = Image.open('../xx.jpg') #读取图片存入变量img中
print(img.format) #输出图片格式(str)
print(img.size) #输出图片大小信息 (宽度w,高度h)tuple = (int,int)
获取像素信息
#像素载入
pix = img.load()
width = img.size[0] #.size 方法返回的是一个元组 tuple =(int,int)
height = img.size[1]
#获取像素点的RGB值
rgb_list = [] #创建一个数组存储RGB值
for y in range(height):#遍历每一个像素点,将图像看作是一个二维数组,
for x in range(width): #如果x循环在外层输出的图像会发生一个九十度的翻转
r,g,b =pix[x,y] #此处的r,g,b是像素点pix[x,y]的RGB值
rgb_list.append(r)
rgb_list.append(g)
rgb_list.append(b)
输出图像
#输出图像
j = 0
pixels = [] #以[(r1,g1,b1),(r2,g2,b2)]形式存放每个像素点的RGB值,于绘制图像
img_out = Image.new(img.mode,img.size) #生成新图像,以原图的格式和大小
#img_out此时还是一张白纸,下面的代码旨在更新img_out的像素信息
while j<len(rgb_list): #循环次数高达786432次
pixels.append((rgb_list[j],rgb_list[j+1],rgb_list[j+2])) #以元组的形式
j += 3
img_out.putdata(pixels)#放置像素信息
img_out.save("img_out2.png")#将图像保存为,程序运行后会出现在根目录
2.问题:信息转换与提取
二进制转文本(解密)
def bina_to_txt(bina):
#只要传入一个二进制数组成的序列即可翻译成文本
tex = []
for i in bina:
tex.append(chr(int(i,2)))
return tex #返回一个单字符序列 #要求bina的格式为['01010101','11111111']
文本转二进制(加密)
def txt_to_bina(txt):
c=[]
for a in txt:
c.append("{:0>8}".format(bin(ord(a)).lstrip('0b')))
#格式化将二进制码保存起来
#注意要在右侧补齐八位,否则信息会错位
resultlist = []
for i in c:
for j in range(8):
resultlist.append(i[j])
return resultlist
#txt 为字符串类型,如 “hello world!” #print(txt_to_bina("h")) 输出测试
#test_output:['0', '1', '1', '0', '1', '0', '0', '0']
替换信息位(加密)
#替换信息位的信息
i = 0
while i < len(txt_to_bina(txt)):
temp =list(bin(rgb_list[i])) #用 bin()强制转换,bin()返回一个字符串类型
temp[-1]=txt_to_bina(txt)[i] #将二进制型的RGB信息的最后一位转换成文本二进制码
rgb_list[i] = int(''.join(temp),2)
i += 1
#txt_to_bina()是自定义的一个函数,旨在将文本转化成二进制码,返回一个单字符的序列
#这里是直接用第一个像素的RGB值作为隐写的开头,所以rgb_list和txt_to_bina()[]的index是一样的
#此处可以做一个加密 #特别注意在python中字符串不能直接修改,replace方法不会改变原来的string
#修改字符串要将字符串转换成一个序列,修改序列后在将序列转成字符串,实现代码如下
s = 'abcde'
temp = list(s)
temp[-1] = 'f' #假设要将s的最后一位“e”修改为“f”
s = ''.join(temp)
提取信息位信息
#这里直接用的是“hello world!”的长度,后期优化可以加个旗帜识别
c = ''
for i in range(96):
c += bin(rgb_list[i])[-1] #图像处理后得到rgb_list,取二进制码的最后一位
out_list_bin =[]
for i in range(12):
out_list_bin.append(c[i*8:(i+1)*8])#每八位为一组转换出文本
print(''.join(bina_to_txt(out_list_bin)))
Python实践项目——LSB隐写术的更多相关文章
- python实践项目1
python #南昌理工学院人工智能学院实验室 WORKSHOP 实践项目 import time print('welcome to our WORKSHOP') print('.......... ...
- Python实践项目2
#南昌理工学院人工智能学院实验室WORKSHOP实践项目 import time import random SCRIPT_NPC_SCHOOL_SISTER = ['你好!', '你好!', '你是 ...
- python实践项目九:操作文件-修改文件名
描述:多个文件,文件名名包含美国风格的日期( MM-DD-YYYY),需要将它们改名为欧洲风格的日期( DD-MM-YYYY) 代码1:先创建100个文件名为美国风格日期的文件(文件路径为项目当前路径 ...
- python实践项目一:Collatz函数
要求1:编写一个名为 collatz()的函数,它有一个名为 number 的参数.如果参数是偶数,那么 collatz()就打印出 number // 2, 并返回该值.如果 number 是奇数, ...
- python 实践项目
项目一:让用户输入圆的半径,告诉用户圆的面积 思路: 1.首先需要让用户输入一个字符串,即圆的半径 2.判断用户输入的字符串是否为数字 isalpha 3.求圆的面积需要调用到math模块,所以要导 ...
- python实践项目—Collatz序列
Collatz序列题意说明 编写一个名为collatz()的函数,它有一个名为number 的参数.如果参数是偶数,那么collatz()就打印出number // 2,并返回该值.如果number ...
- python实践项目十:zipfile模块-将一个文件夹备份到一个 ZIP 文件
描述:将指定路径下的某文件夹备份到一个zip文件 代码: #!/usr/bin/python # -*- coding: UTF-8 -*- # backupToZip - Copies an ent ...
- python实践项目八:生成随机试卷文件
描述:匹配美国50个州的首府. 下面是程序需要完成的任务: • 创建35 份不同的测验试卷. • 为每份试卷创建50 个多重选择题,次序随机. • 为每个问题提供一个正确答案和3 个随机的错误答案,次 ...
- python实践项目七:正则表达式版本的strip()函数
描述:写一个函数,它接受一个字符串,做的事情和 strip()字符串方法一样.如果只传入了要去除的字符串, 没有其他参数, 那么就从该字符串首尾去除空白字符:否则, 函数第二个参数指定的字符将从该字符 ...
随机推荐
- SQL语句的整理
mysql语句的整理 1.SQL DML 和 DDL 可以把 SQL 分为两个部分:数据操作语言 (DML) 和 数据定义语言 (DDL). SQL (结构化查询语言)是用于执行查询的语法.但是 SQ ...
- 【转载】vscode配置C/C++环境
VScode中配置 C/C++ 环境 Tip:请在电脑端查看 @零流@火星动力猿 2022.4.12 1. 下载编辑器VScode 官网:https://code.visualstudio.com/( ...
- 初识Java GUI
1. 使用Java Swing 显示的窗口如下 在原有代码基础上添加代码实现对窗口大小 标题等信息
- dense_rank()和rank() 窗口函数 mysql
dense_rank()的语法 DENSE_RANK() OVER ( PARTITION BY <expression>[{,<expression>...}] ORDER ...
- Matplotlib(基本用法)
Matplotlib 是数据分析绘图的常见模块,可以算是 2D-绘图(Python)领域使用最广泛的套件,可以将数据图形化,并且提供多样化的输出格式,利于数据的显示并分析. 接下来展示的是Matplo ...
- Odoo14 一些用的熟手的函数
# Odoo14 一些用的熟手的函数 # from odoo.tools import config # 这是直接访问配置文件,也就是当你执行./odoo-bin -c odoo.cfg的时候 # c ...
- 【原创】Python 网易易盾滑块验证
本文仅供学习交流使用,如侵立删! 记一次 网易易盾滑块验证分析并通过 操作环境 win10 . mac Python3.9 selenium.PIL.numpy.scipy.matplotlib 分析 ...
- Luogu2543[AHOI2004]奇怪的字符串 (动态规划 LCS)
04年的省选这么water吗,开个滚动数组算了 #include <iostream> #include <cstdio> #include <cstring> # ...
- java中为什么只存在值传递(以传入自定义引用类型为例)
java中只有值传递 为什么这么说?两个例子: public class Student { int sage = 20; String sname = "云胡不归"; publi ...
- Excel 插入嵌入式图表和独立图表的方法
描述 嵌入式图表:是一种与当前工作表相同位置的图表,且悬浮在表格之上,不受表格限制,因此称之为嵌入式图表. 独立图表:是独立于当前工作表的图表,打印时,需要单独将其打印出来. 插入独立图表的图文教程: ...