详解Framework

2024-09-02 16:32
文章标签 详解 framework

本文主要是介绍详解Framework,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

[Cocoa]深入浅出 Cocoa 之 Framework
罗朝辉( http://blog.csdn.net/kesalin/)
CC许可,转载请注明出处

Framework 简介

Mac OS X 扩展了 framework 的功能,让我们能够利用它来共享代码和资源。framework 在概念上有点像 Window 下的库,但是比库更加强大,通过 framework 我们可以共享所有形式的资源,如动态共享库,nib 文件,图像字符资源以及文档等。系统会在需要的时候将 framework 载入内存中,多个应用程序可以同时使用同一个 framework,而内存中的拷贝只有一份。一个 framework 同时也是一个 bundle,我们可以在 finder 里浏览其内容,也可以在代码中通过 NSBundle 访问它。利用 framework 我们可以实现动态或静态库的功能。与动态/静态库相比,framework 有如下优势:

第一,framework 能将不同类型的资源打包在一起,使之易于安装,卸载与定位;
第二,framework 能够进行版本管理,这使得 framework 能不断更新并向后兼容;
第三,在同一时间,即使有多个应用程序使用同一 framework,但在内存中只有一份 framework 只读资源的拷贝,这减少了对内存的占用

Framework 的结构

下面是一个带有A,B两个版本和一个 resources 目录的 framework 结构,并设定当前版本为 B:

[cpp]  view plain copy print ?
  1. MyFramework.framework/  
  2.     Headers      -> Versions/Current/Headers  
  3.     MyFramework  -> Versions/Current/MyFramework  
  4.     Resources    -> Versions/Current/Resources  
  5.     Versions/  
  6.         A/  
  7.             Headers/  
  8.                 MyHeader.h  
  9.             MyFramework  
  10.             Resources/  
  11.                 English.lproj/  
  12.                     Documentation  
  13.                     InfoPlist.strings  
  14.                 Info.plist  
  15.         B/  
  16.             Headers/  
  17.                 MyHeader.h  
  18.             MyFramework  
  19.             Resources/  
  20.                 English.lproj/  
  21.                     Documentation  
  22.                     InfoPlist.strings  
  23.                 Info.plist  
  24.         Current  -> B  

结合上面的结构,下面我们来看本例中 ExampleFramework 的结构图:


Framework 存放位置
在 Mac OS 中有三个级别的位置来存放 framework。一般我们自己编写的 framework 都应该是应用程序级别。
1,系统级,/Library/Frameworks,放置到该级别,这需要管理员权限,整个系统都可以共享使用该级别的 framework;
2,用户级,/Users/用户名/Library/Frameworks,拥有用户权限的应用程序都可以共享使用该级别的 framework;
3,应用程序级。

在应用程序中内嵌 Framework

1,创建 Framework
新建一个名为 FrameworkDemo 的 Cocoa application 工程,然后选中项目名,向其中添加名为 ExampleFramework  的 Cocoa Framework。

2,添加内容
向 Framework 中添加源代码(请下载源代码),并导出需要向外部公开的头文件。

导出头文件有一些技巧:
1,如果有我们不想向用户公开的类名出现在必须公开的头文件中,我们可以使用 id 替代该类名或使用 @class 前置申明来避免导出该类的头文件,在本例中使用 id 替代 InternalObject,从而避免导出 InternalObject 类的头文件。
2,如果需要导出多个头文件,常见的做法是新建一个与 framework 同名的 .h 文件,将需要导出的头文件包含到该头文件中来。如本例中的  ExampleFramework.h。

3,修改 framework build 选项
我们在使用自己编写的库时,常碰到下面的编译错误:

[cpp]  view plain copy print ?
  1. Library not loaded: path/to/framework  
  2. Referenced from: path/to/app/  
  3. Reason: image not found  
这多半是由于 framework 的 Installation Directory 编译选项设置不正确,导致应用程序无法正确定位 framework 所致。这需要我们设置编译选项 Installation Directory 为 @executable_path/../Frameworks。

4,使用 framework
至此,framework 编写完成,下面我们来在 FrameworkDemo 中来使用它。首先我们需要将 ExampleFramework 导入到 FrameworkDemo 中来,这样 FrameworkDemo 在运行时才能定位该 framework。新建一个 Add copy files 型的 build phase,设置其 destination 为 framework,加入已经编写好的 ExampleFramework。

导入 framework 之后,我们就可以在工程中使用该 framework 了。编写如下代码:

[cpp]  view plain copy print ?
  1. //  
  2. //  FrameworkDemoAppDelegate.m  
  3. //  FrameworkDemo  
  4. //  
  5. //  Created by kesalin on 11-10-16.  
  6. //  Copyright 2011年 kesalin@gmail.com. All rights reserved.  
  7. //  
  8.   
  9. #import "FrameworkDemoAppDelegate.h"  
  10. <span style="color:#ff6666;"><strong>#import <ExampleFramework/ExampleFramework.h></strong></span>  
  11.   
  12. @implementation FrameworkDemoAppDelegate  
  13.   
  14. @synthesize window;  
  15.   
  16. - (void)applicationDidFinishLaunching:(NSNotification *)aNotification  
  17. {  
  18. <span style="color:#cc9933;">        EntityObjectA *objectA = [[EntityObjectA alloc] init];  
  19.     EntityObjectB *objectB = [[EntityObjectB alloc] init];  
  20.       
  21.     NSLog(@"Object A called: %@", [objectA methodOne]);  
  22.     NSLog(@"Object B called: %@", [objectA methodTwo]);  
  23.       
  24.     NSLog(@"Object B called: %@", [objectB methodOne]);  
  25.     NSLog(@"Object B called: %@", [objectB methodTwo]);</span>  
  26. }  
  27.   
  28. @end  
注意:我们使用 framework 的方式为 :framework名/framework名.h,这是约定的常规做法,Cocoa 自带的 framework 也都遵守这一约定,所以我们自己编写的库最后也遵守这一约定。

5,编译运行
至此,工作完成,编译运行,应当输出如下:

[cpp]  view plain copy print ?
  1. Object A called: EntityObjectA:methodOne  
  2. Object B called: EntityObjectA:methodTwo - InternalObject:description  
  3. Object B called: EntityObjectB:methodOne  
  4. Object B called: EntityObjectB:methodTwo - InternalObject:description  

6,清除冗余文件
这时可选项,且只对使用内嵌 framework 的应用程序有效。当我们拷贝导入 framework 之后,应用程序 bundle 已经拷贝了一份 framework,那么原本编译生成的那一份 framework就变得多余了,我们可以将其清理掉。在使用内嵌 framework 的应用程序的 build phases 中加入 run script phase,脚本内容如下:
[cpp]  view plain copy print ?
  1. echo "build path ${TARGET_BUILD_DIR}"  
  2. cd ${TARGET_BUILD_DIR}/${FULL_PRODUCT_NAME}/Contents/Frameworks   
  3. rm -rf */Headers   
  4. rm -rf */Versions/*/Headers 
  5. rm -rf */Versions/*/PrivateHeaders  
  6. rm -rf */Versions/*/Resources/*/Contents/Headers  


使用外部 framework

上面的示例是在应用程序内嵌 framework,供应用程序本身使用,很多时候,我们是使用第三方编写的 framework,下面接着来演示如何将 ExampleFramework 当做外部framework。
1,新建名为 TestExampleFramework 的 Cocoa Application 程序,在 TestExampleFrameworkAppDelegate.m 中添加如上步骤 4 中使用 framework 的代码。

2,编译运行,这时会报找不到头文件,类名的错误。这时因为我们还没有导入framework。在 Build Phase 的 Link Binary With Libraries 中加入生成好的 ExampleFramework,该 framework 的默认生成路径在: /用户名/Library/Developer/XCode/DerivedData/FrameworkDemo-XXXX/Build/Products/Debug/下。至此,编译运行,输出应当如上步骤 5 相同。


此外还有一种方式使用第三方 framework,如果我们拥有第三方 framework 的源代码工程,想在我们的工程中编译该 framework,并使用它。我们可以将第三方 framework 的工程文件加入我们自己的工程,并在 Target Dependencies 和 Link Binary With Libraires 加入第三方 framework,这样我们就可以使用该 framework了。如下图所示:


这篇关于详解Framework的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1130482

相关文章

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

OpenHarmony鸿蒙开发( Beta5.0)无感配网详解

1、简介 无感配网是指在设备联网过程中无需输入热点相关账号信息,即可快速实现设备配网,是一种兼顾高效性、可靠性和安全性的配网方式。 2、配网原理 2.1 通信原理 手机和智能设备之间的信息传递,利用特有的NAN协议实现。利用手机和智能设备之间的WiFi 感知订阅、发布能力,实现了数字管家应用和设备之间的发现。在完成设备间的认证和响应后,即可发送相关配网数据。同时还支持与常规Sof

6.1.数据结构-c/c++堆详解下篇(堆排序,TopK问题)

上篇:6.1.数据结构-c/c++模拟实现堆上篇(向下,上调整算法,建堆,增删数据)-CSDN博客 本章重点 1.使用堆来完成堆排序 2.使用堆解决TopK问题 目录 一.堆排序 1.1 思路 1.2 代码 1.3 简单测试 二.TopK问题 2.1 思路(求最小): 2.2 C语言代码(手写堆) 2.3 C++代码(使用优先级队列 priority_queue)

K8S(Kubernetes)开源的容器编排平台安装步骤详解

K8S(Kubernetes)是一个开源的容器编排平台,用于自动化部署、扩展和管理容器化应用程序。以下是K8S容器编排平台的安装步骤、使用方式及特点的概述: 安装步骤: 安装Docker:K8S需要基于Docker来运行容器化应用程序。首先要在所有节点上安装Docker引擎。 安装Kubernetes Master:在集群中选择一台主机作为Master节点,安装K8S的控制平面组件,如AP

嵌入式Openharmony系统构建与启动详解

大家好,今天主要给大家分享一下,如何构建Openharmony子系统以及系统的启动过程分解。 第一:OpenHarmony系统构建      首先熟悉一下,构建系统是一种自动化处理工具的集合,通过将源代码文件进行一系列处理,最终生成和用户可以使用的目标文件。这里的目标文件包括静态链接库文件、动态链接库文件、可执行文件、脚本文件、配置文件等。      我们在编写hellowor

LabVIEW FIFO详解

在LabVIEW的FPGA开发中,FIFO(先入先出队列)是常用的数据传输机制。通过配置FIFO的属性,工程师可以在FPGA和主机之间,或不同FPGA VIs之间进行高效的数据传输。根据具体需求,FIFO有多种类型与实现方式,包括目标范围内FIFO(Target-Scoped)、DMA FIFO以及点对点流(Peer-to-Peer)。 FIFO类型 **目标范围FIFO(Target-Sc

019、JOptionPane类的常用静态方法详解

目录 JOptionPane类的常用静态方法详解 1. showInputDialog()方法 1.1基本用法 1.2带有默认值的输入框 1.3带有选项的输入对话框 1.4自定义图标的输入对话框 2. showConfirmDialog()方法 2.1基本用法 2.2自定义按钮和图标 2.3带有自定义组件的确认对话框 3. showMessageDialog()方法 3.1

脏页的标记方式详解

脏页的标记方式 一、引言 在数据库系统中,脏页是指那些被修改过但还未写入磁盘的数据页。为了有效地管理这些脏页并确保数据的一致性,数据库需要对脏页进行标记。了解脏页的标记方式对于理解数据库的内部工作机制和优化性能至关重要。 二、脏页产生的过程 当数据库中的数据被修改时,这些修改首先会在内存中的缓冲池(Buffer Pool)中进行。例如,执行一条 UPDATE 语句修改了某一行数据,对应的缓

Spring Framework系统框架

序号表示的是学习顺序 IoC(控制反转)/DI(依赖注入): ioc:思想上是控制反转,spring提供了一个容器,称为IOC容器,用它来充当IOC思想中的外部。 我的理解就是spring把这些对象集中管理,放在容器中,这个容器就叫Ioc这些对象统称为Bean 用对象的时候不用new,直接外部提供(bean) 当外部的对象有关系的时候,IOC给它俩绑好(DI) DI和IO

OmniGlue论文详解(特征匹配)

OmniGlue论文详解(特征匹配) 摘要1. 引言2. 相关工作2.1. 广义局部特征匹配2.2. 稀疏可学习匹配2.3. 半稠密可学习匹配2.4. 与其他图像表示匹配 3. OmniGlue3.1. 模型概述3.2. OmniGlue 细节3.2.1. 特征提取3.2.2. 利用DINOv2构建图形。3.2.3. 信息传播与新的指导3.2.4. 匹配层和损失函数3.2.5. 与Super