python 实现图片批量加入水印!pillow 入门实战!
写文章的时候可以设置是否添加水印。可是,有些图片可能想加水印,有些不想加水印,该怎么办呢?
配置环境
python3 + pillow
pip3 install pillow
引入库
from PIL import Image, ImageSequence
import os
import random
效果预览:


使用方法:
- 在脚本同目录下添加水印图片 logo.png
- 创建目录 input 并在放入要添加水印的图片
- 创建目录 output 执行脚本 addlogo.py
- 结果输出在 output 文件夹下

实现原理
水印图片采集:
先读取水印图片的像素点信息和大小信息。去除透明度为0的像素,并修改透明度像素信息。
img_logo = Image.open("logo.png")
img_logo_width, img_logo_height = img_logo.size
img_logo_pixels = dict()
for w in range(img_logo_width):
for h in range(img_logo_height):
c = img_logo.getpixel((w,h))
if c!=(0, 0, 0, 0):
img_logo_pixels[(w, h)] = c[:3]+(125,)
混合颜色:
对每一个像素点采取颜色混合,其中c1是源图片的像素点信息(r,g,b,a),c2是logo图片像素点的信息。混合算法如下:
def blendPixel(c1,c2):
a1=256-c2[3]
a2=c2[3]-(a1*c2[3])/256.0
a=a1+a2
c=(int((a1*c1[0] + a2*c2[0])/a), int((a1*c1[1] + a2*c2[1])/a), int((a1*c1[2] + a2*c2[2])/a),int(a))
return c
处理源 Image 对象:
随机一个位置开始处理像素,具体代码参考如下。
def dealOneImage(image,offX=None,offY=None):
w, h = image.size
offX = offX if offX else random.random();
offY = offY if offY else random.random();
#如果图片尺寸小于水印图片,不加水印
if w>=img_logo_width and h>=img_logo_height:
top = int((w - img_logo_width)*offX)
left = int((h - img_logo_height)*offY)
for p, c in img_logo_pixels.items():
p_x = (p[0]+top)
p_y = (p[1]+left)
image_c = image.getpixel((p_x,p_y))
if(isinstance(image_c, tuple) and len(image_c)>2):
image.putpixel((p_x, p_y), blendPixel(image_c,c))
return image;
处理单个文件:
对于 gif 文件先拆成一张一张图片,在图片上添加水印,最后再合成 gif 。对于其他格式的图片文件可以多添加几个水印。最后输出保存到 output 文件夹下。
def dealOneFile(filePath):
img_orign = Image.open(filePath)
_,file_type = os.path.splitext(filePath)
basename = os.path.basename(filePath)
if file_type == '.gif':
sequence = [];
offX=random.random()
offY=random.random()
for f in ImageSequence.Iterator(img_orign):
if len(sequence) % 2 == 0:
offX=random.random()
offY=random.random()
sequence.append(dealOneImage(f.convert(),offX,offY))
sequence[0].save(f'./output/{basename}', save_all=True, append_images=sequence[1:])
else:
image_out = (dealOneImage(img_orign))
for x in range(1):
image_out = (dealOneImage(image_out))
image_out.save(f'./output/{basename}')
处理目录:
对当前目录下的文件做一个筛选,只选取图片格式的文件。
def dealSrc(srcDir):
picFiles = [os.path.join(srcDir,fn) for fn in os.listdir(srcDir) if fn.endswith(('.gif', '.jpg', '.png','.jpeg'))]
for filePath in picFiles:
dealOneFile(filePath)
小结
添加水印主要用了 python3 中的 pillow 库来实现。 首先是读取 logo 图片信息,接着在随机一个位置添加混合后的像素点信息,最后再保存起来。
其中,对于 gif 文件的处理是拆帧,再添加水印,最后再组合成一个 gif。这个只能对应比较小的 gif 文件处理,如果有更好的方法欢迎留言交流分享!
本文仅供个人学习交流使用,请勿用于其他用途!
python 实现图片批量加入水印!pillow 入门实战!的更多相关文章
- python爬虫-图片批量下载
# 爬起摄图网的图片批量下载# coding:utf-8 import requests from bs4 import BeautifulSoup from scipy.misc import im ...
- python对图片批量命名
深度学习中经常会有批量对图片进行重命名,从网上看到的资料整理一下,方便以后查看. import os class BatchRename(): ''' 批量重命名文件夹中的图片文件 ''' def _ ...
- 用python给图片添加半透明水印
# coding:utf-8 from PIL import Image, ImageDraw, ImageFont def add_text_to_image(image, text): font ...
- python 工具 图片批量合并
注:代码两处设置 region = (4,3,x-3,y-5) 目的是crop剪去图片的白边,这个可以视情况改变 图片需要命名为 x_1.png .....这样的格式 #encoding=ut ...
- python 图片格式转换png转jpg,如何利用python给图片添加半透明水印
from PIL import Imageim = Image.open(r'd:\test2.png')r, g, b, a = im.split()im = Image.merge("R ...
- Python实用案例,Python脚本,Python实现批量加水印
往期回顾 Python实现自动监测Github项目并打开网页 Python实现文件自动归类 Python实现帮你选择双色球号码 Python实现每日更换"必应图片"为"桌 ...
- [2015.07.27]万峰图片批量处理专家 v8.6
万峰图片批量处理专家,界面简洁易用,功能强大实用.支持多种处理任务同时按顺序执行,真正的批量图片,批量效果处理.支持图片批量自定义的放大缩小,旋转或者翻转,支持图片格式批量转换.支持图片批量文字水印, ...
- 10 行 Python 代码,批量压缩图片 500 张,简直太强大了
本文原创并首发于公众号[Python猫],未经授权,请勿转载. 原文地址:https://mp.weixin.qq.com/s/5hpFDgjCpfb0O1Jg-ycACw 熟悉 "Pyth ...
- 如何用python裁剪图片
如何使用python裁剪图片 如上图所示,这是一张包含了各类象棋棋子的图片.我们需要将其中每一个棋子都裁剪出来,此时可以利用python的 PIL库 实现. 一. 安装PIL库 如果此前没有安装过PI ...
随机推荐
- Just 5分钟!使用k3s部署轻量Kubernetes集群快速教程
大小仅有40MB的k3s为想要节省开销进行开发和测试的企业提供了一个很好的选择.本文将用一种极为简洁的方式,教你在5分钟之内使用k3s部署轻量Kubernetes集群. Kubernetes已经改变了 ...
- 函数进阶(三) day14
目录 昨日内容 迭代器 可迭代对象 迭代器对象 for循环原理 三元表达式 列表推导式 字典生成式 zip 生成器表达式 生成器 yield 递归 今日内容 匿名函数 内置方法 掌握 了解 异常处理 ...
- jquery链式原理.html
<!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...
- vue表单属性
<!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...
- Nginx、WSGI、 uWSGI、 uwsgi的区别
当我们部署完一个应用程序,浏览网页时具体的过程是怎样的呢?首先我们得有一个 Web 服务器来处理 HTTP 协议的内容,Web 服务器获得客户端的请求,交给应用程序,应用程序处理完,返回给 Web 服 ...
- Flask:第一个Flask项目
在上一篇文章:Flask:项目的准备工作中,我写了flask项目的准备工作,不清楚创建flask项目需要做哪些准备的朋友可以点击链接看看 1.最简单的Flask项目 代码: from flask im ...
- MySQL查询-分组取组中某字段最大(小)值所有记录
最近做东西的时候,用到一个数据库的查询.将记录按某个字段分组,取每个分组中某个字段的最大值的所有记录.举栗子来说. 已知分数表“score”,包含字段“id", "name&quo ...
- 《STL源码剖析》——List
List list位于头文件<<stl_list.h>>中 list是sequence containers中的一种 1 List的基本架构 list的基本结构的UML关系图如 ...
- Python调用函数加括号和不加括号的区别
Python调用函数加括号和不加括号的区别 # -*- coding: utf-8 -*- #!/usr/bin/env python # @Time : 2018/7/3 10:03 # @Desc ...
- STM32中断系统
1.中断介绍: 1.1 中断概念 CPU执行程序时,由于发生了某种随机的事件(外部或内部),引起CPU暂 时中断正在运行的程序,转去执行一段特殊的服务程序(中断服务子程序 或中断处理程 ...