python视频转化字节问题的完整实现
作者:白叔King 发布时间:2023-05-21 16:55:28
废话不多说,直接开干!
抖音字符视频在今年火过一段时间。
反正我是始终忘不了那段刘耕宏老师本草纲目的音乐…
这一次自己也来实现一波,做一个字符视频出来
百度好多都是显示模块,这个完整实现效果
步骤
将视频转化为一帧一帧的图片
把图片转化为字符画
按顺序播放字符画
1、准备
安装 Python-OpenCV 库
安装 Numpy 科学计算库
用到模块库
import time
import cv2
import os
from PIL import Image, ImageDraw, ImageFont
import numpy as np
import os
然后新建python代码文档,在开头添加上下面的导入语句
2. 材料
材料来个视频文件了,我这里用的是zimeng.mp4,下载下来和代码放到同一目录下
你也可以换成自己的,建议是学习时尽量选个短一点的视频,十几秒十秒就行了,方便调试用
此外,要选择对比度高的视频。否则的话,就需要彩色字符才能有足够好的表现,有时间我试试。
3、按帧读取视频
现在继续添加代码,实现第一步:按帧读取视频。
下面这个函数,接受视频路径和字符视频的尺寸信息,返回一个img列表,其中的img是尺寸都为指定大小的灰度图。
第一步截取图片
def video_img(file='zimeng.mp4'):
# 在当前目录下新建文件夹
folder_path = "img_bear/"
if folder_path:
pass
else:
os.makedirs(folder_path)
# 进行视频的载入
vc = cv2.VideoCapture(file)
# 判断载入的视频是否可以打开
ret = vc.isOpened()
# 循环读取视频帧
num = 0
while ret:
num = num + 1
# 进行单张图片的读取,ret的值为True或者Flase,frame表示读入的图片
ret, frame = vc.read()
if ret:
# 存储为图像
cv2.imwrite('img_bear/' + str(num) + '.jpg', frame)
# 输出图像名称
print('img_bear/' + str(num) + '.jpg')
# 在一个给定的时间内(单位ms)等待用户按键触发,1ms
cv2.waitKey(1)
else:
break
# 视频释放
vc.release()
time.sleep(0.5)
video_image(num)
如果运行没报错,就没问题
代码里的注释应该写得很清晰了,继续下一步
第二步对图片做灰度处理
视频转换成了图像,这一步便是把图像转换成字符画
上面这个函数,一个img对象为参数,前往对应的字符画
def video_image(num=''):
# 创建字符图片文件夹
folder_path = "bear/"
if folder_path:
pass
else:
os.makedirs(folder_path)
for i in range(1, num):
filename = 'img_bear/' + str(i) + '.jpg'
im = Image.open(filename) # 返回一个Image对象
width = im.size[0]
heigth = im.size[1]
print('宽:%d,高:%d' % (im.size[0], im.size[1]))
# 字符列表
ascii_char = list("$@B%8&WM#*oahkbdpqwmZO0QLCJUYXzcvunxrjft/\|()1{}[]?-_+~ <>i!lI;:,\"^`'. ")
# 判断图片是否存在
if os.path.exists(filename):
# 将图片转化为灰度图像,并重设大小
img_array = np.array(Image.open(filename).resize((160, 160), Image.ANTIALIAS).convert('L'))
# 创建新的图片对象
img = Image.new('L', (width, heigth), 255)
draw_object = ImageDraw.Draw(img)
# 设置字体
font = ImageFont.truetype('consola.ttf', 10, encoding='unic')
# 根据灰度值添加对应的字符
for j in range(160):
for k in range(160):
x, y = k * 8, j * 8
index = int(img_array[j][k] / 4)
draw_object.text((x, y), ascii_char[index], font=font, fill=0)
name = 'bear/' + str(i) + '.jpg'
print(name)
# 保存字符图片
img.save(name, 'JPEG')
time.sleep(0.5)
video(num)
第三步字符转视频
写了这么多代码,如今终于要出效果了。如今就是最激动人心的一步:播放字符画了。
异样的,我把它封装成了一个函数。上面这个函数承受一个字符画的列表并播放。
def video(num):
filename = 'img_bear/' + str(1) + '.jpg'
im = Image.open(filename) # 返回一个Image对象
width = im.size[0]
heigth = im.size[1]
# 设置视频编码器,这里使用使用MJPG编码器
fourcc = cv2.VideoWriter_fourcc(*'MJPG')
# 输出视频参数设置,包含视频文件名、编码器、帧率、视频宽高(此处参数需和字符图片大小一致)
videoWriter = cv2.VideoWriter('bear_character.avi', fourcc, 20.0, (width, heigth))
for i in range(1, num):
filename = 'bear/'+str(i)+'.jpg'
# 判断图片是否存在
if os.path.exists(filename):
img = cv2.imread(filename=filename)
# 在一个给定的时间内(单位ms)等待用户按键触发,100ms
cv2.waitKey(100)
# 将图片写入视频中
videoWriter.write(img)
print(str(i) + '.jpg' + ' done!')
# 视频释放
videoWriter.release()
time.sleep(1)
# 删除图片
remove_img()
remove_img_bear()
下面完整代码
可能要等很久。我使用示例视频大概需要 500 秒左右。
ctrl+f10执行对应的文件
完整代码里面加了
执行生成图片,生成灰度图片,最后通过灰度生成字节视频删除多余文件
说了那太多废话就是:最后还需删除一些临时的文件及文件夹。
import time
import cv2
import os
from PIL import Image, ImageDraw, ImageFont
import numpy as np
import os
# 第一步截取图片
def video_img(file='zimeng.mp4'):
# 在当前目录下新建文件夹
folder_path = "img_bear/"
if folder_path:
pass
else:
os.makedirs(folder_path)
# 进行视频的载入
vc = cv2.VideoCapture(file)
# 判断载入的视频是否可以打开
ret = vc.isOpened()
# 循环读取视频帧
num = 0
while ret:
num = num + 1
# 进行单张图片的读取,ret的值为True或者Flase,frame表示读入的图片
ret, frame = vc.read()
if ret:
# 存储为图像
cv2.imwrite('img_bear/' + str(num) + '.jpg', frame)
# 输出图像名称
print('img_bear/' + str(num) + '.jpg')
# 在一个给定的时间内(单位ms)等待用户按键触发,1ms
cv2.waitKey(1)
else:
break
# 视频释放
vc.release()
time.sleep(0.5)
video_image(num)
# 第二步对图片做灰度处理
def video_image(num=''):
# 创建字符图片文件夹
folder_path = "bear/"
if folder_path:
pass
else:
os.makedirs(folder_path)
for i in range(1, num):
filename = 'img_bear/' + str(i) + '.jpg'
im = Image.open(filename) # 返回一个Image对象
width = im.size[0]
heigth = im.size[1]
print('宽:%d,高:%d' % (im.size[0], im.size[1]))
# 此字符表用于生字符帧,对应256个像素,字符越多且不同样式,字符帧越精细
ascii_char = list("$@B%8&WM#*oahkbdpqwmZO0QLCJUYXzcvunxrjft/\|()1{}[]?-_+~ <>i!lI;:,\"^`'. ")
# 判断图片是否存在
if os.path.exists(filename):
# 将图片转化为灰度图像,并重设大小
img_array = np.array(Image.open(filename).resize((160, 160), Image.ANTIALIAS).convert('L'))
# 创建新的图片对象
img = Image.new('L', (width, heigth), 255)
draw_object = ImageDraw.Draw(img)
# 设置字体
font = ImageFont.truetype('consola.ttf', 10, encoding='unic')
# 根据灰度值添加对应的字符
for j in range(160):
for k in range(160):
x, y = k * 8, j * 8
index = int(img_array[j][k] / 4)
draw_object.text((x, y), ascii_char[index], font=font, fill=0)
name = 'bear/' + str(i) + '.jpg'
print(name)
# 保存字符图片
img.save(name, 'JPEG')
time.sleep(0.5)
video(num)
# 第三步字符转视频
def video(num):
filename = 'img_bear/' + str(1) + '.jpg'
im = Image.open(filename) # 返回一个Image对象
width = im.size[0]
heigth = im.size[1]
# 设置视频编码器,这里使用使用MJPG编码器
fourcc = cv2.VideoWriter_fourcc(*'MJPG')
# 输出视频参数设置,包含视频文件名、编码器、帧率、视频宽高(此处参数需和字符图片大小一致)
videoWriter = cv2.VideoWriter('bear_character.avi', fourcc, 20.0, (width, heigth))
for i in range(1, num):
filename = 'bear/'+str(i)+'.jpg'
# 判断图片是否存在
if os.path.exists(filename):
img = cv2.imread(filename=filename)
# 在一个给定的时间内(单位ms)等待用户按键触发,100ms
cv2.waitKey(100)
# 将图片写入视频中
videoWriter.write(img)
print(str(i) + '.jpg' + ' done!')
# 视频释放
videoWriter.release()
time.sleep(1)
# 删除图片
remove_img()
remove_img_bear()
# 原图片删除
def remove_img():
files = os.getcwd() # files中保存的是当前的执行目录
file_name = files + "/img_bear"
del_list = os.listdir(file_name)
for f in del_list:
file_path = os.path.join(file_name, f)
print(file_path)
if os.path.isfile(file_path):
os.remove(file_path)
print('成功删除文件:')
else:
print('未找到此文件:')
# 灰度图片删除
def remove_img_bear():
files = os.getcwd() # files中保存的是当前的执行目录
file_name = files + "/bear"
del_list = os.listdir(file_name)
for f in del_list:
file_path = os.path.join(file_name, f)
print(file_path)
if os.path.isfile(file_path):
os.remove(file_path)
print('成功删除文件:')
else:
print('未找到此文件:')
def main():
video_img('video.mp4')
if __name__ == "__main__":
main()
进一步优化
到了这里,核心功能基本都完成了。
不过仔细想想,其实还有很多可以做的:
什么是指定要转换的区间、帧率?
每次转换都要很久的时间,能不能边转换边播放?或者转换后把数据保存起来,下次播放时,就直接读缓存
看下效果图
来源:https://blog.csdn.net/weixin_37254196/article/details/124473889


猜你喜欢
- 创建:list = [5,7,9]取值和改值:list[1] = list[1] * 5列表尾插入:list.append(4)去掉第0个值
- 目前SQL INJECTION的攻击测试愈演愈烈,很多大型的网站和论坛都相继被注入。这些网站一般使用的多为SQL SERVER数据库,正因为
- 一、前言MYSQL中MDL锁一直是一个比较让人比较头疼的问题,我们谈起锁一般更加倾向于INNODB下层的gap lock、next key
- 前言本篇文章主要讲解vue响应式原理的逻辑,也就是vue怎么从最开始一步步推导出响应式的结构框架。 先从头构建一个简单函数推导出Vue3的R
- 这篇文章主要介绍了Python urlopen()和urlretrieve()用法解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作
- scapy是python写的一个功能强大的交互式数据包处理程序,可用来发送、嗅探、解析和伪造网络数据包,常常被用到网络攻击和测试中。这里就直
- 扩展Code:var blogModule = (function (my) { my.AddPhoto = function (
- CSS3草案中定义了{opacity:来声明元素的透明度,这已经得到了大多数现代浏览器的支持,而IE则很早通过特定的私有属性filter来实
- 一、subprocess以及常用的封装函数运行python的时候,我们都是在创建并运行一个进程。像Linux进程那样,一个进程可以fork一
- 如下所示:node2:/django/mysite/blog#cat views.py1,# -*- coding: utf-8 -*-fr
- 一个最最简单的例子:绘制一个从 0 到 360 度完整的 SIN 函数图形import numpy as npimport matplotl
- 前言之前一直写不出来,这周周日花了一下午终于弄懂了, 顺便放博客里,方便以后忘记了再看看。 要实现的是输入一张 图,起点,终点,输出起点和终
- 上传html文件内容如下:操作步骤<html><head><meta http-equiv="con
- 一、设置主窗口# -*- coding: utf-8 -*-import tkinter from tkinter import ttkim
- 在数组中搜索一个特定值,如果找到返回TRUE否则返回FALSE boolean in_array(mixed needle,array ha
- Python版本是2.7.9,在win8上测试成功,就是抓取有点慢,本来想用多线程的,有事就罢了。模板之家的网站上的url参数与页数不匹配,
- 约定:import pandas as pdimport numpy as npReIndex重新索引reindex()是pandas对象的
- 最近做了一个项目其中有项目需求涉及到访问控制,在访问需要登录才能使用的页面或功能时,会弹出登录框:效果如下: 图 1-点击用户名时,如未登录
- Pygame是一个超好用的SDL绑定。自从有了Pygame,妈妈再也不用担心我内存泄漏了。但是这里有一个问题,Pygame的Movie模块已
- 本文实例讲述了Python计算一个给定时间点前一个月和后一个月第一天的方法。分享给大家供大家参考,具体如下:python如何获取一个时间点的