程序入口-UIApplicationMain

Xcode4.2之前的main函数如下:

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

{

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

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

[pool release];

return retVal;

}

Xcode4.2工程中的主函数为

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

{

@autoreleasepool {

return UIApplicationMain(argc, argv, nil, NSStringFromClass([TCAppDelegate class]));

}

}

可以看出一个重要的变化是在4.2使用了ARC技术后,NSAutoreleasePool被废弃,改用@autoreleasepool ,这里请不要该回原先的方式,如果改变后,在开启ARC选项后,程序将不能通过编译。

不论那个版本,UIApplicationMain函数都是程序的关键点,下面是对这个函数的分析:

UIApplicationMain()函数是初始化程序的核心,它接受4个参数。

argc和argv:来自于main()接受的两个参数;


第三个参数:主要类(principal class),必须是UIApplication或其子类的名字,它代表着当前iPhone程序本身,这个程序会去读info.plist文件获取配置信息,其中包括主nib文件的值,一般为MainWindow(.xib);如果该参数为nil,则默认为@"UIApplication";

第四个参数:代理类(delegate class),MainWindow.xib文件中遵循UIApplicationDelegate的类的类名,因 为UIApplication定义了一个delegte变量,这个变量应该遵循UIApplicationDelegate,负责控制程序的运行,如果主 nib文件没有这个类,你应该自定义一个这样的类,并将第四个参数改为这个类的类名,否则这个程序不知道如何进行运作,因为前三个参数代表应用程序本身, 它除了把应用的事件循环启动起来,并读取info.plist里的配置信息,不做其它任何事情。如果该参数为nil,则程序假设程序的代理来自Main nib文件。

根据上面的分析,我们来看以下iOS程序的声明周期

对于UIApplicationMain函数中的第四个参数,我们也可以看出新旧版本的不同,我们建议在原先的工程中使用新的版本,以提高程序的速度,共修改如下几处

假如你的工程类都是以TC开头。

1.import你的appdelegate类,并修改第四个参数如下:

UIApplicationMain(argc, argv, nil, NSStringFromClass([TCAppDelegate class]));

2.删除MainWindow.xib文件

3.在工程的Info.plist文件中删除下面一行

4.在TCAppDelegate.m文件中,修改

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions函数,如下:

self.window = [[UIWindow alloc] initWithFrame:[[UIScreen mainScreen] bounds]];

    // Override point for customization after application launch.

    self.viewController = [[TCViewController alloc] initWithNibName:@"TCViewController" bundle:nil];


self.window.rootViewController = self.viewController;

[self.window makeKeyAndVisible];

return YES;

其中加粗部分为新增部分。

经过上面的修改,我们就可以在程序load的过程中,省去加载MainWindow.xib文件,提高程序的速度。

iPhone应用程序是由主函数main启动,它负责调用UIApplicationMain函数,该函数的形式如下所示: 
 int UIApplicationMain ( 
 int argc, 
 char *argv[], 
 NSString *principalClassName, 
 NSString *delegateClassName 
 ); 
 那么UIApplicationMain函数到底做了哪些事情呢?这个函数主要负责三件 事情:

1)从给定的类名初始化应用程序对象,也就是初始化UIApplication或者子类对象的一个实例,如果你在这里给定的是nil,那么 系统会默认UIApplication类,也就主要是这个类来控制以及协调应用程序的运行。在后续的工作中,你可以用静态方法 sharedApplication 来获取应用程序的句柄。

2)从给定的应用程序委托类,初始化一个应用程序委托。并把该委托设置为应用程序的委托,这里就有如果传入参数为nil,会调用函数访问 Info.plist文件来寻找主nib文件,获取应用程序委托。

3)启动主事件循环,并开始接收事件。

上面是UIApplicationMain函数的工作,接下来一个问题是应用程序视图的显示、消息的控制怎么办?下面就是UIApplication(或 者子类)对象的职责,这个对象主要做下面几件事:

1)负责处理到来的用户事件,并分发事件消息到应该处理该消息的目标对象(sender,  action)。 
 2)管理以及控制视图,包括呈现、控制行为、当前显示视图等。 
 3)该对象有一个应用程序委托对象,当一些生命周期内重要事件(可以包括系统事件或者生命周期控制事件)发生时,应用程序通知该对象。例如,应用程序启 动、内存不够了或者应用程序结束等,让这些事件发生时,应用程序委托去响应。

通 过上面的分析,可以知道UIApplication对开发者来说,是一个黑箱,它也可以是。因为所有的操作,都可以由它的委托来帮我们完成,它只需要在 后面维护一些不可更改的东西,如事件消息分发和传递、给委托发送事件处理请求等等,如,应用程序加载处理完毕,它会发送消息给委托,然后委托可以在 applicationDidFinishLanching委托函数中去实现开发者想要的动作。利用XCODE在创建应用程序时,会默认实现一个应用程序 委托类。而对于加载的视图,则有视图相关的委托类来处理视图加载过程的生命事件。下面说明委托主要可以办哪些事情: 
 控制应用程序的行为

- (void)applicationDidFinishLaunching:(UIApplication *)application 
           应用程序启动完毕。 
 - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
         当由于其它方法打开应用程序(如URL指定或者连接),通知委托启动完毕 
 - (void)applicationWillTerminate:(UIApplication *)application 
          通知委托,应用程序将在关闭 退出,请做一些清理工作。 
 - (void)applicationDidReceiveMemoryWarning:(UIApplication *)application 
         通知委托,应用程序收到了为来自系统的内存不足警告。-(void)applicationSignificantTimeChange:(UIApplication *)application 
       通知委托系统时间发生改变(主要是指时间属性,而不是具体的时间值) 
 打开URL 
 - (BOOL)application:(UIApplication *)application handleOpenURL:(NSURL *)url 
            打开指定的URL 
 控制状态栏方位变化 
 – application:willChangeStatusBarOrientation:duration: 
         设备方向将要发生改变 
 – application:didChangeStatusBarOrientation: 
 活动状态改变 
 - (void)applicationWillResignActive:(UIApplication *)application 
    通知委托应用程序将进入非活动状态,在此期间,应用程序不接收消息或事件。-(void)applicationDidBecomeActive:(UIApplication *)application 
       通知委托应用程序进入活动状态,请恢复数据

1.设置icon上的数字图标
 
    //设置主界面icon上的数字图标,在2.0中引进, 缺省为0
     [UIApplicationsharedApplication].applicationIconBadgeNumber = 4;
 
2.设置摇动手势的时候,是否支持redo,undo操作
 
   //摇动手势,是否支持redo undo操作。
    //3.0以后引进,缺省YES
     [UIApplicationsharedApplication].applicationSupportsShakeToEdit =YES;
 
3.判断程序运行状态
 
    //判断程序运行状态,在2.0以后引入
    
   if([UIApplicationsharedApplication].applicationState ==UIApplicationStateInactive){
         NSLog(@"程序在运行状态");
     }
 
4.阻止屏幕变暗进入休眠状态
 
   //阻止屏幕变暗,慎重使用,缺省为no 2.0
     [UIApplicationsharedApplication].idleTimerDisabled =YES;
 
慎重使用本功能,因为非常耗电。
 
5.显示联网状态
 
    //显示联网标记 2.0
     [UIApplicationsharedApplication].networkActivityIndicatorVisible =YES;
 
6.在map上显示一个地址
 
   NSString* addressText [email protected]"1 Infinite Loop, Cupertino, CA 95014";
    // URL encode the spaces
     addressText =  [addressTextstringByAddingPercentEscapesUsingEncoding:NSASCIIStringEncoding];
    NSString* urlText = [NSStringstringWithFormat:@"http://maps.google.com/maps?q=%@", addressText];
    
    [[UIApplicationsharedApplication]openURL:[NSURLURLWithString:urlText]];
 
7.发送电子邮件
 
   NSString *recipients [email protected]"mailto:[email protected][email protected],[email protected]&subject=Hello from California!";
    NSString *body [email protected]"&body=It is raining in sunny California!";
    
    NSString *email = [NSStringstringWithFormat:@"%@%@", recipients, body];
     email = [emailstringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding];
    
    [[UIApplicationsharedApplication]openURL:[NSURLURLWithString:email]];
 
8.打电话到一个号码
 
   // Call Google 411
     [[UIApplicationsharedApplication]openURL:[NSURLURLWithString:@"tel://8004664411"]];
 
9.发送短信 
    // Text to Google SMS
     [[UIApplicationsharedApplication]openURL:[NSURLURLWithString:@"sms://466453"]];
 
10.打开一个网址
 
   // Lanuch any iPhone developers fav site
     [[UIApplicationsharedApplication]openURL:[NSURLURLWithString:@"http://itunesconnect.apple.com"]];

可以看到UIApplication的头文件实现 
 @interface UIApplication :UIResponder <UIActionSheetDelegate>{ 
 @package 
 id<UIApplicationDelegate> _delegate ;  //这就是应用程序委托。 
 NSTimer ....... 
 } 
 因此,在UIApplication中处理的系统事件时,只需转到_delegate这个类去处理, 这个类对象就是应用程序委托对象。我们可以从应用程序的单例类对象中得到应用程序委托的对象 
 UIApplicationDelegate* myDelegate = [[UIApplication sharedApplication] delegate]; 
 
UIApplication 接收到所有的系统事件和生命周期事件时,都会把事件传递给UIApplicationDelegate进行处理,对于用户输入 事件,则传递给相应的目标对象去处理。比如我们在应用程序被来电等消息后,可以调用应用程序委托类的 applicationWillResignActive()方法,这个方法在用户锁住屏幕时,也会调用,与之相适应的是应用程序重新被用户打开时的委托 方法。另外常用的就是内存不足的系统警告,此时会调用应用程序委托类的applicationDidReceiveMemoryWarning()方法, 然后我们就可以试着释放一些内存了。 
 
上面就是应用程序生命周期(启动,中止,恢复,退出等过程)的应用程序处理UIApplication sharedApplication

时间: 2024-10-24 06:46:53

程序入口-UIApplicationMain的相关文章

如何确定android中的程序入口?

1.如何确定android中的程序入口? 需要在AndroidMainifest.xml文件中添加这个Activity的配置信息,同时将DemoActivity配置中的intent-filter移动到LauncherActivity配置中. <activity android:name=".DemoActivity" android:label="@string/app_name" > </activity> <activity and

开源GUI-Microwindows之程序入口分析

*************************************************************************************************************************** 作者:EasyWave                                                               时间:2014.10.05 类别:开源GUI系统-Microwindows之程序入口分析      声明

Egret解疑释惑之程序入口

关于本教程 本教程是连续的,记录作者在使用Egret的心得体会,并把它们罗列成文,帮助有需要的朋友. 该系列是Egret解疑释惑,收集.发布一些关于Egret的使用技巧,中间会有解答初学者的疑惑,还会讲解到一些例子,当然也会有一定高度的深入探讨. 另外,文章中所介绍的技巧.方法.解决方案,都不是唯一的,可能会有两种,甚至多种途径.因为本系列中,为了不增加读者需要更多的知识量来阅读,会讲得比较直白,通俗一点. 同时为了保证让初学者简单.明了,文章中也不会太过于涉及与文章主题以外的知识(除非作者觉得

Hadoop源码分析—— Job任务的程序入口

这篇文章大致介绍Hadoop Job的程序是如何启动的. 通常用Java编写的Hadoop MapReduce程序是通过一个main方法作为程序的整个入口,如下: public static void main(String[] args) throws Exception { int res = ToolRunner.run(new Configuration(), new CalculateSumJob(),args); System.exit(res);} 可以看到这个Job任务的MapR

MFC 程序入口和执行流程

一 MFC程序执行过程剖析 1)我们知道在WIN32API程序当中,程序的入口为WinMain函数,在这个函数当中我们完成注册窗口类,创建窗口,进入消息循环,最后由操作系统根据发送到程序窗口的消息调用程序的窗口函数.而在MFC程序当中我们不在能找到类似WinMain这样的程序入口,取而代之的是一系列派生类的声明和定义以及一个冲CWinApp类派生而来的类的全局对象.CWinApp类被称之为应用程序对象,在一个MFC程序当中只允许有一个应用程序对象.由于CWinApp的派生对象是全局的,因此这个对

[iOS] 使用xib做为应用程序入口 with Code

[iOS] 使用xib做为应用程序入口 with Code 前言 开发iOS APP的时候,使用storyboard能够快速并且直觉的建立用户界面.但在多人团队开发的情景中,因为storyboard是以单一档案的方式存在,很容易造成签出.签入时,档案被锁定.档案合并冲突等等问题的发生.这时开发人员可以选择使用xib做为用户接口的开发单位,将用户接口拆散为独立存在的xib档案,分散团队成员同时编辑同一文件的风险. 在Xcode中默认是以storyboard做为应用程序入口,变更为使用xib做为应用

微信小程序再添能力:搜一搜增加小程序入口

随着移动互联网的快速发展,流量正悄然的变成大数据,借助"入口+流量"模式,互联网激活了大量的个性化入口,积累了巨量流量,并形成高企的资本市场估值.小程序作为当前最新的应用模式,在微信不断开放入口之后,或将迎来流量的爆发期. 今天,笔者发现,微信又为小程序开放了一个超级入口:微信搜一搜增添小程序入口.用户在微信发现中,通过搜一搜 输入关键词,比如水果,搜索页面便会出现相关小程序推荐,点击更多,可直接进入小程序"水果"的搜索页面,相关小程序都会出现在页面中. 点击微信搜

[iOS] 使用xib作为应用程序入口 with IDE

[iOS] 使用xib作为应用程序入口 with IDE 在「使用xib做为应用程序入口 with Code」这篇文章中,介绍了如何透过写Code的方式,来使用xib做为应用程序的入口.但其实在Xcode中,透过IDE所提供的工具,也可以达到相同的功能.本篇文章说明在iOS APP开发的过程中,如何透过IDE的操作,来使用xib做为应用程序的入口,为自己留个纪录也希望能帮助到有需要的开发人员.(透过IDE操作的方式,会比使用Code的方式复杂许多.但是在操作的过程中,可以理解xib档案的操作与概

【REACT NATIVE 系列教程之七】统一ANDROID与IOS两个平台的程序入口&&区分平台的组件简介

本站文章均为 李华明Himi 原创,转载务必在明显处注明: 转载自[黑米GameDev街区] 原文链接: http://www.himigame.com/react-native/2260.html       本篇介绍两个细节:       1. 关于如何将index.android.js 与index.ios.js统一管理起来.       2.  Platform 组件的简单介绍与使用   一:将index.android.js 与index.ios.js统一管理起来. 由于React本身

Swift程序入口深度分析

1.swift为什么不需要main 在c/c++及其它语言中都有一个main函数,程序从main作为起点,开始执行程序,如下: int main(int argc, const char * argv[]) { printf("Hello, World!\n"); return 0; } main函数实际上是一个特殊的函数,为了能找到程序入口,大多楼语言都约定main()函数作为入口.那么为什么在Swift中没有这样的一个函数呢?先看一下官方的解释 Code written at gl