iOS完全自学手册——[二]Hello World工程

1.前言

写第二篇文章之前,我在想第二篇应该怎么写?后来觉得与其写Objective-C语言的相关语法,不如直接开始写个小项目。语法简单入门即可。因为,即便把语法看的很深到最后还是不一定能做项目,运用到项目中去。那就从HelloWorld开始吧。

2.​学习目标

1)用Storyboard和纯代码实现Hello World简单应用,了解iOS项目的创建流程。

2)了解 AppDelegate 作用

3)了解 View Controller 作用

4) Storyboard、xib、代码搭建应用UI的对比

3.创建HelloWorld工程

1)用storyboard创建

为了方便理解,我懒得截图。录了一段视频,可以看一下。

2)用纯代码创建

I)修改info.plist文件

创建工程的过程和用storyboard相近,工程创建完成之后,需要修改info.plist文件,把Main storyboard file base name属性去掉。

II)修改AppDelegate.m文件。

引入View Controller的头文件

#import "ViewController.h"

并修改- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions 方法。

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
    // Override point for customization after application launch.
    self.window = [[UIWindow alloc] initWithFrame:[UIScreen mainScreen].bounds];
    self.window.rootViewController = [[ViewController alloc] init];
    self.window.backgroundColor = [UIColor whiteColor];
    [self.window makeKeyAndVisible];
    return YES;
}

III)修改ViewController.m文件

#import "ViewController.h"

#define HELLO_EN @"Hello World"
#define HELLO_ZH @"你好世界"

@interface ViewController ()

@property (strong, nonatomic) UILabel *label;
@property (strong, nonatomic) UIButton *button;

@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    // Do any additional setup after loading the view, typically from a nib.
    self.label = [[UILabel alloc] initWithFrame:CGRectMake(10, 50, 120, 20)];
    self.label.text = HELLO_EN;
    [self.view addSubview:self.label];
    
    self.button = [[UIButton alloc] initWithFrame:CGRectMake(10, 80, 120, 30)];
    [self.button setTitle:@"点我" forState:UIControlStateNormal];
    [self.button setTitleColor:[UIColor blueColor] forState:UIControlStateNormal];
    [self.button addTarget:self action:@selector(tapButton) forControlEvents:UIControlEventTouchUpInside];
    [self.view addSubview:self.button];
}

- (void)didReceiveMemoryWarning {
    [super didReceiveMemoryWarning];
    // Dispose of any resources that can be recreated.
}

- (void)tapButton {
    self.label.text = ([self.label.text isEqualToString:HELLO_EN])?HELLO_ZH:HELLO_EN;
}

@end

3)xib

另,用xib创建工程的方法见http://www.cnblogs.com/nycoder/p/4333879.html

4.了解AppDelegate

1)AppDelegate的作用——管理生命周期

应用从开始运行到被用户/系统终止都有其周期,那应用具体的生命周期是怎样的呢?

我们可以先分析程序的main.m 和 AppDelegate.h的代码。

main.m文件中代码:

#import <UIKit/UIKit.h>
#import "AppDelegate.h"

int main(int argc, char * argv[]) {
    @autoreleasepool {
        return UIApplicationMain(argc, argv, nil, NSStringFromClass([AppDelegate class]));
    }
}

这个程序文件中:

i)引入了UIKit库

ii)引入AppDelegate.h文件

iii)应用程序单一入口main函数,其中main函数的自动释放池中调用了UIApplicationMain方法,其中参数列表的第四个参数为NSStringFromClass([AppDelegate class]),查看UIApplicationMain的文档(鼠标指到方法名上,按住option键,在弹出的视图中找到UIKit Function Reference链接,点开即可

The name of the class from which the application delegate is instantiated. If principalClassName designates a subclass of UIApplication, you may designate the subclass as the delegate; the subclass instance receives the application-delegate messages. Specify nil if you load the delegate object from your application’s main nib file.

即可看到这个参数的描述,这个参数就是实例化的应用代理,即AppDelegate类的实例。并且如果用UIApplication子类的子类做参数,那须指定该子类为delegate。

再看AppDelegate.h文件

#import <UIKit/UIKit.h>

@interface AppDelegate : UIResponder <UIApplicationDelegate>

@property (strong, nonatomic) UIWindow *window;

@end

可以观察到这段代码:

i)引入了UIKit库文件

ii)声明了AppDelegate的类,并且继承自UIResponder,且遵循UIApplicationDelegate协议

iii)声明了一个属性window

打开苹果官方文档(command+shift+0)查找UIApplicationDelegate可以看到下面的叙述。

The UIApplicationDelegate protocol defines methods that are called by the singleton UIApplication object in response to important events in the lifetime of your app.

The app delegate works alongside the app object to ensure your app interacts properly with the system and with other apps. Specifically, the methods of the app delegate give you a chance to respond to important changes. For example, you use the methods of the app delegate to respond to state transitions, such as when your app moves from foreground to background execution, and to respond to incoming notifications. In many cases, the methods of the app delegate are the only way to receive these important notifications.

主旨大意是,简述了App Delegate单例在App的生命周期中起到的作用——为开发者提供App不同运行状态的方法,让应用可以协调的跟系统其他的应用一同正常运行。

也就是,AppDelegate起到了管理应用的生命周期的作用,文档为开发者提供了应用的五个状态表格:

Table 1 App states

State

Description

Not running

The app has not been launched or was terminated, either by the user or the system.

Inactive

The app is running in the foreground but is not receiving events. (It may be executing other code though.) An app usually stays in this state only briefly as it transitions to a different state.

Upon entering this state, the app should put itself into a quiescent state with the expectation of moving to the background or active state shortly.

Active

The app is running in the foreground and receiving events. This is the normal mode for foreground apps.

An app in the active state has no special restrictions placed on it. It is the foreground app and should be responsive to the user.

Background

The app is executing code but is not visible onscreen. When the user quits an app, the system moves the app to the background state briefly before suspending it. At other times, the system may launch the app into the background (or wake up a suspended app) and give it time to handle specific tasks. For example, the system may wake up an app so that it can process background downloads, certain types of location events, remote notifications, and other types of events.

An app in the background state should do as little work as possible. Apps that request time to process specific types of events should process those events and return control back to the system as quickly as possible.

Suspended

The app is in memory but is not executing code. The system suspends apps that are in the background and do not have any pending tasks to complete. The system may purge suspended apps at any time without waking them up to make room for other apps.

 从上表中可以看到,应用共不运行、活动的、不活动的、后台运行、挂起五个状态。

其中这五个状态的转变如图所示

 

而对应这五个状态,有相应的7个代理协议的方法,可以打开AppDelegate.m文件

#import "AppDelegate.h"
#import "ViewController.h"
@interface AppDelegate ()

@end

@implementation AppDelegate

- (BOOL)application:(UIApplication *)application willFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
//这个方法是我自己加的,默认创建的时候没有
    return YES;
}
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
    // Override point for customization after application launch.
    self.window = [[UIWindow alloc] initWithFrame:[UIScreen mainScreen].bounds];
    self.window.rootViewController = [[ViewController alloc] init];
    self.window.backgroundColor = [UIColor whiteColor];
    [self.window makeKeyAndVisible];
    return YES;
}

- (void)applicationWillResignActive:(UIApplication *)application {
    // Sent when the application is about to move from active to inactive state. This can occur for certain types of temporary interruptions (such as an incoming phone call or SMS message) or when the user quits the application and it begins the transition to the background state.
    // Use this method to pause ongoing tasks, disable timers, and throttle down OpenGL ES frame rates. Games should use this method to pause the game.
}

- (void)applicationDidEnterBackground:(UIApplication *)application {
    // Use this method to release shared resources, save user data, invalidate timers, and store enough application state information to restore your application to its current state in case it is terminated later.
    // If your application supports background execution, this method is called instead of applicationWillTerminate: when the user quits.
}

- (void)applicationWillEnterForeground:(UIApplication *)application {
    // Called as part of the transition from the background to the inactive state; here you can undo many of the changes made on entering the background.
}

- (void)applicationDidBecomeActive:(UIApplication *)application {
    // Restart any tasks that were paused (or not yet started) while the application was inactive. If the application was previously in the background, optionally refresh the user interface.
}

- (void)applicationWillTerminate:(UIApplication *)application {
    // Called when the application is about to terminate. Save data if appropriate. See also applicationDidEnterBackground:.
    // Saves changes in the application's managed object context before the application terminates.
   
}
@end

 

从代码的注释中就可以看到这些方法对应的应用生命周期的阶段。

  • 应用开始运行
    • application:willFinishLaunchingWithOptions: 
    • application:didFinishLaunchingWithOptions: 
  • 应用进入系统前台
    • applicationDidBecomeActive: 
  • 应用进入系统后台
    • applicationDidEnterBackground: 
  • 应用进入不活动的状态 
    • applicationWillResignActive: (Called when leaving the foreground state.) 
    • applicationWillEnterForeground: (Called when transitioning out of the background state.) 
  • 应用终止
    • applicationWillTerminate: (Called only when the app is running. This method is not called if the app is suspended.)

 

2)根视图及根视图控制器

可以发现不论是创建storyboard的工程,还是创建仅有代码的工程,都会有根视图控制器这个概念。那什么是根视图控制器呢?继续回到AppDelegate的m文件,将鼠标移动到self.window的window上,点开其对应的文档。苹果文档如下:

Providing a Window for Storyboarding
window
 Required Property
The window to use when presenting a storyboard.

Declaration
SWIFT
optional var window: UIWindow? { get set }
OBJECTIVE-C
@property(nonatomic, strong, nullable) UIWindow *window
Discussion
This property contains the window used to present the app’s visual content on the device’s main screen.

文档中简述了作用,为应用提供可视的视图容器。所有自定义的视图控制器都要在该window中显示,成为该视图的子视图。另外,查看UIWindow的苹果文档可以发现

UIWindow 继承自 UIView : UIResponder : NSObject 。即,window实例也是一个视图。

The UIWindow class defines an object known as a window that manages and coordinates the views an app displays on a device screen. Unless an app can display content on an external device screen, an app has only one window.

此段文档说明了window的作用:管理视图及定位视图的坐标,以在设备屏幕上显示。除非一个应用可以在外置的屏幕上显示内容,一个应用仅有一个window。

而window又声明了rootViewController属性。

rootViewController
 Property
The root view controller for the window.

Declaration
SWIFT
var rootViewController: UIViewController?
OBJECTIVE-C
@property(nonatomic, strong, nullable) UIViewController *rootViewController
Discussion
The root view controller provides the content view of the window. Assigning a view controller to this property (either programmatically or using Interface Builder) installs the view controller’s view as the content view of the window. If the window has an existing view hierarchy, the old views are removed before the new ones are installed.

这段文档说明rootViewController,为window提供了内容视图,并且,被分配的视图控制器的视图会装载到window中。例如,上面用代码创建的项目中

self.window.rootViewController = [[ViewController alloc] init];

这句代码就是将应用的根视图的根视图控制器指定为,ViewController类型的实例,那么该视图控制器的视图就装载到window中了。这样程序运行之后,手机屏幕就能显示ViewController的视图了。

那么到此,App Delegate基本介绍完了,其实App Delegate除了上面7个应用生命周期的协议方法以外还有一些比较常用的方法。我补充一下。更多方法还是需要自己查阅官方文档。

3)其他常用代理方法

i)Handling Local and Remote Notifications 管理本地和远程消息推送

- application: didFailToRegisterForRemoteNotificationsWithError:

- application: didRegisterForRemoteNotificationsWithDeviceToken:

- application: didReceiveRemoteNotification: fetchCompletionHandler:

以上方法用于接受苹果的消息通知,比如,收到苹果的推送。如果要对推送消息作处理,就需要实现这些代理方法。

ii)Opening a URL-Specified Resource 打开指定URL的资源

- application:openURL:options: (iOS 9)

该方法一般应用于打开第三方应用,比如分享、支付等情景。

iOS9 失效的方法:

- application:handleOpenURL:

- application:openURL:sourceApplication:annotation:

iii)Handling Memory Warnings  管理内存警告 

- applicationDidReceiveMemoryWarning: 

这个方法在我适配 512MB iPhone4 的时候用到了,它的作用时,收到系统内存警告的时候会掉用该方法。在这个方法中就可以,做一些内存管理的操作,缓解内存压力。(当然,一般内存报警都是由于图像过大造成的。所以,调试的时候看看你的图片是不是又大又多。。囧)

5.了解ViewController

刚才提到了根视图控制器,那何为视图控制器呢?其实,视图控制器,就是管理视图,以及相关业务逻辑的类,作为模型和视图的连接桥梁。iOS应用开发严格遵循软件设计模式的规范,而这里的视图控制器就是MVC三层架构中的C。

打开苹果官方文档,即可看到UIViewController的继承关系 , UIViewController 继承自 UIResponder : NSObject,“The UIViewController class provides the infrastructure for managing the views of your iOS apps.”这句话也表明了视图控制器的作用。例如,helloworld的项目中,在viewDidLoad方法中我们创建了一个label视图,和一个button视图,并且将视图添加到self.view中。

iOS开发中自定义视图控制器最需要了解的就是视图控制器的生命周期方法。

1)视图控制器的生命周期

初始化:

- init

- initWithNibName:bundle:  从xib文件初始化视图控制器

- loadView  不要重写这个方法(If you want to perform any additional initialization of your views, do so in the viewDidLoad method.

- viewDidLoad  视图加载完成,一般这个方法中创建各种视图

- viewWillAppear:     视图将要显示,一般在可以在这个方法请求数据,不要在该方法中创建视图,还可以设置观察者,但是要在disappear的时候remove观察者

- viewDidAppear:      视图已经显示

- viewWillLayoutSubviews

- viewDidLayoutSubviews 

- viewWillDisappear:    视图将要消失

- viewDidDisappear:    视图已经消失

- dealloc    销毁实例

6.小结

1)文档!文档!

文档很重要,作为学习资源,和开发的辅助工具文档使用非常频繁。刚开始转iOS开发的时候,我用文档很不习惯,原因是英语能力略渣,但是用得久了读起来就没那么吃力了,之后我也会写一篇文档使用文档的文章,分享给大家。

2)App Delegate

AppDelegate实质是应用单一入口的代理实例,管理应用的生命周期,也有其他协议方法,如,打开URL资源,管理消息通知,内存报警等。具体其他方法请自行查阅文档。AppDelegate的生命周期共不运行、活动的、不活动的、后台运行、挂起 这五个状态,而这也是整个应用的生命状态,即当应用运行在iOS操作系统中,从用户点击图标,到用户点按Home键,再到用户两次点按Home键杀掉应用的进程,应用都会经历这5个状态,并且在这5个状态之间进行切换。

(所以,插一句,这就是为什么我觉得,对于像我这种在学习iOS开发前完全没有接触过苹果的童鞋,可以买一台iPhone使用,哪怕是iOS7 512MB的iPhone4,至少体验iOS的操作系统以及iOS应用基本操作、UI风格、动画特效等,这样有助于开发)

3)View Controller 

View Controller视图控制器最根本的作用就是管理视图。之后还会接触到UITableViewController、UITabBarController、UINavigationController等其它视图控制器,但不管他们是啥控制器都继承自UIViewController类,并对要显示的视图进行管理。

其实,不论是WEB前端、Android前端、iOS前端其本质的作用就是为用户提供可视化的图形界面,说白了就是在屏幕上画画。那么开发者的任务就是利用开发工具将UI设计做好的图变成屏幕上可交互的软件,就酱~

4)Storyboard、xib、代码搭建应用UI的对比

在我刚开始学习iOS开发的时候,由于完全没有接触过iOS开发,所以对于搭建应用的UI并不清楚如何选择。在网上搜索,很多实例都是代码直接创建的,所以对于Xcode的Interface Builder使用的非常不熟练,到正式加入工作之后,才发现Interface Builder搭建项目十分方便,为开发者提供了很良好的敏捷开发、可视化开发的工具。这里,我在对比一下Storyboard、xib、代码的异同利弊。

i)Storyboard和xib的相同点

两者创建UI的方式都是在Xcode的对象库中拖拽控件,两者实质都是XML文件,可以右键点击文件,打开源码文件,就可以看到XML的源码。(我觉得比较有意思的是,在打开Storyboard的XML文档时,可以很清楚的看到视图的XML元素~一个元素对应代码的一个对象,而这个对象的属性又可以用XML元素属性来对应表示。当然,Android貌似也是这样😁)

 

ii)Storyboard和xib的不同点

 Storyboard相对于xib最大的不同是,sb把view controller也可视化了,可以让开发者看到所谓的视图控制器是怎么回事,也可以进行属性的初始化。而且,storyboard对整个项目的架构可以做到一目了然,而不是像xib只能看到视图。

iii)Storyboard(xib)和代码创建UI的不同点

这个很清楚了就是拖拽和手写代码的区别。

iv)如何选择?利弊分析 

私以为,如果是刚开始学iOS最好还是选用storyboard。因为Xcode的Interface Builder提供了很好的图形化界面,每个inspector选项版都非常清楚地将对象的属性展现给开发者。除此之外,实际开发中storyboard项目的确比单纯用代码创建项目要快太多,而且,像自定义table view的cell完全可以直接往cell的content view里拖拽控件,不需要调用注册自定义单元格什么的方法,但是必须要定义cell的属性,标识cell的identifier,也就是说对于tableview的重用机制,storyboard已经帮你注册好了。另外,再比如table view、collection view 的delegate 或者 datasource都可以直接拖拽到对应的view controller上。所以,方便快捷利于项目快速开发。此外,我认为不要过分地崇尚代码,把代码神话,觉得用Interface Builder拖拽控件就是很low的事情,如果觉得low,那是因为你不会用。

当然,如果已经有了稳定版本的产品,不需要快速开发,可以慢慢用代码重构嘛~~

7.预告&结语

下一篇将写一篇利用Objective-C创建对象的文章,将会介绍Objective-C的基础语法,Objective-C与其它语言的异同,同时,介绍Xcode和苹果文档的使用方法。

感谢您的阅读,如有错误,敬请批评指正,感谢~~😁

posted @ 2016-01-02 20:07  nycode  阅读(1905)  评论(0编辑  收藏  举报