iPhone开发指南-应用程序核心

应用程序内核

每个iPhone应用程序都基于UIKit框架而构建,因此拥有同样的核心架构。UIKit提供运行应用程序的关键对象并且协调用户输入处理和屏幕内容显示。应用程序彼此之间区分开来的地方在于如何配置这些缺省对象和它们如何把自定义对象整合到它们的应用程序用户界面和行为中。

尽管定制你的应用程序用户界面和基本行为发生在你的自定义代码中,仍然有很多定制必须使用在应用程序的最高层。因为这些应用层的定制影响了你的应用程序和系统以及其它安装程序之间的交互方式,理解什么时候该采取行动而什么时候默认行为已经足够这一点很重要。本章提供了一个核心应用程序架构和高级别定制点方面的总体描述来帮助你做出决定。

核心应用程序架构

从你的程序被用户启动,到它退出,UIKit框架管理着大多数应用程序关键基础结构。一个iPhone应用程序不断从系统中接收事件并必须响应这些事件。接收事件是UIApplication 对象的工作但是响应这些事件是你自定义代码的责任。为了理解你在哪些地方需要响应事件,其实,这对理解一点整个应用程序生命周期和事件循环也有帮助。下面的章节描述了这些周期并且还提供了一些贯穿iPhone应用程序开发的核心设计模式的总结。

应用程序生命周期

应用程序生命周期构成发生在你的应用程序启动和退出期间的事件序列。在iPhone 操作系统中, 用户通过点击桌面上的菜单启动你的应用程序。在点击发生之后的短时间内,系统显示一些过渡图片并开始通过调用main函数启动你的应用程序。从这一点开始,一大堆的初始化工作被移交给UIKit,它将加载应用程序的用户界面并准备好它的事件循环。在事件循环期间,UIKit 协调你自定义对象的事件交付和应用程序发出命令的响应。当用户执行一个动作会让你的应用程序退出时,UIKit通知你的应用程序并开始这个结束过程。

1-1 描绘了一个iPhone应用程序的简单生命周期。这个图显示了应用程序启动到退出期间发生的事件序列。在初始化和结束阶段,UIKit发送特定的消息给应用程序代理对象以便其知道发生了什么。在事件循环阶段,UIKit分发事件给你的应用程序自定义事件处理器。处理初始化和结束事件在“Initialization and Termination,” 中描述,而事件处理过程在“The Event-Handling Cycle” 中介绍,在后面的章节将覆盖更多的细节。

Figure 1-1  应用程序生命周期

 

iphone app life cycle

 

Main函数

iPhone应用程序里, main函数功能被最小化了。大部分实际工作是在UIApplicationMain 函数中完成的。 当你在Xcode中开始一个新的应用程序项目时,每个项目模版都提供了一个标准main函数实现如同在 “Handling Critical Application Tasks.”里的那个。Main函数只做了三件事: 创建了一个自释放池(autorelease pool,调用UIApplicationMain,然后释放autorelease pool 除了很少的特例,你不应该修改它。

Listing 1-1  iPhone应用程序的main函数

#import <UIKit/UIKit.h>

 

int main(int argc,char *argv[])

{

    NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];

    int retVal = UIApplicationMain(argc,argv,nil,nil);

    [pool release];

    return retVal;

}

注意: 自释放池用在内存管理中。它是一个Cocoa机制用来延迟在一个函数体内创建的对象的释放。更多信息参见Memory Management Programming Guide for Cocoa。对于和自释放池相关的iPhone应用程序特定的内存管理指南,参见“Allocating Memory Wisely.”

上述列表中的中心部分UIApplicationMain函数采用了4个参数并使用它们来初始化应用程序。尽管你没必要更改传入参数的默认值,还是值得解释一下它们起动应用程序时的用途。除了argc argv 参数外,这个函数采用两个字符串参数来识别基本类(也就是,应用程序对象类)和应用程序代理类。如果这个基本类字符串为空,UIKit使用UIApplication类作为缺省值。 如果应用程序代理类为空,UIKit假设它为从你的应用程序主nib文件加载的对象其中之一(对于使用Xcode模版创建的应用程序)。 设置这些参数任意一个为非空数值将导致UIApplicationMain 函数在应用程序启动时创建一个相应类实例并为声明的目的使用它。这样,如果你的应用程序使用一个自定义的UIApplication子类(不推荐这样,但是当然是可能的),你将在第三个参数中指定你自定义类的名字。

应用程序代理

监控你的应用程序的高层行为是应用程序代理对象的职责,也就是你提供的自定义对象。代理是用来避免子类化复杂UIKit对象的一个机制,比如缺省的UIApplication对象。和使用子类化以及重写父类方法相反,你无需修改就可以使用这个复杂对象而把自定义的代码放在代理类中。当感兴趣的事件发生时,这个复杂对象发送消息给你的代理对象。你可以使用这些“钩子”来执行自定义代码并实现你需要的行为。

重要: 这个代理设计模式是用来节约你创建应用程序的时间和精力的。所以理解这个模式很重要。想对iPhone应用程序采用的关键设计模式有个总体了解,请参见“Fundamental Design Patterns.” 关于代理和其他UIKit设计模式的更详细的描述,请参见Cocoa Fundamentals Guide.

应用程序代理对象负责处理一些关键的系统消息而且在每个iPhone应用程序中必须存在。这个对象可以是任何你喜欢的类实例,只要它采用了UIApplicationDelegate 协议。 这个协议的方法定义了应用程序生命周期挂载的钩子,同时也是你实现自定义行为的途径。尽管你不需要实现所有的方法,每个应用程序代理应该实现在“Handling Critical Application Tasks.”中描述的方法。

关于UIApplicationDelegate协议的另外的信息,请参见UIApplicationDelegate Protocol Reference

Nib文件

另外一个初始化时期发生的任务是加载应用程序的主nib文件。如果应用程序信息property list (Info.plist) 文件包含了NSMainNibFile关键值, 作为初始化过程的一部分,UIApplication 对象加载这个关键值指定的nib文件。主nib文件是为你自动加载的唯一nib文件;但是,你可以按照需要加载其他的nib文件。

Nib 文件是基于磁盘的资源文件,保存了一个或多个对象的一份快照。 一个iPhone应用程序的主nib文件通常包含一个窗口对象,应用程序代理对象,和可能一个或多个其他管理这个窗口的关键对象。加载一个nib文件重新构成nib文件中的对象,从它的磁盘表示转换成一个实际的可以被你的应用程序操作的内存版本。从nib文件中加载的对象和你编程序创建的对象没有区别。不过,对于用户界面而言,图形化的创建和用户界面相关联的对象并存放在nib文件中(使用Interface Builder)比编程实现要简便直观得多。

关于nib文件以及如何使用的更多信息,请参见“Nib Files.”。更多关于如何指定你的应用程序主nib文件的信息,请参见“The Information Property List.”

事件处理循环

UIApplicationMain 函数初始化了应用程序之后,它起动必要的基础组件来管理这个应用程序的事件和绘制循环,这在图Figure 1-2中描述。当用户和设备交互时,iPhone OS 侦测触摸事件并把它们放到应用程序事件队列中。UIApplication对象的事件处理组件从事件队列顶部提取每个事件并递交给最合适的对象来处理它。比如,一个发生在一个按钮上的触摸事件将会被递交给相应的按钮对象。事件也可以被递交给控制器对象(controller objects)和其他不是直接负责处理该触摸事件的对象。

Figure 1-2  事件和绘画循环

 

event draw cycle

iPhone OS 多点触摸事件模型中,触摸数据被包装在一个简单的事件对象中(UIEvent)。为了跟踪单独的触摸动作,事件对象中包含了触摸对象(UITouch),每一个代表一个手指触摸了屏幕。当这个用户把手指放在屏幕上,并四处移动它们,最后从屏幕上移开,系统会在相应的触摸对象中报告每个手指的变化。

当应用程序启动时,系统为这个应用程序创建一个进程和线程。这个初始线程成为应用程序主线程, UIApplication 对象就在这里建立主运行循环(main run loop)并配置应用程序的事件处理编码。1-3 显示了事件处理编码和主运行循环之间的关系。系统发送的触摸事件被排队直到它们能被应用程序的主循环处理。

Figure 1-3  在主运行循环中处理事件

 

main event loop

注意: 一个运行循环监控一个给定的执行线程的输入源。当一个输入源有数据处理时,这个运行循环唤醒线程并把控制权派发给输入源处理器。当处理完成时,控制权返回运行循环,继续下一个事件或者如果没有什么事情做的话就让这个线程休眠。你可以安装你自己的输入源,包括端口和时钟,在一个运行循环中使用基础框架中的NSRunLoop 类。更多关于NSRunLoop 以及运行循环的总体描述,请参见Threading Programming Guide.

UIApplication用一个输入源对象配置主运行循环来处理触摸事件,把它们分发给合适的响应者对象。一个响应者对象是从UIResponder 类继承而来并且实现了一个或多个处理触摸事件不同阶段的方法。应用程序中的响应者对象包括UIApplication实例,UIWindowUIView,和所有UIView子类。 应用程序通常分发事件给代表应用程序的主窗口的UIWindow 对象。这个窗口对象,依次,转发这个事件给它的第一响应者first responder,这通常是发生触摸事件的视图对象(UIView)

除了定义处理事件的方法之外,UIResponder类还定义了响应者链的程序结构,这是一个协同事件处理的Cocoa机制。响应者链是应用程序中的一个响应者对象连接序列,通常从第一响应者开始。如果第一响应者对象不能处理这个事件,它传递给响应链中的下一个。这个消息继续回溯响应链-给更高级别的响应者对象比如窗口,应用程序,和应用程序代理-直到事件被处理。如果事件最终仍然未被处理,则被抛弃。

处理事件的响应者对象倾向于在移动中设置一系列程序动作而导致应用程序重画所有或它的用户界面的一部分(以及其它可能的输出,比如播放声音)。例如,一个控制器对象 (也就是,一个UIControl子类),通过发送一个动作消息给另外一个对象来处理一个事件,通常这个控制器管理着当前激活视图集。当处理当前消息时,这个控制器可能改变用户界面或者调整视图位置,需要部分视图重新绘制自己。当这个发生时,视图和绘图基础组件接管并以可能的最有效率的方式来处理这些必要的重绘事件。

关于更多事件,响应者以及你如何在自定义对象中处理事件的信息,请参见“Event Handling.”

关于窗口和视图如何和事件处理机制融为一体,请参见“The View Interaction Model.”

关于图形基础组件以及视图如何更新的其他信息,请参见“The View Drawing Cycle.” 

基本设计模式

UIKit框架的设计融合了许多Mac OS X上的Cocoa应用程序创建的设计模式。理解这些模式对于创建iPhone应用程序是关键的,所以值得花一些时间来了解它们。下面提供了一个这些设计模式的总体描述:

1-1  iPhone应用程序使用的设计模式

设计模式

描述

模型-视图-控制器(Model-View-Controller

The Model-View-Controller (MVC) 设计模式是一种把你的应用程序划分为独立的功能块的方法。模型(model部分定义了应用程序底层数据引擎并负责维护数据完整性。视图(view)部分定义了应用程序的用户界面并且没有关于用于显示的原始数据的明确知识。控制器(controller部分作为模型和视图之间的桥梁并负责促进它们彼此之间的更新。

代理(Delegation

代理(delegation 设计模型是无需通过子类化就可以修改复杂对象的设计模式,你可以按原样使用复杂对象然后把任意的自定义修改其行为的代码放在一个单独的对象里面,也就是代理对象。在预定义时期,复杂对象调用代理对象的方法以给它运行自定义代码的机会。

目标-行为(Target-action

控件使用目标-行为(target-action设计模式来把用户交互通知给你的应用程序。当用户和控件以预定方式交互时(比如通过触击一个按钮),这个控件发送一个消息(动作)给你指定的对象(目标)。在接收这个动作消息后,目标对象然后可以用一种恰当的方式响应(比如根据按中的按钮更新应用程序状态)。

托管内存模型(Managed memory model

Objective-C 语言使用一个引用技术机制来决定什么时候从内存中释放对象。当一个对象第一次创建时,引用计数为1。其它对象可以使用这个对象的retain,release,autorelease 方法来恰当的增加和减少计数值。当一个对象引用计数为0时,Objective-C运行时调用对象的清除方法然后释放它。

关于这些设计模式的更加透彻的讨论,请参见 Cocoa Fundamentals Guide .

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


我正在用TitaniumDeveloper编写一个应用程序,它允许我使用Javascript,PHP,Ruby和Python.它为API提供了一些可能需要的标准功能,但缺少的是全局事件.现在我想将全局热键分配给我的应用程序并且几乎没有任何问题.现在我只针对MAC,但无法找到任何Python或Ruby的解决方案.我找到了Coc
我的问题是当我尝试从UIWebView中调用我的AngularJS应用程序中存在的javascript函数时,该函数无法识别.当我在典型的html结构中调用该函数时,该函数被识别为预期的.示例如下:Objective-C的:-(void)viewDidLoad{[superviewDidLoad];//CODEGOESHERE_webView.d
我想获取在我的Mac上运行的所有前台应用程序的应用程序图标.我已经使用ProcessManagerAPI迭代所有应用程序.我已经确定在processMode中设置了没有modeBackgroundOnly标志的任何进程(从GetProcessInformation()中检索)是一个“前台”应用程序,并显示在任务切换器窗口中.我只需要
我是一名PHP开发人员,我使用MVC模式和面向对象的代码.我真的想为iPhone编写应用程序,但要做到这一点我需要了解Cocoa,但要做到这一点我需要了解Objective-C2.0,但要做到这一点我需要知道C,为此我需要了解编译语言(与解释相关).我应该从哪里开始?我真的需要从简单的旧“C”开始,正
OSX中的SetTimer在Windows中是否有任何等效功能?我正在使用C.所以我正在为一些软件编写一个插件,我需要定期调用一个函数.在Windows上,我只是将函数的地址传递给SetTimer(),它将以给定的间隔调用.在OSX上有一个简单的方法吗?它应该尽可能简约.我并没有在网上找到任何不花哨的东西
我不确定引擎盖下到底发生了什么,但这是我的设置,示例代码和问题:建立:>雪豹(10.6.8)>Python2.7.2(由EPD7.1-2提供)>iPython0.11(由EPD7.1-2提供)>matplotlib(由EPD7.1-2提供)示例代码:importnumpyasnpimportpylabasplx=np.random.normal(size=(1000,))pl.plot
我正在使用FoundationFramework在Objective-C(在xCode中)编写命令行工具.我必须使用Objective-C,因为我需要取消归档以前由NSKeyedArchiver归档的对象.我的问题是,我想知道我现在是否可以在我的Linux网络服务器上使用这个编译过的应用程序.我不确定是否会出现运行时问题,或者可
使用cocoapods,我们首先了解一下rvm、gem、ruby。rvm和brew一样,但是rvm是专门管理ruby的版本控制的。rvmlistknown罗列出ruby版本rvminstall版本号   可以指定更新ruby版本而gem是包管理gemsource-l查看ruby源gemsource-rhttps://xxxxxxxx移除ruby源gemsou
我有一个包含WebView的Cocoa应用程序.由于应用程序已安装客户群,我的目标是10.4SDK.(即我不能要求Leopard.)我有两个文件:index.html和data.js.在运行时,为了响应用户输入,我通常会使用应用程序中的当前数据填充data.js文件.(data.js文件由body.html上的index.html文件用于填充
如何禁用NSMenuItem?我点击后尝试禁用NSMenuItem.操作(注销)正确处理单击.我尝试通过以下两种方式将Enabled属性更改为false:partialvoidLogout(AppKit.NSMenuItemsender){sender.Enabled=false;}和partialvoidLogout(AppKit.NSMenuItemsender){LogoutI
我在想,创建一个基本上只是一个带Web视图的界面的Cocoa应用程序是否可行?做这样的事情会有一些严重的限制吗?如果它“可行”,那是否也意味着你可以为Windows应用程序做同样的事情?解决方法:当然可以创建一个只是一个Cocoa窗口的应用程序,里面有一个Web视图.这是否可以被称为“可可应
原文链接:http://www.cnblogs.com/simonshi2012/archive/2012/10/08/2715464.htmlFrom:http://www.idev101.com/code/Cocoa/Notifications.htmlNotificationsareanincrediblyusefulwaytosendmessages(anddata)betweenobjectsthatotherwi
如果不手动编写GNUmake文件,是否存在可以理解Xcode项目的任何工具,并且可以直接针对GNUstep构建它们,从而生成Linux可执行文件,从而简化(略微)保持项目在Cocoa/Mac和GNUstep/Linux下运行所需的工作?基本上,是否有适用于Linux的xcodebuild样式应用程序?几个星期前我看了pbtomake
我正在将页面加载到WebView中.该页面有这个小测试Javascript:<scripttype="text/javascript">functiontest(parametr){$('#testspan').html(parametr);}varbdu=(function(){return{secondtest:function(parametr){$('#testspan&#039
我正在尝试使用NSAppleScript从Objective-C执行一些AppleScript…但是,我正在尝试的代码是Yosemite中用于自动化的新JavaScript.它在运行时似乎没有做任何事情,但是,正常的AppleScript工作正常.[NSAppactivateIgnoringOtherApps:YES];NSAppleScript*scriptObject=[[NSApple
链接:https://pan.baidu.com/s/14_im7AmZ2Kz3qzrqIjLlAg           vjut相关文章Python与Tkinter编程ProgrammingPython(python编程)python基础教程(第二版)深入浅出PythonPython源码剖析Python核心编程(第3版)图书信息作者:Kochan,StephenG.出
我正在实现SWTJava应用程序的OSX版本的视图,并希望在我的SWT树中使用NSOutlineView提供的“源列表”选项.我通过将此代码添加到#createHandle()方法来破解我自己的Tree.class版本来实现这一点:longNSTableViewSelectionHighlightStyleSourceList=1;longhi=OS.sel_regist
我的Cocoa应用程序需要使用easy_install在用户系统上安装Python命令行工具.理想情况下,我想将一个bash文件与我的应用程序捆绑在一起然后运行.但据我所知这是不可能的,因为软件包安装在Python的“site-packages”目录中.有没有办法创建这些文件的“包”?如果没有,我应该如何运行ea
摘要: 文章工具 收藏 投票评分 发表评论 复制链接 Swing 是设计桌面应用程序的一个功能非常强大工具包,但Swing因为曾经的不足常常遭到后人的诟病.常常听到旁人议论纷纷,”Swing 运行太慢了!”,”Swing 界面太丑嘞”,甚至就是说”Swing 简直食之无味”. 从Swing被提出到现在,已是十年光景,Swing早已不是昔日一无是处的Swing了. Chris Adamson 和我写
苹果的开发:   我对于Linux/Unix的开发也是一窍不通,只知道可以用Java.不过接触了苹果过后,确实发现,世界上确实还有那么一帮人,只用苹果,不用PC的.对于苹果的开发,我也一点都不清楚,以下是师兄们整理出来的网站. http://www.chezmark.com/osx/    共享软件精选 http://www.macosxapps.com/    分类明了,更新及时的一个重要Mac