利用 python 实现多张图片的无损拼接

以下两段是作者的心路历程,图个乐的可以看一看,想学干货可以直接略过。

就在昨天晚上,我要为文章配图,要求是将6张图拼接为3张图。我就想,可以呀,PS应该很容易搞定,虽然我PS功底差,但架不住我有位朋友是 king of PS 呀(朋友间的调侃啦)。所以我将每张图都剪切成一样的高度,希望能对PS的过程有所帮助。
king of PS
好朋友不愧是 king of PS ,每张图都拼接的很棒。但是,三张图一一对比,细看还是有一些不太对齐,身患强迫症的我当然心中有些疙瘩。不过好朋友已经做的很棒了,我不能以自己的强迫症去强迫别人,所以我就走上了“自主研发”的道路。

往期博客:

用python画心形函数,属于数学家的浪漫~

拒绝“过劳死”!看看你已经敲了多久的代码!

用python计算每天什么时候下班

导入 python 库

import matplotlib.pyplot as plt
import skimage.io as io
import numpy as np

查看需要拼接的图片

因为工作需要,所以就不使用昨天晚上的图片了。
我就拿了两张截图作为示例演示。

首先看看拼接前的图片是什么样子:

jzg = io.imread('jzg.jpg')   # np.ndarray, [h, w, c], 值域[0, 255], RGB
plt.imshow(jzg)   #查看图片
plt.show()

解释说明:“jzg”保存的是numpy的数组。

jzg

lgz = io.imread('lgz.jpg')   # np.ndarray, [h, w, c], 值域[0, 255], RGB
plt.imshow(lgz)
plt.show()

lgz
因为我使用的是 jupyter Notebook,所以图片显示的不是太清晰。

查看一下图片的大小和数组元素的数据类型。

print(jzg.shape)   #查看图片的大小
print(jzg.dtype)   #查看数组元素数据类型
print(lgz.shape)
print(lgz.dtype)

输出:

(720, 1280, 3)
uint8
(720, 1280, 3)
uint8

(720, 1280, 3)表示的是数组的大小,物理意义为[h, w, c],分别是图片的高度h,图片的宽度w,图片的通道数c。

可以看出两者的大小完全一致,数组元素的数据类型为“uint8”。

查看数组中元素的值域:

print([jzg.min(), jzg.max()])

输出:

[0, 255]

横向拼接

创建拼接用的数组:

pj1 = np.zeros((720,1280 + 1280,3))   #横着拼接
pj1[:,:1280,:] = jzg.copy()   #图片jzg在左
pj1[:,1280:,:] = lgz.copy()   #图片lgz在右
print(pj1.dtype)   #查看数组元素类型

输出:

float64

可以看出拼接后的数据类型不一样了,所以要改一下,不然显示的就是错误的。

pj1=np.array(pj1,dtype=np.uint8)   #将pj1数组元素数据类型的改为"uint8"
plt.imshow(pj1)   #查看拼接情况
plt.show()

拼接后的图片

保存拼接后的图片

将拼接后的图片保存在当前目录下,也可以改为其它的路径。

io.imsave('pj1.jpg', pj1)   #保存拼接后的图片

总结

横向拼接的代码总结如下:

import matplotlib.pyplot as plt
import skimage.io as io
import numpy as np

jzg = io.imread('jzg.jpg')   # np.ndarray, [h, w, c], 值域(0, 255), RGB
plt.imshow(jzg)   #查看图片
plt.show()

lgz = io.imread('lgz.jpg')   # np.ndarray, [h, w, c], 值域(0, 255), RGB
plt.imshow(lgz)
plt.show()

print(jzg.shape)   #查看图片的大小
print(jzg.dtype)   #查看数组元素数据类型
print(lgz.shape)
print(lgz.dtype)

pj1 = np.zeros((720,1280 + 1280,3))   #横向拼接
pj1[:,:1280,:] = jzg.copy()   #图片jzg在左
pj1[:,1280:,:] = lgz.copy()   #图片lgz在右
print(pj1.dtype)   #查看数组元素类型

pj1=np.array(pj1,dtype=np.uint8)   #将pj1数组元素数据类型的改为"uint8"

plt.imshow(pj1)   #查看拼接情况
plt.show()

io.imsave('pj1.jpg', pj1)   #保存拼接后的图片

纵向拼接

当然,可以横向拼接自然也可以纵向拼接,只需将拼接用的数组改为如下:

pj2 = np.zeros((720 + 720,1280,3))   #横向拼接

将拼接操作改为:

pj1[:720,:,:] = jzg.copy()   #图片jzg在上
pj1[720:,:,:] = lgz.copy()   #图片lgz在下

然后其他步骤都一样。

图片间距

有些时候要求要有缝拼接,这时候就将拼接用的数组横向或纵向变大,空白区域使用“0”或者“255”填充(我不知道“0”和“255”中,哪个代表白色,哪个代表黑色,需要的就自己去实验或者查询一下)。

  • 6
    点赞
  • 60
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
首先需要了解一下 LZ78 编解码的基本原理: LZ78 编码是一种无损压缩算法,它通过建立字符串字典来实现压缩。在压缩时,LZ78 将输入文本划分为一个个字符串,然后将每个字符串表示为已知字典中的一个字符串和一个新的字符。这个新的字符可以是输入文本中的下一个字符,也可以是新的字符串的第一个字符。在解压时,LZ78 只需将字典中的字符串逐个拼接起来就可以还原出原始文本。 下面是一个使用 Python 实现 LZ78 编解码的示例代码: ```python def lz78_encode(text): dictionary = {} code = [] i = 0 while i < len(text): j = i while text[i:j+1] in dictionary and j < len(text): j += 1 if j == len(text): code.append((dictionary[text[i:j]], '')) else: dictionary[text[i:j+1]] = len(dictionary) + 1 code.append((dictionary[text[i:j]], text[j])) i = j + 1 return code def lz78_decode(code): dictionary = {1: ''} text = '' for item in code: if item[1] == '': text += dictionary[item[0]] else: dictionary[len(dictionary) + 1] = dictionary[item[0]] + item[1] text += dictionary[len(dictionary)] return text ``` 在这个示例代码中,`lz78_encode` 函数接受一个字符串作为输入,返回一个由元组组成的列表,每个元组表示一个编码后的字符串。元组的第一个元素表示字典中已有的字符串的编号,第二个元素表示新增的一个字符。`lz78_decode` 函数接受一个由元组组成的列表作为输入,返回解码后的原始字符串。 例如,对于输入字符串 `"abababcabababa"`,`lz78_encode` 函数会返回以下编码后的字符串: ```python [(0, 'a'), (0, 'b'), (2, 'c'), (2, 'a'), (3, 'b'), (3, 'a'), (0, '')] ``` 这个编码后的字符串表示的意思是: - 第一个字符串是字符 'a',没有在字典中出现过; - 第二个字符串是字符 'b',没有在字典中出现过; - 第三个字符串是字典中的第二个字符串 'ab' 后面加上一个新的字符 'c'; - 第四个字符串是字典中的第二个字符串 'ab' 后面加上一个新的字符 'a'; - 第五个字符串是字典中的第三个字符串 'abc' 后面加上一个新的字符 'b'; - 第六个字符串是字典中的第三个字符串 'abc' 后面加上一个新的字符 'a'; - 第七个字符串是空字符串,表示输入文本结束。 `lz78_decode` 函数会将上述编码后的字符串还原成原始字符串 `"abababcabababa"`。
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值