注:本文同步发布于微信公众号:stringwu的互联网杂谈 Android开发快速入门iOS开发概览
1 前言
笔者总结了自己在拥有Android
开发的相关基础后入门iOS
开发时遇到的点点滴滴给其他想入门iOS
开发的Android
开发的一些参考,少走一些弯路,快速上手iOS
开发;
文章会以iOS
和 Android
的开发工具,语言,工程文件和启动类的对比为主线,一步步带你了解 iOS
版本的 HelloWord
是如何编写的;
两者的主要对比概览
平台 | IDE | 语言 | 依赖管理 | UI主界面 | 应用入口 |
---|---|---|---|---|---|
Android | Android Studio | java(kotlin) | maven | Activity | Application |
iOS | Xcode | object-c(swift) | Cocoapods | Controller | AppDelegate |
2 工具篇
iOS
的开发工具 Xcode
可直接在线获取并安装,如果仅是简单的开发一个HelloWorld
工程,则可直接新建个工程运行就可。但如果需要使用其他第三方库时,则离不开Cocoapods
工具了;
2.1 Cocoapods
Cocoapods
就是iOS
开发中的maven
,主要用于集成管理第三方的依赖库。本文仅简单介绍在Mac
下安装 Cocoapods
最快捷的安装方式 gem install cocoapods
,详细内容可参考文档cocoapods安装文档 ;
安装完成后在命令行运行:pod --version
查看本地安装的版本就可以,几个常用的pods
命令,这几个命令在开发中的使用概率非常大
命令 | 作用 |
---|---|
pods init | 创建podfile文件 |
pods install | 根据pofile.lock指定的版本去拉对应的依赖库 |
pods update | 更新依赖库(平时开发一般使用这个命令就可) |
备注:podfile
类似于Android开发中在 build.gradle
中指定第三方库的依赖和对应的版本的文件;
podfile
的文件内容一般为:
#指定 pod的依赖来源方
source 'https://github.com/CocoaPods/Specs.git'
# 指定依赖的第三方库
pod `AFNetworking`,'4.0.1'
# 指定依赖的第三方库,大于某个版本
pod '***', '~> 1.8.4.0'
3 语言篇
iOS
开发一般使用oc
或 swift
来开发,需要注意的是iOS
没有Android
里包的概念,类的名字必须保持全局唯一,一般约定是添加特定的前缀(公司 + 项目)如WXGPUImage
下面会以oc
为例来说明
3.1 基础语法
- 头文件(.h)里定义的方法和变量都是公共的,源文件(.m)文件里定义的方法与变量是私有的;
property
声明一个变量属性;- 类的定义从
@interface
开始,到@end
结束; - 类的实现则是从
@implementation
到@end
结束;
对象初始化
- [xxx new];
- [[xxx alloc]init]; //推荐使用
对象可变性:object-c里对象可变(类似于Java 里的final)和不可变时使用时使用的类是不一样的,以String为例子: - NSString :不可变;
- NSMutableString : 可变的;
NSString *saveKey = [NSString stringWithString:host];
[saveKey appendString:@"|"]; //非法
NSMutableString *saveKey = [NSMutableString stringWithString:@"123"];
[saveKey appendString:@"|"]; //合法
变量时的原子性(线程安全):
- atomic 默认的,只保证值有效,不保证这个值是什么;
- nonatomic,不保证你读到的是什么值;
变量的内存相关:
- weak,弱引用,不对所赋值对象进行持有,但是是安全的,对象不可用时,会被置为nil;
- strong,对新对象进行强引用,释放旧对象,其引用计数+1,用在ARC中,用于对象或指针类型的数据类型
- retain 对新对象进行强引用,释放旧对象,其引用计数+1,用在MRC中;
- assign,直接赋值,和引用计数无关,用于声明基本类型,如int;
- copy,在实现Setter方法时,采用copy函数,会生成新的对象被自己持有,一般用来修饰 NSString;
- unsafe_unretained,弱引用,和weak不同的是,若引用对象不可用,当前指针不会被置为nil,会产生野指针;
@interface WSDemoObject : NSObject
//定义一个属性 @property(原子性,内存性)
@property (nonatomic, assign) NSInteger index;
// 构造方法
- (instancetype)initWithSymbolString:(NSString *)symbol;
// 静态方法
+ (void)callSomething;
@end
3.2 文件类型
在iOS
中新建一个文件(类)时,文件类型一般会有以下四种类型:
- protocol,协议,类似于java的接口 (并且协议里方法可以选择是否是必须的,如果不是必须的,可使用关键字
optional
进行标识); - Extention,扩展 ,对某个类的功能进行扩展,需要拥有源码,生效于编译期;
- category,分类,也是对类的功能进行扩展,生效于运行期(可用于扩展系统类,只能扩展方法);
- 其他,正常的类文件;
//定义一个protocol
@protocol NsCopying
- (id) copyWithZone : (NSZone *) zone
@end
//采用协议,类似于Java的实现接口
@interface Car :NSObject<NsCopying,NSCoding>
{ // something
}
@end
NS_ASSUME_NONNULL_BEGIN
//Category,扩展AppDelegate的方法
@interface AppDelegate (WSTest)
- (id) copyWithZone : (NSZone *) zone
@end
NS_ASSUME_NONNULL_END
NS_ASSUME_NONNULL_BEGIN
//Extension,扩展AppDelegate的方法,小括号里无内容
@interface AppDelegate ()
- (id) copyWithZone : (NSZone *) zone
@end
NS_ASSUME_NONNULL_END
3.3 单例
使用object-c
来编写一个单元测试的用例:
//类定义
@interface Singleton : NSObject
+(instancetype) shareInstance;
@end
//类实现
# import "Singleton.h"
@implementation Singleton
+(instancetype) shareInstance
{
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
_instance = [[self alloc] init];
});
return _instance;
}
4 iOS篇
4.1 生命周期
iOS
应用的生命周期的管理主要是通过AppDelegate
来完成的,生命周期方法主要有:
-
applicationWillEnterForground ; //应用即将进入前台
-
applicationDidBecomeActive; //应用变成活跃
-
applicationWillResignActive;
-
applicationDidEnterBackground ;//应用即将进入后台
-
applicationWillTerminate ; // 应用程序即将终止的回调;
iOS
中页面是由Controller
构造出来的,类似于Android
的 Activity
,其关键的生命周期为:
- loadView : 加载View(一般使用xib构建时)
- viewDidLoad: View 加载完毕 ;//类似于Activity.onCreate
- viewWillAppear: View将要显示;//类似于Activity.onStart
- viewDidAppear :View完全显示;//类似于Activity.onResume
- viewWillDisappear:View将到消失;//类似于Activity.onPause
- viewDidDisappear:View完全消失;//类似于Activity.onDestory
4.2 工程文件
iOS
的工程是通过配置把项目的各个文件管理起来的,每个文件都会有一个唯一编号。新增代码文件时,IDE会自动在工程文件project.pbxproj
里给代码文件增加(修改)其对应编号。一般情况下是不需要手动去编辑project.pbxproj
文件,通过Xcode的选项buildsetting
、build Phases
就可修改project.pbxproj
了。但如果遇到协同开发,代码有冲突时,则需要直面该文件,并谨慎修改它。
project.phxproj 文件
- project.pbxproj
- info.plist 工程描述文件
PS:这里为什么会把工程文件单独拉出来呢?协同开发时,解冲突最麻烦了,而iOS
里的工程文件冲突最不好解了,稍微不注意就会解错导致编译不了
4.3 第三方依赖
在Android
工程中,有两种方式来依赖第三方的库:一种是通过直接把第三方库(.aar
,.jar
)放到library
目录,并在工程的gradle
文件里指定要编译的library
目录下的文件,另一种就是直接在gradle文件里写要依赖的第三方库,如
dependencies{
implementation 'com.tencent.qapm:abc:1.5.4' // 依赖abc库的 1.5.4版本
}
类似的,在iOS
工程里也有两种依赖第三方库的方法:一种是直接直接把第三方库(.a
,.framework
)放到工程目录中,并在buildsetting
里引用这些库,另一种就是使用pod
依赖了;
有一个点需要注意:在集成第三方库时,要特别留意符号冲突(因为iOS是可以对类进行扩展的,如果有两个库同时对一个公共类进行了扩展,并且扩展的方法是一样的,则有可能会发生运行时错误),一般建议如果是对公共的类或库进行扩展时,扩展的方法名也带上前缀,这样可以避免很多坑;
iOS
中的第三方库有两种形态:
- 静态库,后缀为(.a,.framework)
- 动态库,后缀为(.dylib,.framework)
静态库与动态库的区别主要在于在打包链接时的操作不一样;
4.3.1 静态库
静态库在打包生成可执行的目标文件时,会把汇编生成的目标文件.o
与引用的库一起链接打包到目标文件中,静态库的特点:
- 在编译期完成对函数库的链接
- 运行时与函数库不再有关,可独立运行;
- 会比较占用空间(如果有函数库被多个静态库依赖,则这个函数库里会有多份拷贝)
- 更新麻烦,如果静态库A更新了,则所有依赖静态库A的库都需要重新编译,更新;
- 加载速度较动态库快
4.3.2 动态库
动态库与静态库类似,只是在打包时不用把所链接的文件全部拷贝进目标文件中,只是拷贝一些重定位和符号信息,这些信息在运行时可完成真正的链接,动态库的特点:
- 在运行时完成对函数库的链接(不同应用程序调用相同的库,在内存里只需要有一份该库的实例);
- 运行时与函数库有关,需要依赖对应的函数库才可运行;
- 会比较省空间(只需要拷贝少量的符号与链接信息)
- 更新简单,只需要更新对应的动态库就可以了;
- 可以用来实现进程之间的资源共享;
- 加载速度较静态库慢;
5 View篇
这里主要列举几个常用的UI控件
Android | iOS |
---|---|
View(ViewGroup) | UIView |
TextView | UILabel |
Button | UIbutton |
ImageView | UIImageView |
ListView(RecyclerView) | UITableView(UICollectionView) |
ScrollView | UIScrollView |
这五个主要View可以覆盖到80%的业务需求了,具体的用法可在使用到时自行搜索,这里就不再一一列举。
在创建一个Helloword
的 Demo时可直接利用Xcode创建项目的选项,一步步按照操作进行就可以了;虽然iOS
也可以在.storyboard
文件里进行界面的编写,但绝大部分时候都是需要使用代码来创建界面的,主要是在创建好的ViewController
的 viewDidLoad
方法里加入View来显示内容,eg:
//ps:代码是swift的,oc的代码也是类似的,只是语法有稍微的区别
override func viewDidLoad() {
super.viewDidLoad()
// Do any additional setup after loading the view.
let copyButton = UIButton(type:.custom)
copyButton.setTitle("Hello World", for: .normal)
//设置颜色
copyButton.titleLabel?.font = UIFont.boldSystemFont(ofSize: 13)
copyButton.backgroundColor = UIColor.blue
//设置大小(边界),关键(告诉系统在哪里绘制这个View)
copyButton.frame = CGRect.init(x: 0, y: 100, 240, height:240)
//addTarget方法用来设置一些事件的处理,类似于Android的 addClickListener
// action 用来响应该事件的方法
// for : 要监听处理哪些事件
copyButton.addTarget(self, action: #selector(defaultButtonPressed), for: UIControl.Event.touchUpInside)
//把View加入到界面中
self.view.addSubview(copyButton)
}