其实这个问题在网上能搜到大把的解决方案。大家的统一答案都是

otool -L yourapp.app/Contents/MacOS/yourapp

根据输出信息在运行 install_name_tool

install_name_tool的使用方法在这里

比较复杂的是当依赖了很多第三方库,尤其是向QT这样的库的时候。打包那叫一个麻烦啊。

QT有个官方的文档告诉你如何手动一步步打包程序,还提供了一个macdepotqt这么个程序。帮你进行打包。

可是啊,macdepotqt不支持其他第三方库,也不灵活,按照官方文档的方式手动打包,坑爹啊。

说他坑爹一点都不假,当你遇到了成片的符号需要你敲命令的话,想死的心情油然而生。

那么好了。自己动手吧。这里我给出一个我写的打包的Python脚本半成品,思想跟手动敲命令是一样的,只不过用脚本实现自动化了同时递归检查并将缺少的库考到bundle目录中。脚本中针对QT的库进行打包了,如果希望把其他的依赖库也打包的话在keywordList里边添加相应的关键字就好了。希望对大家能有帮助。

  1. import os
  2. import sys
  3. import commands
  4. binaryName = "yourapp"
  5. bundleName = "yourapp.app"
  6. bundleFrameworkDir = "Contents/Frameworks/"
  7. bundleBinaryDir = "Contents/MacOS/"
  8. bundleLibraryList = [];
  9. systemFrameworkDir = "/Library/Frameworks/"
  10. keyWordList = ["Qt"]
  11. #add more keywords to work better
  12. def hasKeyWord(word):
  13. for it in keyWordList:
  14. if word.find(it) != -1:
  15. return True
  16. return False
  17. def findApp(name):
  18. return name+bundleBinaryDir+binaryName
  19. def getBundleDependsInfo(app):
  20. dependList = commands.getoutput("otool -L " + app).replace("\t","").split("\n");
  21. del(dependList[0])
  22. dependList = [item.split(" ")[0] for item in dependList if hasKeyWord(item)];
  23. return dependList
  24. def copyLibrary(base, src, dst):
  25. systemFullPath = src
  26. print "library %s depend %s" % (os.path.basename(base), os.path.basename(dst))
  27. if not os.path.exists(dst):
  28. bundleFullPath = os.path.dirname(dst)
  29. os.system("mkdir -p %s" % (bundleFullPath))
  30. os.system("cp %s %s" % (systemFullPath, bundleFullPath))
  31. infoList = getBundleDependsInfo(dst)
  32. copyDependFiles(dst, infoList)
  33. os.system("install_name_tool -id @executable_path/../Frameworks/%s %s" % (src, dst))
  34. os.system("install_name_tool -change %s @executable_path/../Frameworks/%s %s" % (src, src, base))
  35. def getFrameworkName(dirname):
  36. if dirname.find("framework") == -1:
  37. return
  38. while not dirname.endswith("framework"):
  39. dirname = os.path.dirname(dirname)
  40. return dirname
  41. def copyFrameworkExtDir(src):
  42. sysPath = systemFrameworkDir + src
  43. destPath = ""
  44. if not os.path.exists(sysPath):
  45. return
  46. frameworkPath = getFrameworkName(sysPath)
  47. frameWorkName = getFrameworkName(src)
  48. bundlePath = bundleName + "/" + bundleFrameworkDir + frameWorkName + "/"
  49. for it in bundleFrameworkExtDir:
  50. destPath = bundlePath + it
  51. srcPath = frameworkPath + "/" + it
  52. if not os.path.exists(destPath) and os.path.exists(srcPath):
  53. print "copying %s %s" % (frameWorkName, it)
  54. os.system("cp -r %s %s" % (srcPath, destPath))
  55. def copyFramework(base, src, dst):
  56. print "framework %s depend %s" % (os.path.basename(base), os.path.basename(dst))
  57. systemFullPath = systemFrameworkDir+src
  58. if not os.path.exists(dst):
  59. bundleFullPath = os.path.dirname(dst)
  60. os.system("mkdir -p %s" % (bundleFullPath))
  61. os.system("cp %s %s" % (systemFullPath, bundleFullPath))
  62. copyFrameworkExtDir(src)
  63. infoList = getBundleDependsInfo(dst)
  64. copyDependFiles(dst, infoList)
  65. ("install_name_tool -id @executable_path/../Frameworks/%s %s" % (src, dst))
  66. os.system("install_name_tool -id @executable_path/../Frameworks/%s %s" % (src, dst))
  67. os.system("install_name_tool -change %s @executable_path/../Frameworks/%s %s" % (src, src, base))
  68. def copyDependFiles(base, infoList):
  69. targetDir = ""
  70. for it in infoList:
  71. targetDir = bundleName + "/" + bundleFrameworkDir + it
  72. if it.find("framework") != -1:
  73. copyFramework(base, it, targetDir)
  74. else:
  75. copyLibrary(base, it, targetDir)
  76. def makeBundleDirs():
  77. os.system("mkdir -p " + bundleName + "/" + bundleFrameworkDir)
  78. if __name__ == "__main__":
  79. target = findApp(bundleName + "/")
  80. makeBundleDirs()
  81. infoList = getBundleDependsInfo(target)
  82. copyDependFiles(target, infoList)

http://blog.csdn.net/livemylife/article/details/7294778

mac下的应用程序发布 及 打包(Python写的脚本,可打包第三方库)的更多相关文章

  1. Mac下将C程序创建为动态链接库再由另一个C程序调用

    写C的时候需要调用之前的一个C程序,想用动态链接库的方式.Mac下的动态链接库是dylib,与Linux下的.os或Windows下的.dll不同.由于之前没有接触过,所以翻了大量的博客,然而在编译过 ...

  2. 如何发布自己用python写的py模块

    Python——怎么发布你的Python模块 我们在学习Python的时候,除了用pip安装一些模块之外,有时候会从网站下载安装包下来安装,我也想要把我自己编写的模块做成这样的安装包,该怎么办,如何发 ...

  3. Mac下修改应用程序的菜单快捷键!

    点击左上角苹果按钮,系统偏好设置 > 键盘 > 快捷键 > 应用快捷键 点击右下角添加按钮,选择chrome程序,输入菜单中文名以及快捷键 1.如何用F5刷新 鼠标悬停在左上角的刷新 ...

  4. Mac下Homebrew将程序文件存在什么位置

    一般情况是这么操作的: 1.通过brew install安装应用最先是放在/usr/local/Cellar/目录下. 2.有些应用会自动创建软链接放在/usr/bin或者/usr/sbin,同时也会 ...

  5. linux安装配置apk打包程序gradle+jdk+Android_sdk+python自动化编译脚本

    安装gradle: 1.下载gradle包 去这里下载需要的tar.gz包:https://services.gradle.org/distributions/ 2.解压 tar zxvf gradl ...

  6. Mac下各种编程环境的配置问题(python java)

    首先,去官网下载安装包.直接运行安装.安装完成后,启动器中会多两个应用程序IDLE和Python Launcher. 如果,你习惯在IDLE,直接运行即可. 但你在Terminal中运行python3 ...

  7. 在mac下使用python抓取数据

    2015已经过去,这是2016的第一篇博文! 祝大家新年快乐! 但是我还有好多期末考试! 还没开始复习,唉,一把辛酸泪! 最近看了一遍彦祖的文章叫做 iOS程序员如何使用Python写网路爬虫 所以自 ...

  8. Python将自己写的模块进行打包

    将项目打包成模块的想法来自于flask文档教程,这不是在PyCon上和阿明合了照嘛,这不得多看看人家的东西.有兴趣的可以看看文档的项目可安装化部分,作者将flask项目打包成一个包,使其可以再任何地方 ...

  9. mac下git+maven+jenkins自动打包发布

    随着springboot+springcloud(dubbo)越来越多人使用,流行的微服务的概念越来越深入人心.分布式部署越来越复杂,给手动发布带来很大工作量.为了方便前期测试和后期线上部署更新,可使 ...

随机推荐

  1. Android GPS应用:动态获取位置信息

    在上文中,介绍了GPS概念及Android开发GPS应用涉及到的常用类和方法.在本文中,开发一个小应用,实时获取定位信息,包括用户所在的纬度.经度.高度.方向.移动速度等.代码如下: Activity ...

  2. debian下samba配置

    debian下samba配置  http://blog.chinaunix.net/uid-2282111-id-2113216.html 服务器端配置过程:1. apt-get install sa ...

  3. C++学习之虚继承

    http://blog.csdn.net/wangxingbao4227/article/details/6772579 C++中虚拟继承的概念 为了解决从不同途径继承来的同名的数据成员在内存中有不同 ...

  4. 杭电oj find your present (2)

    <span style="font-family: Arial, Helvetica, sans-serif; background-color: rgb(255, 255, 255) ...

  5. COM编程-注册DLL形式的COM服务器

    这篇文章不涉及任何的有关COM的技术的讲解,仅仅的说一下写好的DLL形式的COM怎么使用.如下图所示,我已经有了一个DLL形式的COM服务器和一个使用COM服务器的COM客户端: 现在这个DLL的CO ...

  6. Laravel OAuth2 (三) ---使用 services 和 facades

    前言 既然要判断用户是否存在,然后创建用户,那么就要实现几个功能函数.为了方便调用,于是我尝试着写了第一个service 和 facade . 创建 Facade class Social exten ...

  7. drawInRect:withAttributes:

    - (void)drawRect:(CGRect)frame { NSMutableParagraphStyle *textStyle = [[NSMutableParagraphStyle defa ...

  8. Microsoft Jet 数据库引擎找不到对象'Sheet1$_'。请确定对象是否存在,并正确地写出它的名称和路径

    We have a CRM add-on for Importing Price Lists into CRM. For this tool, we expect the details to be ...

  9. mybatis通用DAO

    扫扫关注"茶爸爸"微信公众号 坚持最初的执着,从不曾有半点懈怠,为优秀而努力,为证明自己而活. 回复:茶爸爸了解他这个人!! 花了几天的时间研究了一下mybatis的源代码,觉得这 ...

  10. 如何提高banner设计含量--网上的一篇文章--感悟

    "修改": 本质上是改什么?改大小?图片?文字?颜色? 老板说:修改本质上是提高“设计含量”.检测一个作品设计含量的高低,可以将作品中每一个设计元素进行分析,看它的“属性”与“操作 ...