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()

进一步优化

到了这里,核心功能基本都完成了。

不过仔细想想,其实还有很多可以做的:

什么是指定要转换的区间、帧率?

每次转换都要很久的时间,能不能边转换边播放?或者转换后把数据保存起来,下次播放时,就直接读缓存

看下效果图

python视频转化字节问题的完整实现

python视频转化字节问题的完整实现

来源:https://blog.csdn.net/weixin_37254196/article/details/124473889

标签:python,字节,转换
0
投稿

猜你喜欢

  • 基于Python实现牛牛套圈小游戏的示例代码

    2022-04-04 03:50:23
  • 完美解决mysql启动后随即关闭的问题(ibdata1文件损坏导致)

    2024-01-12 14:43:03
  • 详解Python如何实现批量为PDF添加水印

    2022-06-20 23:33:58
  • JS加载器如何动态加载外部js文件

    2024-04-16 08:47:06
  • js中arguments的用法(实例讲解)

    2024-06-15 08:20:08
  • js实现楼层效果的简单实例

    2024-06-07 15:26:01
  • 一个入门级python爬虫教程详解

    2023-03-27 17:34:08
  • 白 刃之战:PHP vs. ASP.NET(节选)-架构比较

    2023-11-15 12:31:22
  • Java正则表达式API边界匹配

    2023-07-03 19:36:18
  • 从零开始实现Vue简单的Toast插件

    2024-05-13 09:13:55
  • 详解用python -m http.server搭一个简易的本地局域网

    2023-09-29 13:03:34
  • 网页特效文字之—粗糙字

    2013-07-23 04:34:56
  • 机器学习经典算法-logistic回归代码详解

    2021-05-06 23:56:12
  • 基于Python中isfile函数和isdir函数使用详解

    2023-05-10 00:46:50
  • 精致的web设计

    2009-12-04 19:07:00
  • python 调用win32pai 操作cmd的方法

    2023-11-15 12:32:09
  • Python爬取qq空间说说的实例代码

    2021-04-28 01:29:50
  • PHP+Mysql基于事务处理实现转账功能的方法

    2023-10-21 16:53:06
  • python内存管理分析

    2022-04-16 22:03:03
  • 一次MySQL性能优化实战

    2009-03-09 15:01:00
  • asp之家 网络编程 m.aspxhome.com