目录
  • 步骤
    • 1、准备
    • 2. 材料
    • 3、按帧读取视频
  • 什么是指定要转换的区间、帧率?
    • 看下效果图
      • 总结

        废话不多说,直接开干!

        抖音字符视频在今年火过一段时间。

        反正我是始终忘不了那段刘耕宏老师本草纲目的音乐…

        这一次自己也来实现一波,做一个字符视频出来

        百度好多都是显示模块,这个完整实现效果

        步骤

        将视频转化为一帧一帧的图片

        把图片转化为字符画

        按顺序播放字符画

        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视频转化字节问题的完整实现

        总结

        声明:本站所有文章,如无特殊说明或标注,均为本站原创发布。任何个人或组织,在未征得本站同意时,禁止复制、盗用、采集、发布本站内容到任何网站、书籍等各类媒体平台。如若本站内容侵犯了原著者的合法权益,可联系我们进行处理。