Python 图像数组变换及手绘效果实现
时间:2022-07-25
本文章向大家介绍Python 图像数组变换及手绘效果实现,主要内容包括其使用实例、应用技巧、基本知识点总结和需要注意事项,具有一定的参考价值,需要的朋友可以参考一下。
文章目录
一、图像的RGB色彩模式
图像一般使用RGB色彩模式,即每个像素点的颜色由红R绿G蓝B组成
RGB三个颜色通道的变化和叠加得到各种颜色,其中红绿蓝的取值范围都是0-255
RGB形成的颜色包括了人类视力所能感知的所有颜色
二、Python的PIL库
PIL,Python Image Library
PIL库是一个具有强大图像处理能力的第三方库
在Anaconda 中是已经安装好的,命令行下安装方法: pip install pillow
# 用到的第三方库
from PIL import Image # Image是PIL库中代表一个图像的类(对象)
import numpy as np
三、图像的数组表示
图像是一个由像素组成的二维矩阵,每一个元素是一个RGB值。
from PIL import Image
import numpy as np
im = np.array(Image.open(r'D:test 01.jpg')) # 打开一个图片 生成数组对象
print(im.shape, im.dtype)
# 输出结果
# (854, 960, 3) uint8
图像是一个三维数组,维度分别表示高度、宽度、和像素RGB值。
四、图像的变换
图像可以表示为数组,而数组是可以运算的,经过运算后的数组可以改变图像形状,对图像进行变换。读入图像后,获得像素RGB的值,修改后保存为新的文件。
原始图片如下:
from PIL import Image
import numpy as np
a = np.array(Image.open(r'D:test 01.jpg')) # 打开一个图片 生成数组对象
print(a.shape, a.dtype)
b = [255, 255, 255] - a # 计算RGB三个通道的补值
im = Image.fromarray(b.astype('uint8')) # 重新生成图片对象
im.save(r'D:test 02.jpg') # 保存为新的jpg图片
变换后得到的图片如下:
from PIL import Image
import numpy as np
# 调用convert('L') 将一个彩色图片变成灰度值的图片
a = np.array(Image.open(r'D:test 01.jpg').convert('L')) # 打开一个图片 生成数组对象 得到灰度值
print(a.shape, a.dtype)
c = 255 - a # 对灰度值取反
im = Image.fromarray(c.astype('uint8'))
im.save(r'D:test 03.jpg')
变换后得到的图片如下:
from PIL import Image
import numpy as np
a = np.array(Image.open(r'D:test 01.jpg').convert('L')) # 打开一个图片 生成数组对象 得到灰度值
print(a.shape, a.dtype)
d = (100 / 255) * a + 150 # 区间变换
im = Image.fromarray(d.astype('uint8'))
im.save(r'D:test 04.jpg')
变换后得到的图片如下:
from PIL import Image
import numpy as np
a = np.array(Image.open(r'D:test 01.jpg').convert('L')) # 打开一个图片 生成数组对象 得到灰度值
print(a.shape, a.dtype)
e = 255 * (a / 255) ** 2 # 像素平方
im = Image.fromarray(e.astype('uint8'))
im.save(r'D:test 05.jpg')
得到变换后的图片如下:
五、图像的手绘效果实现
手绘效果的特征:黑白灰色、边界线条较重、相同或相近色彩趋于白色、略有光源效果。手绘风格是在对图像进行灰度化的基础上由立体效果和明暗效果叠加而成的,灰度实际代表了图像的明暗变化,而梯度表示的灰度的变化率。所以可以通过调整像素的梯度值来间接改变图像的明暗程度,立体效果则通过添加虚拟深度值来实现。
from PIL import Image
import numpy as np
a = np.asarray(Image.open(r'D:test 01.jpg').convert('L')).astype('float')
# 根据灰度变化来模拟人类视觉的明暗程度
depth = 10. # 预设虚拟深度值为10 范围为0-100
grad = np.gradient(a) # 提取梯度值
grad_x, grad_y = grad # 提取x y方向梯度值 解构赋给grad_x,grad_y
# 利用像素之间的梯度值和虚拟深度值对图像进行重构
grad_x = grad_x * depth / 100.
grad_y = grad_y * depth / 100. # 根据深度调整 x y 方向梯度值
A = np.sqrt(grad_x ** 2 + grad_y ** 2 + 1.)
uni_x = grad_x / A
uni_y = grad_y / A
uni_z = 1./ A
vec_el = np.pi / 2.2 # 光源的俯视角度 弧度值
vec_az = np.pi / 4. # 光源的方位角度 弧度值
dx = np.cos(vec_el) * np.cos(vec_az) # 光源对x轴影响
dy = np.cos(vec_el) * np.sin(vec_az) # 光源对y轴影响
dz = np.sin(vec_el) # 光源对z轴影响
b = 255 * (dx * uni_x + dy * uni_y + dz * uni_z) # 光源归一化
b = b.clip(0, 255) # 为了避免数据越界,将生成辉度值裁剪至0-255区间
im = Image.fromarray(b.astype('uint8')) # 图像重构
im.save(r'D:test 06.jpg') # 保存图片
手绘效果图像:
本文为看了网课视频后的学习总结,也参考了网上其他的文章,测试图片来源于网络,仅用于学习交流之用。
作者:叶庭云 微信公众号:修炼Python CSDN:https://yetingyun.blog.csdn.net/ 本文仅用于交流学习,未经作者允许,禁止转载,更勿做其他用途,违者必究。 觉得文章对你有帮助、让你有所收获的话,期待你的点赞呀,不足之处,也可以在评论区多多指正。
- 网页超过一屏时自动浮动在网页最上方的图层特效
- 关于gcc、glibc和binutils模块之间的关系
- 贝叶斯过滤算法
- 強大的jQuery Chart组件-Highcharts
- vue2.0 配置 选项 属性 方法 事件 ——速查
- 亲密接触IIS 8和Web Deploy 3.0
- vue2.0 组件通信
- 在MongoDB中实现聚合函数
- 原生JS编写的照片墙效果实例演示特效
- 使用SuperWebSocket 构建实时 Web 应用
- Vue2.0的变化 ,组件模板,生命周期,循环,自定义键盘指令,过滤器
- 原生js编写的安全色拾色器
- 原生js写的贪吃蛇网页版游戏特效
- JS双月份显示联动效果,点击日期浮出消息提示
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- 了解红黑树的起源,理解红黑树的本质
- 在java中进行日期时间比较的4种方法
- 【crossbeam系列】4 crossbeam-channel:加强版channel
- 不想得手指关节炎?帮你提炼IDEA常用代码补全操作
- SpringBoot实现本地存储文件上传及提供HTTP访问服务
- 从 0 到 1 搭建技术中台之 iOS 可视化埋点实践
- 超给力,一键生成数据库文档-数据库表结构逆向工程
- 精讲RestTemplate第6篇-文件上传下载与大文件流式下载
- 【Rust日报】2020-09-12 Apple 使用 Rust 进行底层编程
- 精讲RestTemplate第4篇-POST请求方法使用详解
- 精讲RestTemplate第3篇-GET请求使用方法详解
- 通过JSch运行远程linux主机上的shell脚本
- 精讲RestTemplate第5篇-DELETE、PUT等请求方法使用详解
- 一个小小的签到功能,到底用MySQL还是Redis?
- 在IntelliJ IDEA中多线程并发代码的调试方法