实践环境

python 3.6.2

scikit-build-0.16.7

win10

opencv_python-4.5.4.60-cp36-cp36m-win_amd64.whl

下载地址:

https://pypi.org/project/opencv-python/4.5.4.60/#files

https://files.pythonhosted.org/packages/57/6c/7f4f56b2555d5c25dd4f41fc72a16dc6402cb2b4f967da11d8d26c669b55/opencv_python-4.5.4.60-cp36-cp36m-win_amd64.whl

注意:下载时不用下abi版的,比如 opencv_python-4.6.0.66-cp36-abi3-win_amd64.whl 不能用,

因为数据类型为 np.uint8,也就是0~255,

依赖包安装

pip3 install -i https://pypi.tuna.tsinghua.edu.cn/simple scikit-build # 解决     ModuleNotFoundError: No module named 'skbuild'问题
pip install opencv_python-4.5.4.60-cp36-cp36m-win_amd64.whl

代码实践

示例图片

代码

import os
import numpy as np
import cv2
from datetime import datetime
from PIL import Image def capture_image(image_file_path, left, upper, width, height, target_file_name=None):
'''截取图片''' right = left + width
lower = upper + height
if os.path.exists(image_file_path):
image = Image.open(image_file_path)
# width, height = image.size
# print('图片宽度', width, '图片高度', height) head, ext = os.path.splitext(image_file_path)
if not target_file_name:
target_file_name = 'pic_captured%s%s' % (datetime.now().strftime('%Y%m%d%H%M%S%f'), ext) target_file_path = '%s%s' % (head, target_file_name)
image.crop((left, upper, right, lower)).save(target_file_path)
return target_file_path
else:
error_msg = '图片文件路径不存在:%s' % image_file_path
print(error_msg)
raise Exception(error_msg) def append_picture(image1_path, image2_path):
'''拼接图片''' image1 = cv2.imread(image1_path, -1)
shape = image1.shape
height1, width1, channel1 = shape
# print(shape) # 输出:(315, 510, 4)
# print(image1) # 输出一3维数组
# print(len(image1), len(image1[0])) # 输出:315 510 image2 = cv2.imread(image2_path, -1)
height2, width2, channel2 = image2.shape total_height = max(height1, height2)
total_width = width1 + width2 dst = np.zeros((total_height, total_width, channel1), np.uint8)
dst[0:height1, 0:width1] = image1 dst[0:height2, width1:total_width] = image2
cv2.imwrite("merge.png", dst) if __name__ == '__main__':
# 截取图片
image_path1 = capture_image('example.png', 10, 30, 510, 315)
image_path2 = capture_image('example.png', 520, 30, 518, 315) append_picture(image_path1, image_path2)

运行结果

截取的图片

合并的图片

代码补充说明

  1. imread(filename, flags=None)

    • filename 图片路径

    函数返回一个3三元组:(height, width, channel) ,元素中元素从左到右分别表示图片的高度,宽度,通道数(彩色图片是三通道的,每个通道表示图片的一种颜色(RGB),对于OpenCV读取到的图片的通道顺序是BGR) ,假设图片3元组为 (315, 510, 4) ,表示有315行,即315个二维数组,510列,即每个二维数组有510个一维数组。

    • flags 标志位

      • cv2.IMREAD_COLOR:默认参数,表示读入一副彩色图片,忽略alpha通道,可用1作为实参替代

      • cv2.IMREAD_GRAYSCALE:读入灰度图片,可用0作为实参替代

      • cv2.IMREAD_UNCHANGED:读入完整图片,包括alpha通道,可用-1作为实参替代

        PS:alpha通道,又称A通道,是一个8位的灰度通道,该通道用256级灰度来记录图像中的透明度复信息,定义透明、不透明和半透明区域,其中黑表示全透明,白表示不透明,灰表示半透明

  2. imwrite(filename, img, params=None)

    将图片矩阵以文件的形式储存起来

    • filename 待保存的图片路径

    • img Mat或Mat的矢量)要保存的一个或多个图像。

    • params 特定格式的参数对(paramId_1、paramValue_1、paramId_2、paramValue_2……),参阅cv::ImwriteFlags

  3. zeros(shape, dtype=None, order='C')

    返回一个用零填充的给定形状和类型的新数组(ndarray)

    • shape 整数或者整数元组。新数组的形状,例如(2, 3) or 2
    • dtype 数据类型,可选。数组所需的数据类型,比如,numpy.int8。 默认 numpy.float64
    • order {'C', 'F'},可选,默认: 'C'。是否在内存中按行优先(row-major)顺序(C语言风格)或者列优先(column-major)(Fortran风格)顺序存储多维数据。

    示例

    >>> import numpy as np
    
    # 创建2维数组
    >>> array = np.zeros([2, 3])
    >>> print(array) # 输出一个二维数组 一个包含2个一维数组,每个一维数组包含3个元素
    [[0. 0. 0.]
    [0. 0. 0.]]
    >>> array = np.zeros([2, 3], np.int64) # 指定数组元素数据类型为int64
    >>> print(array)
    [[0 0 0]
    [0 0 0]]
    >>> array = np.zeros([2, 3], np.float64) # 指定数组元素数据类型为float64
    >>> print(array)
    [[0. 0. 0.]
    [0. 0. 0.]]
    >>> array = np.zeros([3, 2]) # 输出一个二维数组 一个包含3个一维数组,每个一维数组包含2个元素
    >>> print(array)
    [[0. 0.]
    [0. 0.]
    [0. 0.]] # 创建3维数组
    >>> array = np.zeros((2, 3, 4), np.int8)
    >>> print(array) # 输出一个3维数组 一个包含2个二维数组,每个二维数组包含3个一维数组,每个一维数组包含4个元素
    [[[0 0 0 0]
    [0 0 0 0]
    [0 0 0 0]] [[0 0 0 0]
    [0 0 0 0]
    [0 0 0 0]]]
  4. 冒号在Numpy数组索引中的作用说明

    3维数组为例

    ndarray[index1:index2, index3:index4, index5:index6]

    indexN:indexM 表示获取索引在范围[indexN, indexM)内的数组元素(注意,不包含索引为indexM的元素),这里的indexN代表起始元素索引,可选,默认为0,indexM代表结束元素索引,可选,默认为所在层级数组元素个数+1

    index1:index2 表示获取三维数组中,索引在范围[index1, index2)内的数组元素,即二维数组

    index3:index4 表示获取上述二维数组中,索引在范围[index3, index4)内的数组元素,即一维数组

    index5:index6 表示获取上述一维数组中,索引在范围[index5, index6)内的数组元素

    示例

    >>> array = np.array([[[1, 2, 3], [4, 5, 6], [7, 8, 9]], [[11, 12, 13], [14, 15, 16], [17, 18, 19]], [[20, 21, 22], [23, 24, 25], [26, 27, 28]]]) # 创建一个3维 ndarray
    >>> array
    array([[[ 1, 2, 3],
    [ 4, 5, 6],
    [ 7, 8, 9]], [[11, 12, 13],
    [14, 15, 16],
    [17, 18, 19]], [[20, 21, 22],
    [23, 24, 25],
    [26, 27, 28]]])
    >>> array[:] # 获取全部元素,等价于array[:, :, :]
    array([[[ 1, 2, 3],
    [ 4, 5, 6],
    [ 7, 8, 9]], [[11, 12, 13],
    [14, 15, 16],
    [17, 18, 19]], [[20, 21, 22],
    [23, 24, 25],
    [26, 27, 28]]])
    >>> array[:, :, :]
    array([[[ 1, 2, 3],
    [ 4, 5, 6],
    [ 7, 8, 9]], [[11, 12, 13],
    [14, 15, 16],
    [17, 18, 19]], [[20, 21, 22],
    [23, 24, 25],
    [26, 27, 28]]]) >>> array[1:2] # 获取索引在[1,2)范围内的二维数组
    array([[[11, 12, 13],
    [14, 15, 16],
    [17, 18, 19]]])
    >>> array[1:] # 获取索引在[1,3)范围内的二维数组
    array([[[11, 12, 13],
    [14, 15, 16],
    [17, 18, 19]], [[20, 21, 22],
    [23, 24, 25],
    [26, 27, 28]]])
    >>> array[:2] # 获取索引在[0,2)范围内的二维数组
    array([[[ 1, 2, 3],
    [ 4, 5, 6],
    [ 7, 8, 9]], [[11, 12, 13],
    [14, 15, 16],
    [17, 18, 19]]]) >>> array[1:2, 1:2] # 获取索引在[1,2)范围内的二维数组,二维数组中只获取索引在[1,2)范围内的一维数组
    array([[[14, 15, 16]]])
    >>> array[1:2, :2] # 获取索引在[1,2)范围内的二维数组,二维数组中只获取索引在[0,2)范围内的一维数组
    array([[[11, 12, 13],
    [14, 15, 16]]])
    >>> array[1:2, 1:] # 获取索引在[1,2)范围内的二维数组,二维数组中只获取索引在[1,3)范围内的一维数组
    array([[[14, 15, 16],
    [17, 18, 19]]])
    >>> array[1:2, :] # 获取索引在[1,2)范围内的二维数组的全部元素
    array([[[11, 12, 13],
    [14, 15, 16],
    [17, 18, 19]]])
    >>> array[1:2, 1:2, 1:2] # 获取索引在[1,2)范围内的二维数组,二维数组中只获取索引在[1,2)范围内的一维数组,一维数组中只获取索引在[1,2)范围内的元素
    array([[[15]]])
    >>> array[1:2, 1:2, 1:] # 获取索引在[1,2)范围内的二维数组,二维数组中只获取索引在[1,2)范围内的一维数组,一维数组中只获取索引在[1,3)范围内的元素
    array([[[15, 16]]])
    >>> array[1:2, 1:2, :2] # 获取索引在[1,2)范围内的二维数组,二维数组中只获取索引在[1,2)范围内的一维数组,一维数组中只获取索引在[0,2)范围内的元素
    array([[[14, 15]]])
    >>> array[1:2, 1:2, :] # 获取索引在[1,2)范围内的二维数组,二维数组中只获取索引在[1,2)范围内的一维数组,获取一维数组的所有元素
    array([[[14, 15, 16]]])

Python 结合opencv实现图片截取和拼接的更多相关文章

  1. 用 Python 和 OpenCV 检测图片上的条形码

      用 Python 和 OpenCV 检测图片上的的条形码 这篇博文的目的是应用计算机视觉和图像处理技术,展示一个条形码检测的基本实现.我所实现的算法本质上基于StackOverflow 上的这个问 ...

  2. 用 Python 和 OpenCV 检测图片上的条形码(转载)

    原文地址:http://python.jobbole.com/80448/ 假设我们要检测下图中的条形码: # load the image and convert it to grayscale 1 ...

  3. python 利用opencv去除图片水印

    python 去除水印"人工"智能去除水印 这两天公司来了一个新的需求--去除水印,对于我一个从未接触过的这种事情的人来说,当时我是蒙的.不过首先我就去搜索了一下是否有该种合适的功 ...

  4. Python cv2 OpenCV 中传统图片格式与 base64 转换

    Base64是网络上最常见的用于传输8Bit字节码的编码方式之一,是一种基于64个可打印字符来表示二进制数据的方法.通过http传输图片常常将图片数据转换成base64之后再进行传输. Base64简 ...

  5. Python 图像处理 OpenCV (16):图像直方图

    前文传送门: 「Python 图像处理 OpenCV (1):入门」 「Python 图像处理 OpenCV (2):像素处理与 Numpy 操作以及 Matplotlib 显示图像」 「Python ...

  6. Python各类图像库的图片读写方式总结

    最近在研究深度学习视觉相关的东西,经常需要写python代码搭建深度学习模型.比如写CNN模型相关代码时,我们需要借助python图像库来读取图像并进行一系列的图像处理工作.我最常用的图像库当然是op ...

  7. Python爬虫下载美女图片(不同网站不同方法)

    声明:以下代码,Python版本3.6完美运行 一.思路介绍 不同的图片网站设有不同的反爬虫机制,根据具体网站采取对应的方法 1. 浏览器浏览分析地址变化规律 2. Python测试类获取网页内容,从 ...

  8. 使用 opencv 将图片压缩到指定文件尺寸

    前言 图片压缩应用很广泛,如生成缩略图等.前期我在进行图片处理的过程中碰到了一个问题,就是如何将图片压缩到指定尺寸,此处尺寸指的是生成图片文件的大小. 我使用 opencv 进行图片处理,于是想着直接 ...

  9. 针对于Python的OpenCV环境搭建

    OpenCV 依赖 下载OpenCV 配置 总结 给Python搭建opencv的环境还真是略嫌麻烦,于是做下笔记,以备不时之需. OpenCV 依赖 opencv有些依赖,我们必须安装一下,否则接下 ...

  10. BugKu 2B+基于python的opencv的安装-------CTF 盲水印的套路

    BugKu杂项-2B 下载图片后,binwalk下跑一跑,发现有个zip,分离. 值得一提的是,这个zip是伪加密的. 但是你在分离的时候,伪加密的图片也给你分离出来了.这两个图片2B和B2肉眼看起来 ...

随机推荐

  1. Python实现五子棋人机对战的二次开发

    Python实现人机对战的二次开发     在网上找到了一个使用python实现五子棋游戏,其中通过加入一个简单的AI算法实现了人机对战的功能,我觉得这个人机对战还是蛮有意思的,下面我分析一下五子棋游 ...

  2. nginx+php,nginx+tomcat动静分离实战

    1. 动静分离实战 1.1.1 nginx+tomcat 动静分离 主机 用途 10.0.0.63 tomcat服务器 10.0.0.64 nginx服务器 1.1.2 安装 java+tomcat环 ...

  3. linux的账号和组

    1.0 账号与用户组 1.1 用户标识符:UID,GID 虽然我们登陆Linux主机的时候输入的是账号,但其实Linux主机并不会直接认识你的账号名称,账号只是为了方便人. 一个文件如何判断他的拥有者 ...

  4. SELinux 安全模型——TE

    首发公号:Rand_cs SELinux 安全模型--TE 通过前面的示例策略,大家对 SELinux 应该有那么点感觉认识了,从这篇开始的三篇文章讲述 SELinux 的三种安全模型,会涉及一些代码 ...

  5. LeetCode 40. Combination Sum II 组合总和 II (C++/Java)

    题目: Given a collection of candidate numbers (candidates) and a target number (target), find all uniq ...

  6. 关于前三次pta的总结

    前言 这三次pta难度在不断上升的同时,要求我们线上慕课+自主学习来了解更多的java中的各种方法,如:正则表达式 List Map等.与此同时要求我们展开尝试并熟练类的构造,类的引用,链表的基本运用 ...

  7. UDP端口探活的那些细节

    一 背景 商业客户反馈用categraf的net_response插件配置了udp探测, 遇到报错了,如图  udp是无连接的,无法用建立连接的形式判断端口. 插件最初的设计是需要配置udp的发送字符 ...

  8. MySql 增、删、改、查数据库

    前言 之前几天写了MySql 的GROUP BY 语句和 JOIN 语句,今天补一下创建数据库.表的语句.首先假设已经暗转好MySQL 数据库,然后创建数据库.表. 创建数据库 create data ...

  9. 你不知道的 CSS 之包含块

    你不知道的 CSS 之包含块 一说到 CSS 盒模型,这是很多小伙伴耳熟能详的知识,甚至有的小伙伴还能说出 border-box 和 content-box 这两种盒模型的区别. 但是一说到 CSS ...

  10. DotNet Web应用单文件部署系列

    目录 一.    pubxml文件配置 二.    打包wwwroot文件夹 三.    混淆dll文件 四.    csproj文件配置 五.    批处理 六.    Windows服务安装 七. ...