Android和iOS是移动端的两大平台,Android以它的开源、易上手、开发成本低而受到广大开发者的青睐,而iOS作为苹果的封闭系统,以它的简单、流畅高效、高冷等特点也吸引了一大批果粉开发者,笔者在学校进行了近2年的Android 开发,现在因为工作原因转到了iOS,刚好学习研究iOS。目前学习了iOS刚好一个月,有一些心得体会,遂对比Android,并根据一些资料结合自己的理解进行了总结,力求做到全面。因为笔者对Android熟悉一点,所以笔者某些方面是站在Android的角度上分析。
不管是移动端、桌面端还是web端,UI一直是非常重要的,作为MVC的一部分,View承担了向用户展示界面的责任。
2.1 创建和布局
Android 有两种方式绘制View,一是使用xml文件,你可以在xml文件中用代码声明你的界面,也可以直接在可视化的界面上拖拽,控件可以通过R文件生成唯一对应的id,在Java 代码中就能通过id来找到控件并处理逻辑。二是在Java中使用代码绘制View和布局。笔者之前做过的项目,和实习是参加的项目基本都是用xml文件(代码声明),很少会直接使用代码。
iOS提供了三种方式来布局和绘制View,一是使用storyboard,二是使用xib文件(nib),三是直接在代码中布局。前两种方式是不需要代码的,将View和Controller剥离开来,而且不需要在类似xml的源码中添加代码,只需要在视图上鼠标操作就可以。第三种是在OC 中使用代码来绘制View与布局。使用代码布局更自由,而且能达到一切想要的效果。但是目前的问题是,在现在的项目代码中多数使用第三种方式布局和绘制View,我们的代码并没有将View 和Controller分离开,这对于MVC这种模式来讲,并不是好的。但是对于一些复杂的动态布局,只设置frame值可能不够,所以iOS新增了Auto layout自动布局,就是给View添加约束,大部分约束有点像Android里的相对布局。
还有一个小点让笔者不能适应的是,View的长宽属性中没有wrap_content这个概念,这使得在创建动态高度的自定义cell时,你需要自己计算每个cell的高度,这个折磨了我一段时间。
这里有个问题,为什么iOS的绘图性能一直优于Android,我觉得其中一个原因就是因为它简单的布局系统不会因为布局的复杂性增强而增加计算量,而Android不一样。
上面说到的5种布局都是ViewGroup的派生类,一个ViewGroup的生命周期经历3个阶段,分别是measure,layout和draw。在大部分情况下,ViewGroup都会自动的为它的子View撑开足够的空间,给子View计算出建议的宽和高和测量模式,决定子View的位置 。所以ViewGroup的子控件都是需要再次measure和layout的,在这就消耗了时间。在iOS中,如果不采用autolayout,使用View的frame那么所有位置(x,y,width,height)都已经确定了。所以iOS就不用进行measure和layout了,那么iOS只用尽管draw就行了。
2.2 控件
对于Android 和iOS,两个平台提供的官方的UI控件基本类似,基本能满足搭建一个简单App的需要。在iOS中基本都能找到和Android对应的控件,不同的是它们的API不同,使用方法不同,某些控件的外观能明显看出苹果的设计风格。(Android中的Toast其实很好用,但是不知道iOS中为什么没有)。
这里想重点说明的是TableView和ListView,列表是使用频率比较高的控件,Android的ListView使用了适配器Adapter来处理ListView的显示,这里使用了适配器模式。而iOS使用DataSource和delegate来代替Adapter的作用,可以通过dataSource 和delegate来设置每一行的外观样式,每一行的高度等等。
还需要补充说明的是,作为苹果这样一个使用闭源框架和设计主导的公司,其特点其实在控件中有所体现。Google的感觉就是扔给你一堆控件你可以随意组合,自由搭配。虽然都是给你一堆控件,但是苹果已经给你制定好了一些规范,并推荐应该怎样做。例如用UINavigationController创建一个需要层次化流程的界面,用UITarBarController创建我们大多数App 的主要功能页,用UIPageControl创建引导页或者广告栏,用UISearchBar创建搜索栏。Android中虽然有类似的比如TabHost这种控件,但是想要创造UITarBarController和UINavigationController搭配的效果,还是需要很多代码的。就这点来看,iOS方便了很多,只需要一些简单的官方控件就能构建起一个App。
Controller本职工作是控制UI和处理逻辑,其中UI Controller作为对View的Controll是很关键的,另外数据传递和IPC也算逻辑的一部分,所以放到这一节。
3.1 UIController
学习Android 的过程中,扑面而来的概念是Android的四大组件,而最重要的组件莫非Activity,强调一个Activity就是一个界面,而笔者在IOS 的学习过程中(看了《疯狂iOS》、《精通iOS开发》)都没有强调这个概念,而且对界面的跳转也没有做详细说明,也没有对UIViewController的管理做说明,笔者很是笔者一直在iOS 中寻找与Android对应的四大组件,首先UIViewController其实就是对应的Activity,UIViewController生命周期也与Activity的很类似。
Activity在onCreate()中初始化操作, 在onResume()中可以加一些改变界面和状态的操作。
UIViewController在 -viewDidLoad中初始化操作, 在-viewWillAppear 中可以加一些改变界面和状态的操作。
UIViewController整个生命周期:-viewDidLoad–> -viewWillAppear–> -viewDidAppear –> 运行态–> -viewWillDisAppear–> -viewDidDisAppear-> -viewDidUnload。
Android的Activity是由Activity栈进管理,当来到一个新的Activity后,此Activity将被加入到Activity栈顶,之前的Activity位于此Activity底部。可以设置Activity的taskAffinity和launchMode,以改变Activity入栈的形式。
iOS没有UIViewController启动模式,但提供有三种对视图的管理方式:
另外Android可以自由控制一个Activity ,想要销毁一个Activity的时候,就调用finish函数。而iOS 使用了ARC,并不主动提供销毁UIViewController的方法,只有等内存不足时,自动销毁非前台的UIViewController,这点是不一样的。但是我们可以让通过以下操作UIViewController提前自动释放。
1) 将任何引用到该controller的变量设置为空。
2) [controller.view removeFromSuperview]
3) controller.view = nil
4) controller = nil
3.2 数据传递
我们这里的数据传递主要是指的在一个App中进行数据传递,即在一个进程内的数据传递。Android里数据传递主要由Intent来承担,主要的数据传递方式如下:
广播BroadCast,是Android四大组件之一,应用程序可以拥有任意数量的广播接收器以对所有它感兴趣的通知信息予以响应。要实现广播,首先需要注册一个BroadCastReceiver,通过sendBroadcast这个方法来发送,之后在BroadCastReceiver的onReceiver接收消息。
iOS的数据传递方式有以下这些:
对象的@property属性值,这种方式主要用于在界面跳转时的数据传递。
3.3 IPC
与iOS相比,Android的进程间通信方式有很多,如ContentProvider,Activity,广播和AIDL等,不过这些方式的本质都是binder。
对于Binder推荐浏览一篇博客,写的非常详细:
http://blog.sina.com.cn/s/blog_73799d980101sa48.html
而对于iOS来说,为了避免数据冲突和更好的提供更好的安全机制,iOS提供了沙盒机制。尽管如此,iOS还是提供了若干今晨间通信机制,如URL Scheme、剪切版、CoreCFMessagePort和CFNotificationCenter等等。
4.1 存储
Android存储数据的方式有以下几种:
iOS都是沙盒存储,数据都在App的目录下,iOS的存储方式有以下几种:
4.2 垃圾回收
对于垃圾回收,由于Android使用Java语言,所以就自动沿用了Java的GC机制,Java有两种方式进行垃圾标记,一种是引用计数,一种是可达性分析算法,但一般选择可达性分析算法,因为引用计数会有循环引用问题。对于垃圾清理有四种方式:标记清理、复制清理、标记整理和分带收集。不同的GC器会选择不同的清理方式。Android有自己的虚拟机,5.0以前是Dalvik,以后是ART,不同的虚拟机也有不同的GC器,不同的GC器也会选择不同的清理方式。
对于iOS,iOS 5以前都是使用OC的手动回收机制,也就是引用计数,它的规则是这样的:如果需要持有一个对象,那么对其发送retain,如果之后不再使用该对象,那么需要对其发送release或者autorelease,每一次对retain,alloc或者new的调用,需要对应一次release或autorelease调用。所以我在网上很多博客里看到的iOS的代码都还有release和autorelease。
iOS5以后苹果引入了ARC内存管理机制,这是一种编译期的内存管理方式,从此不再需要手动释放对象了,要注意的是这里的ARC并不是GC,而只是由编译器自动添加retain和release代码,归根到底还是引用计数。ARC带来的好处是代码变得简单,代码的总量变少了,代码高速化,由于使用编译器管理引用计数,减少了低效代码的可能性。
尽管两个平台现在都已经有了自动处理垃圾的机制,但是我们还是需要小心处理某些对象和过程,以防发生内存泄漏。
Android可能的内存泄漏,例如使用完Bitmap之后没有recycle,使用完Sqlite数据库后没有关闭数据库,ListView中没有使用ViewHolder,使用完广播没有注销,使用Thread没有置空。
iOS里例如OC上的循环引用,NSTimer 会持有target,block循环引用变量,使用了NSNotificationCenter后没有注销,View中存在大量对象,显示结束时,没有及时置空,图片加载的复用等等。
Android中有主线程也叫UI线程,如果我们在UI线程进行了长时间(大于5s)的耗时操作,那么就会出现ANR(Application Not Responding)。iOS中也有主线程的概念,尽管没有ANR,但是长时间的占用主线程进行耗时操作将会带来非常差的体验感,所以Android和iOS都需要异步处理。
5.1 工具
Android与异步相关的工具有以下几种:
iOS中异步工具有以下几种:
5.2 looper &runloop
Android里的消息处理机制:消息是存放在一个消息队列中,应用程序的主线程围绕这个消息队列进入一个无限循环的,直到应用程序退出。Looper就是消息处理机制的关键,每一个线程都有唯一的Looper,主线程中的Looper默认开启。如果队列中有消息,应用程序的主线程就会把它取出来,并分发给相应的Handler进行处理;如果队列中没有消息,应用程序的主线程就会进入空闲等待状态,等待下一个消息的到来。
这里要注意的是其它线程不是默认开启的。但是如果你需要线程在处理完一些事情后,不要自己停止,那么就需要使用Looper。Looper由四个部分组成:
Message:消息,其中包含了消息ID,消息处理对象以及处理的数据等,由MessageQueue统一列队,终由Handler处理。
Handler:处理者,负责Message的发送及处理。使用Handler时,需要实现handleMessage(Message msg)方法来对特定的Message进行处理,例如更新UI等。
MessageQueue:消息队列,用来存放Handler发送过来的消息,并按照FIFO规则执行。当然,存放Message并非实际意义的保存,而是将Message以链表的方式串联起来的,等待Looper的抽取。
Looper:消息泵,不断地从MessageQueue中抽取Message执行。因此,一个MessageQueue需要一个Looper。
详细介绍见:
http://www.cnblogs.com/codingmyworld/archive/2011/09/14/2174255.html
iOS也有类似的东西名叫runloop,它的作用其实和looper 是一样,每个线程只有一个Runloop,在主线程中默认启动,一直循环接收消息。当你需要在其它线程干一些其它事时也可以自己启动。不能自己创建RunLoop,CFRunLoopGetMain和 CFRunLoopGetCurrent可以获得Runloop。RunLoop负责处理两种消息事件,输入源事件和计时器事件。
它具体的作用是维护线程的生命周期,让线程不自动退出;创建常驻线程,执行一些会一直存在的任务。
详细介绍见:http://blog.ibireme.com/2015/05/18/runloop/
Android使用Java编写,iOS使用OC,两大语言一个在Java虚拟机上静态执行,另一个动态执行,虽然原理不同,但是也有很多相似的地方。
6.1 基础
类似点:
Java的顶层类是Object,OC的顶层类是NSObject。
Java的toString()对应OC的description。
Java有匿名内部类和闭包,OC有block,但block可以访问外部代码。
Java有String、List、Set、Map等数据结构,OC有NSString、NSArray/NSMutableArray、NSSet/NSMutableSet、NSDictionary/NSMutableDictionary。
Java有各种基础类型的包装类,OC也有。
Java用new创建对象,OC用alloc init创建对象。
Java有强、软、弱、虚四种引用类型,OC使用weak, strong用来修饰变量。
不同点:
首先Java是静态的,OC是动态的且使用消息传递。
Java有空指针异常,OC没有。
Java 语法简洁,OC稍微有点啰嗦。
Java有命名空间和包管理,OC没有,所以需要加类前缀。
Java使用引用,OC使用指针。
Java类写在一个文件里,OC有头文件
OC方法名可以进行拆分 ,Java不可以。
OC 有KVO、KVC。
OC中有category,也叫非正式协议、类目。可以对现有的类进行扩展。
6.2 面向对象
面向对象三个特点是:封装,继承,多态。
另外Java的类成员变量需要自己写setter和getter,而OC中可以通过@ property字段来声明变量,就避免了写繁琐的setter和getter了。另外在OC的对象方法中,对成员变量的获取一般不用getXXX,而直接写成员变量的名称来作为getter方法的函数名。
继承:Java和OC继承是类似的,都有父类子类,子类继承父类的变量和方法,而不允许多重继承,this 对应self ,不同的是Java有重写和重载,OC 只有重写。
而OC中也有类似接口一样的东西,它的名字叫协议(protocol)。协议的字面意思就是你要遵守一组特定协议。用关键字@protocol来声明一个协议。所有绑定该协议的类也要实现该协议所声明的方法,只不过有些方法是必须实现的,有些是可选的,协议中用@required @optional来区分。达到的效果和Java接口是类似的。
6.3 高级特性
Java有反射机制,它允许程序创建和控制任何类的对象,无需提前硬编码目标类。它使类和数据结构能按名称动态检索相关信息,并允许在运行着的程序中操作这些信息。不过因为Java的方法是与class静态绑定的,所以不能改变方法。
然而OC与Java最大的不同是OC是动态的,因此方法、类和对象可以在运行时确定和修改,所以诞生出了Method Swizzing,它提供以下方法。
Method Swzzing能够做到修改系统API和面向切面的编程,这是非常强大的,可以利用它进行动态换肤,针对特定的版本的系统库函数打Patch,服务器动态修改客户端逻辑等等。
Material Design是Google2014 年提出的一套UI风格,中文名叫质感设计。这套风格定义了一系列的设计理念,包括主题、颜色、阴影、布局和动画等等。这套风格不仅包括移动端还包括web端。Android
5.0上大量的使用了Material Design风格的东西,还推出了官方的Material Design控件包,使用这个控件包可以很方便的搭建出一个具有Material Design风格的App。Material Design风格的App在体验上比现在普通的App要好很多。Android 6.0上又对这个包进行了扩展。凭借这套设计,Google力求统一用户UI的风格。遗憾的是现在市面上常用的android App 只有很少的几款使用了这个设计,笔者发现的只有格瓦拉、印象笔记、知乎。
Apple一直都是一家设计著称的公司,每一款iPhone、每一款pad都能吸引成千上万的人购买,可见大众对Apple Design的认同。在系统上也一样,iOS一直以流畅精美著称,系统中每一个控件或者都是Apple仔细斟酌的而设计的。每次的更新中iOS 也在对自己的UI进行不断优化,并提供新的控件使用,如最新的UIStackView(这个其实类似Android的线性布局),Widget等等。
Kotlin,可能大多是人对这门语言并不熟悉,它被成为Android上的swift。Kotlin由JetBrains设计开发并开源,虽然不是Google官方推出的语言,但是它一样被很多人看好。与Java相比,Kotlin的语法更简洁、更具表达性,而且提供了更多的特性,比如,高阶函数、操作符重载、字符串模板。它与Java高度可互操作,可以同时用在一个项目中。前段时间Oracle还就Java一事从Google那儿拿了不少赔偿费,Google会不会使用新的语言来代替Java呢?我们不得而知。
Swift是苹果新推出的用来代替OC的新语言,它一发布便被很多开发者迅速的接受并使用,它与OC相比更简单、灵活、执行速度快、支持更多高级特性,但是因为项目代码遗留的原因,大部分大公司的iOS都还没开始使用Swift。但是苹果官方是支持Swift的,并且将Swift开源,大部分API都在往Swift上靠,趋势已经很明朗了,未来Swift必然会替代Objective-C,但现在还是OC的时代。
React Native是Facebook推出的一套开源框架,它的目标是Learn once,write anywhere。React Native能够让你在使用Javascript和React的基础上获得完全一致的开发体验(android、iOS、web和服务器都使用的一样的语言),构建世界一流的原生APP。当然现在它在移动端火爆的原因不是因为语言统一,而是它能够像网页一样,随时更新,随时变化,这是现在的Android和iOS很难做到的。由于使用一个语言,将再也没有前端,终端,后台的区分,开发人员所关注的就是做一整套应用程序,所以现在新的创业公司或者个人会比较倾向于React Native。但是由于React Native技术刚刚出来不久,还有很多待改进的和bug,组件不全第三方组件也不全,性能并不如Native的,很多公司还是小心尝试,不过还是有一些大公司用它来完成部分页面。未来React Native会不会成为移动端的主流框架,这还需要看发展。但是这是一个互联网的时代,互联网生态必然会积累出更加优秀的框架,不管是React Native还是其它框架,它们一定会支撑更加伟大的产品出现。
Android和iOS作为移动端的OS,市场占有率超90%。一个属于Google一个属于Apple,在过去的几年中它们斗志斗勇,每个版本的发布都让开发者们兴奋好一段时间。
笔者学习了iOS一个月,总的来说觉得iOS和Android还是有很多共通之处的,比如:Activity 和UIViewController,相对布局和auto Layout,大部分控件,BroadCast和广播,looper和runloop等等,Java和OC也有很多共通性,虽然出自两个公司,但是很多思想很相相似。对不不同的东西,上文已经叙述的足够多,从宏观上讲iOS和Android最大的不同应该是一个底层是linux系统,一个是苹果特有的封闭系统。苹果特有的系统,能够保证iOS和Android相同的配置下,在显示、动画和运行效率上高出好几个档次,这是Android需要考虑优化的。另外还有一点是,两个系统一个开源一个闭源,这就像这就像搭积木的时候,一个给你各种各样积木块,你想怎么搭就怎么搭,你还可以拆开积木块,自己创造新的积木,但是搭出来的东西是好是坏,是丑是美自己负责。另一个只提供部分的改善过的积木块,并提供一些标准化规则和建议,让你能按照规则搭建,虽然不自由,但是能保证你最后做出来的东西是精美的。我个人来说更喜欢自由,因为自由往往代表着想象力和创造力,而且现在讲求共享,共享的东西往往能够产生出更多的东西。
未来是Android一统江湖,还是iOS独领风骚,又或者像现在这样二分天下一直竞争下去,我们拭目以待!
由于笔者水平有限,如果文章中有错误、认识存在偏差或者不全面的地方,还请各位前辈不吝赐教!!!