在开发moviepy的Python程序使用pyinstaller打包后执行,报了两个错:

  1. AttributeError: module ‘moviepy.video.fx.all’ has no attribute ‘crop’
  2. AttributeError: module ‘moviepy.audio.fx.all’ has no attribute ‘audio_fadein’

这两个错是因为moviepy包下子包audio.fx.all、video.fx.all对应的目录moviepy\audio\fx\all、moviepy\video\fx\all下的包文件__init__.py中,分别使用了如下的方式import模块:

audio.fx.all子包

import pkgutil

import moviepy.audio.fx as fx

__all__ = [name for _, name, _ in pkgutil.iter_modules(
fx.__path__) if name != "all"] for name in __all__:
exec("from ..%s import %s" % (name, name))

video.fx.all子包

import pkgutil

import moviepy.video.fx as fx

__all__ = [name for _, name, _ in pkgutil.iter_modules(
fx.__path__) if name != "all"] for name in __all__:
exec("from ..%s import %s" % (name, name))

可以看到这两个子包使用的是一种动态加载模块的模式加载包下的模块的,pyinstaller对这种模式不能处理。

解决办法:

1、将这2个__init__.py文件的上面所列最后两行注释掉;

2、在Python中将上述注释掉两行代码后前面的内容复制后加一个如下输出语句的代码:

for name in __all__:
print("from moviepy.video.fx import %s" % (name))

如针对audio.fx.all,在Python中手工执行如下代码:

import pkgutil

import moviepy.video.fx as fx

__all__ = [name for _, name, _ in pkgutil.iter_modules(
fx.__path__) if name != "all"]
for name in __all__:
print("from moviepy.video.fx import %s" % (name))

输出如下:



将下面的输出语句拷贝到audio.fx.all的__init__.py,替换被注释的2行代码即可,最后完整的文件内容如下:

"""
Loads all the fx !
Usage:
import moviepy.video.fx.all as vfx
clip = vfx.resize(some_clip, width=400)
clip = vfx.mirror_x(some_clip)
""" import pkgutil import moviepy.video.fx as fx __all__ = [name for _, name, _ in pkgutil.iter_modules(
fx.__path__) if name != "all"] #for name in __all__:
# exec("from ..%s import %s" % (name, name))
#for name in __all__: print("from moviepy.video.fx import %s" % (name)) from moviepy.video.fx import accel_decel
from moviepy.video.fx import blackwhite
from moviepy.video.fx import blink
from moviepy.video.fx import colorx
from moviepy.video.fx import crop
from moviepy.video.fx import even_size
from moviepy.video.fx import fadein
from moviepy.video.fx import fadeout
from moviepy.video.fx import freeze
from moviepy.video.fx import freeze_region
from moviepy.video.fx import gamma_corr
from moviepy.video.fx import headblur
from moviepy.video.fx import invert_colors
from moviepy.video.fx import loop
from moviepy.video.fx import lum_contrast
from moviepy.video.fx import make_loopable
from moviepy.video.fx import margin
from moviepy.video.fx import mask_and
from moviepy.video.fx import mask_color
from moviepy.video.fx import mask_or
from moviepy.video.fx import mirror_x
from moviepy.video.fx import mirror_y
from moviepy.video.fx import painting
from moviepy.video.fx import resize
from moviepy.video.fx import rotate
from moviepy.video.fx import scroll
from moviepy.video.fx import speedx
from moviepy.video.fx import supersample
from moviepy.video.fx import time_mirror
from moviepy.video.fx import time_symmetrize

同样对moviepy\video\fx\all下的包文件__init__.py进行类似处理。这样处理后,再重新打包就可以解决掉该问题。

上述处理过程如果担心破坏原始的moviepy包,可以将python目录下的moviepy包拷贝一份到对应代码目录下再修改更好。

如老猿的moviepy安装目录如下:



将其复制一份到应用代码目录后应用代码目录结构如下:

此时再执行打包命令:

pyinstaller Video2Gif.py -w -p F:\study\python\project\loadApp

就可以打包后在生成的目录中直接执行Video2Gif.exe即可正常运行。

不过老猿使用PyQt开发的界面调用moviepy打包还是遇到了一点问题:

当同时使用-F和-w选项时,生成的不带其他文件的单一文件Video2Gif.exe执行时报错:



反复测试发现:

1、去掉 -F -w中任意一个都能正常执行生成的exe文件,不过去掉-F时是生成的一个带很多文件的子目录(包括要生成的exe文件),不是单一EXE文件,去掉-w时,会生成单一exe文件,但执行时除了图形界面窗口外还会出现控制台窗口,如图:



2、将代码中与moviepy相关的代码注释掉后,带 -F -w执行打包生成的exe文件能正常执行。

以上问题老猿还没找到解决办法,不过去掉 -F -w中任意一个打包的方式生成的文件也可用,因此也无需过于纠结。

修订说明:

本文部分内容存在错误,具体问题请见《关于moviepy打包报错AttributeError: module audio/video.fx.all has no attribute fadein、crop文章的纠错和抄袭

更多moviepy的介绍请参考《PyQt+moviepy音视频剪辑实战文章目录》或《moviepy音视频开发专栏》。

关于收费专栏

老猿的付费专栏《使用PyQt开发图形界面Python应用》专门介绍基于Python的PyQt图形界面开发基础教程,付费专栏《moviepy音视频开发专栏》详细介绍moviepy音视频剪辑合成处理的类相关方法及使用相关方法进行相关剪辑合成场景的处理,两个专栏加起来只需要19.9元,都适合有一定Python基础但无相关专利知识的小白读者学习。

对于缺乏Python基础的同仁,可以通过老猿的免费专栏《专栏:Python基础教程目录》从零开始学习Python。

如果有兴趣也愿意支持老猿的读者,欢迎购买付费专栏。

老猿Python,跟老猿学Python!

最新文章

  1. python django基础(二)
  2. EF+MySQL乐观锁控制电商并发下单扣减库存,在高并发下的问题
  3. ReactiveCocoa基础知识内容
  4. Spring中ApplicationContext对事件的支持
  5. Android应用性能优化之使用SparseArray替代HashMap
  6. 解析使用ThinkPHP应该掌握的调试手段
  7. 将本地仓库托管到GitHub
  8. [大牛翻译系列]Hadoop 翻译文章索引
  9. Java [Leetcode 258]Add Digits
  10. sun.misc.BASE64Encoder找不到包,解决方法
  11. DataSet、DataTable、DataRow区别
  12. java多线程向数据库写入数据
  13. HDU 1686 Oulipo(kmp)
  14. android使用百度app分享,app统计出现的异常,FrontiaApplication类
  15. 2016年,总结篇 之 VueJS 如何入门(一)
  16. JavaScript 闭包究竟是什么
  17. Java Web高级编程(三)
  18. MySQL 之 视图、触发器、存储过程、函数、事物与数据库锁
  19. H5 音频标签自定义样式修改以及添加播放控制事件
  20. .NET Core中的路由约束

热门文章

  1. (3)ASP.NET Core3.1 Ocelot认证
  2. sublime中插件
  3. nginx负载均衡基本配置和日志按天写
  4. Java中的日期
  5. 使用flexbox的自适应照片布局
  6. JLC PCB 嘉立创自动确认生产稿,不讲武德?耗子尾汁!!!
  7. centos 6系统下新磁盘分区(MBR格式)
  8. elementui checkbox复选框实现层级联动
  9. Android开发-AlertDialog,Progress,ProgressDialog,自定义layout
  10. ctf-web-sql