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

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#与.NET的区别

    C#与.NET的关系 C# 可以通过.NET平台来编写 部署 运行.NET应用程序VB.NET.......NET语言 C#是专门为.NET平台而生的(面向对象) .NET平台的重要组成:1.FCL- ...

  2. 优雅的python

    在知乎上看到的问题--python有哪些优雅的代码实现. 下面的代码大概也算不上优雅. 一下代码在python3中实现 更多内容可见:http://book.pythontips.com/en/lat ...

  3. js 特效 手风琴效果

    $(document).ready(function(){ //定义展开的块 var lastBlock = $('#a1'); //展开的块的宽度 var maxWidth = 406; //折叠的 ...

  4. ThinkPHP第十天(_initialize方法,SESSION销毁,分组配置,include文件引入,JOIN用法)

    1.Action类中的_initialize()函数,先于任何自定义操作函数运行,可认为是控制器的前置操作.可用于检测用户是否登录等检测. 如果多个模块(Action)需要相同_initialize( ...

  5. 深入浅出—JAVA(2)

    2.类与对象 当你在设计类时,要记得对象是靠类的模型塑造出来的. 1.对象是已知的事物2.对象会执行的动作 对象本身已知的事物被称为 实例变量:它们代表对象的状态.切该类型的每一个对象都会独立的拥有一 ...

  6. 直播服务器Nginx

    Mac直播服务器Nginx配置对HLS的支持 在上一篇中Mac上搭建直播服务器Nginx+rtmp,我们已经搭建了nginx+rtmp直播服务器.下面需要对Nginx服务器增加对HLS的支持.在Ngi ...

  7. IE读取并显示本地图像文件的方法

    <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...

  8. 复习知识点:UITableView和UICollectionView的常用属性

    UITableView UICollectionView  //UICollectionViewLayout //UICollectionViewLayout决定了UICollectionView如何 ...

  9. POJ--1300--Door Man【推断无向图欧拉通路】

    链接:http://poj.org/problem?id=1300 题意:有n个房间.每一个房间有若干个门和别的房间相连.管家从m房间開始走.要回到自己的住处(0),问是否有一条路能够走遍全部的门而且 ...

  10. 浅析C++基础知识

    近期想对C++的面试题目进行一下更加详细的整理.事实上认真思考一下C++程序猿的面试,我们能够发现对程序猿的能力的考察总是万变不离当中,这些基础知识主要分为五部分:一. C/C++基础知识 二. C/ ...