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

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. C++设计模式之状态模式(四)

    4.状态模式总结 状态模式将一个对象在不同状态下的不同行为封装在一个个状态类中,通过设置不同的状态对象能够让环境对象拥有不同的行为.而状态转换的细节对于client而言是透明的.client不直接操作 ...

  2. c 有关N!阶乘的相关问题----陆续补充上来

    第一个:求N!结果中末尾0的个数问题.思路是末尾0的产生   5*偶数,阶乘中偶数的个数肯定比5多,所以求出阶乘中5的个数就可以求出末尾0的个数. #include<stdio.h> in ...

  3. eclipse主题插件

    打开eclipse ,选择 Help 选择Install New Software 点击 Add 输入http://eclipse-color-theme.github.com/update,选中Ec ...

  4. mobilize扁平化的fullPage.js类工具使用心得

    可以生成一个fullPage效果的主页,但是列表页面和内容页面呢? 主页中的block,可以选择多种组建生成.甚至连form都有: 应该改造其源代码,动态化和cms系统化,添加二三级页面模板: == ...

  5. 基于RSA securID的Radius二次验证java实现(PAP验证方式)

    基于rsa SecurID的二次验证.RSA server自身可以作为Radius服务器,RSA也可以和其他的软件集合,使用其他的server作为Radius服务器. radius的验证的一般流程如下 ...

  6. jquery 动态增加的html元素,初始化设置在id或class上的事件无效

    一般情况,我们会在页面初始化完成后对class定义一些全局事件,举个栗子: $(document).ready(function(){ $(".class").on("m ...

  7. django cbv

    django 提供了一系列现成的类视图,他们都继承自一个 View 基类(django.views.generic.base.View).在这个基类里实现了与 URLs 的接口(as_view).请求 ...

  8. 一个包含所有c++的头文件的头文件

    #include <bits/stdc++.h> 做CF看见别人用这个函数,然后就能直接用vector,set,string那些函数了,摸不着头脑,感觉特神奇就百度了一下,才发现这个是C+ ...

  9. ZOJ3640-Help Me Escape

    Help Me Escape Time Limit: 2 Seconds      Memory Limit: 32768 KB Background     If thou doest well, ...

  10. vs2013内置IISExpress相关问题

    问题描述,以前做的程序迁移到vs2013后出现500.22问题. HTTP 错误 500.22 - Internal Server Error 检测到在集成的托管管道模式下不适用的 ASP.NET 设 ...