从C#到Objective-C,循序渐进学习苹果开发(2)--Objective-C和C#的差异

本随笔系列主要介绍从一个Windows平台从事C#开发到Mac平台开发苹果开发的一系列感想和体验历程,本系列文章是在起步阶段逐步积累的,希望带给大家更好,更真实的转换历程体验。
在上篇《从C#到Objective-C,循序渐进学习苹果开发(1)--准备开发账号和开发环境》介绍了一些基本的转换感悟和一些基础的准备工作,还没有正式真实的介绍Objective-C和C#的之前差异,我们知道,从一种环境或者一种语言转换过去另外一种,我们都会做一些对比和理解,这样可以很容易把我们头脑的知识进行对接,在这个所有东西日益大同的背景下,我们相信,所有的语言特点都是相通的。

1、面向对象的类

1)类的定义
Objective-C(下称OC)和C#都是面向对象的语言,虽然OC比C#古老,起源自C,但是很多特点和C#都很接近了,在C#3.0里面才引入的扩展方法,在OC里面也存在了。
OC和C#都一样,他们继承的关系都是单继承的,没有C++的那种多重继承那么复杂,OC很有特点的一个要求就是把接口和实现完全分开,这点是熟悉C#开发的人员必须转换过来的思路,在OC里面,写一个类,先写接口定义,然后再写实现,它的语法和C#很大不同,但是也很容易理解。
OC的关键字一般都是以@符号进行标识,这点和C#的默认保留关键字不同,一般看到@interface, @property, @关键字,在最新版本的XCode里面,真是发挥到了极致了,包含了很多语法糖,基本上和C#走向了大同,这点在对象的初始化继续介绍。
OC的类定义放到.h文件里面,实现放到了.m文件里面,如下面是类的接口声明。

@interface SimpleClass : NSObject
 
@end

而类的实现操作如下所示。

#import "SimpleClass.h"
 
@implementation SimpleClass
 
@end

上面只是一个演示类的概念,一般情况下,类都有属性或者方法,因此还需要增加很多东西。

另外OC和C#对比,没有了命名空间的概念,OC的类为了避免混淆一般通过前缀进行区分,如你看到的IOS基础类库,很多带有NS,UI,CA,等这样的前缀,就是这个原因。

2)方法的定义

如下面的接口声明一个方法,方法的定义

@interface XYZPerson : NSObject
- (void)sayHello;
@end

我们看到,上面的方法定义(接口定义)很简单,这里有一个 - 符号,是用来标识属于实例方法的,还有一种属于类级别的方法,用+符号标识,这个加号,类似于C#语言里面的static关键字,默认在方法定义为-的实例方法,都是类似于C#里面的public方法了。
这个(void)定义是返回值的标识,C#是不需要括号的void标识无返回值,方法最后需要括号标识。

  • (void)sayHello;
    这个方法的定义没有参数因此是这样写的,如果方法有多个参数,这个OC就很有意思,我感觉这个是OC里面最有个性的一个地方了。
    如果方法如下所示:
  • (void) setCaption: (NSString*)input;
    类方法的调用是通过空格,而C#通过点进行调用,这点也有所不同,OC通过在一个[]里面空格进行引用,如下所示。
[object method];
[object methodWithInput:input];

刚才定义的sayHello方法,它的调用可能就是如下方式了

[self setCaption:@"Default Caption"];

如果方法的定义为多个参数(也叫多重参数),定义如下。

-(void) setNumerator: (int) n andDenominator: (int) d; 

那么方法的调用就很有意思了。

[frac2 setNumerator: 1 andDenominator: 5]; 

如果还有更多的参数,那么也就一直使用这样的累加方式,这个有点接近阅读习惯,呵呵。
3)参数的定义
说完方法的定义和使用,我们介绍下类里面的属性的定义,我们知道C#里面的属性定义很简单了,如

public string Name {get;set;}

回来看看OC如何定义属性的,一般在.h的接口定义里,可以这样定义。

@property NSString *firstName;
@property NSString *lastName;

然后在实现类代码里面,添加它的对应代码@synthesize的关键字

@synthesize firstName, lastName;

属性当然也可以指定为只读的,如下代码所示

@property (readonly) NSString *fullName;

另外,我们还需要清楚,属性默认是线程安全的,也就是atomic,还有它是强类型Strong的。

@interface XYZObject : NSObject
@property NSObject *implicitAtomicObject;          // atomic by default
@property (atomic) NSObject *explicitAtomicObject; // explicitly marked atomic
@end

在很多地方,我们使用属性的时候,都不需要指定它的线程安全特性,因为那样效率更高,一般的属性定义代码如下所示。

@property (strong, nonatomic) IBOutlet UILabel *lblName;
@property (strong, nonatomic) IBOutlet UITextField *txtInput;

至于是不是所有的属性都应该指定为Strong,这个肯定不是的,strong的另外一种类型是weak,它是表示弱类型,强类型和弱类型主要是针对ARC来说的,它是引用计数的范畴,Strong相当于原来的retain。
一般情况下,为了避免一些强类型的对象属性导致出现相互引用的问题,在代理类和数据源对象,还有一些如UITable的对象属性,他们的属性定义必须指定为weak的。

2、对象的类型和初始化工作

在C#里面,我们知道,它里面包含了有一些基本类型(Primitive type)和一些包装后的对象类型,如它的基本类型包括string int char float long double decimal等等,它的对应包装类型有String Int32 Char Single Int64 Double Decimal等等。
在OC里面,同样也有这样的情况,OC的基本类型继承自C语言的基础类型,包括有int float double char 等基础类型,也有很多NS开头的引用类型(或者说包装类型),如NSString NSNumber NSDate NSData NSValue等等,而很多集合类型NSArray NSMutableArray NSDictionary等都需要添加引用类型的对象。
另外和C#的Object对象类似或者动态类型关键字dynamic指定的类型一样,OC里面包含了一个id的类型,这个是一个不确定的类型,它可以看成是一个任何类型的弱定义。
id类型是一个独特的数据类型,在概念上,类似java的Object类,可以转换为任何数据类型。换句话说,id类型变量可以存放任何数据类型的对象。在内部处理上,这种类型被定义为指向对象的指针,实际上是一个指向这种对象的实例变量的指针。需要注意的是id是一个指针,所以在使用id的时候不需要加星号;比如说:id foo=nil;
1)类对象的初始化
我们知道,OC里面很多都是通过alloc init这样的方式进行初始化,如下面代码所示。

XYZObject *object = [[XYZObject alloc] init];

而C#里面大多数使用new方式进行初始化,其实OC里面,也一样可以通过new方式进行初始化,不过仅限在默认构造函数的方式进行,如下的代码是等同于上面的语句的。

XYZObject *object = [XYZObject new];

不过好像很多人都习惯用第一种方式初始化对象。
2)字符串的初始化
相信很多人使用OC的时候,第一个印象最深的我觉得可能是NSString类了,这个是和C#的String有点类似,都是固定的字符串对象,如果需要变化类型的字符串对象,C#里面是可以使用StringBuilder,而OC里面可以使用NSMutalbeString,NSMutableString好比一个字符串链表,它可以任意的动态在字符串中添加字符串 删除字符串 指定位置插入字符串,使用它来操作字符串会更加灵活。
字符串的定义和初始化和简单,我们可以通过下面的方式进行初始化。

NSString *someString = @"Hello, World!";

我们知道,C#也可以使用@字符进行赋值,虽然一般情况使用在多行的情况下,但是在OC,这个@字符不能省略。
其他数据类型初始化,很多都依靠@字符进行,这个@字符可以说是非常强大的,它也可以说是一个很好的语法糖,如下面初始化各种类型的代码如下(在OC里面,NSNumber可以放置任何引用类型)

    NSNumber *myBOOL = @YES;
    NSNumber *myFloat = @3.14f;
    NSNumber *myInt = @42;
    NSNumber *myLong = @42L;

NSNumber类型可以装纳各种类型,同样它也可以转换为其他对应的类型,如下代码所示

   int scalarMagic = [magicNumber intValue];
    unsigned int scalarUnsigned = [unsignedNumber unsignedIntValue];
    long scalarLong = [longNumber longValue];
 
    BOOL scalarBool = [boolNumber boolValue];
 
    float scalarSimpleFloat = [simpleFloat floatValue];
    double scalarBetterDouble = [betterDouble doubleValue];
 
    char scalarChar = [someChar charValue];

另外,由于OC里面引入了一个id类型,可以认为它的作用和C# 3.0引入的动态类型相当,它可以在运行时进行确定对象是否具有某个方法,而不会在编译的时候强制指定。

如下面的代码编译通过,运行的时候可能出错。

id someObject = @"Hello, World!";
[someObject removeAllObjects];

之所以编译的时候,不检查它的对象是否有removeAllObject接口方法,是因为这儿的someObjec指定为了id的动态类型,所以编译器会不检查它的方法。

3)对象集合的初始化

刚才上面介绍了字符串等各种类型的初始化,很多采用了强大的关键字@进行初始化,这个语法糖减少了很多繁琐的方法调用,对于集合的初始化,尤其这样。

如果按照传统的集合定义方式,一般是通过下面的方法。

NSArray *someArray =
  [NSArray arrayWithObjects:someObject, someString, someNumber, someValue, nil];

在里面的集合,最后必须加上一个nil的东西,这个在C#的领域里面是不需要增加这样的标识的,在Object C里面,如果你要通过arrayWithObjects方法进行构造,必须增加一个这样的东西,告诉它这个是最后了,如果你把这个放到第二位,那么构造的集合也只有两个对象了,很奇怪了。
如果采用了强大的@方法构造,一切都和C#相似了,这里你只能佩服它的神奇之处了。

NSArray *someArray = @[firstObject, secondObject, thirdObject];

如下面定义一个字符串的集合是这样的。

NSArray *unsortedStrings = @[@"gammaString", @"alphaString", @"betaString"];

在C#里面,我们经常用到了字典对象,这个对象非常方便。当然在OC里面,也肯定会有这样的东西,毕竟很多语言都会支持的。
这个字典类型也是一个集合类型,它的传统构造方法如下所示

NSDictionary *dictionary = [NSDictionary dictionaryWithObjectsAndKeys:
             someObject, @"anObject",
             @"Hello, World!", @"helloString",
             @42, @"magicNumber",
              someValue, @"aValue",
              nil];

它这个看起来很怪异,添加指点是按照object,key的这样方式添加的,这个与我们使用C#的习惯有很大的不同哦,而且最后还带了一个nil的尾巴。

如果采用@构造函数,一切又都清净了,已经是key,value方式进行存储,而且不用nil了,如果你添加了nil,那么会出错的。

NSDictionary *dictionary = @{
   @"anObject" : someObject,
   @"helloString" : @"Hello, World!",
   @"magicNumber" : @42,
   @"aValue" : someValue
};

集合中,如果取某个对象,那么通过下面的方法进行获取

NSNumber *storedNumber = [dictionary objectForKey:@"magicNumber"];

也可以通过下标括符进行获取

NSNumber *storedNumber = dictionary[@"magicNumber"];

如果是一般的数组集合,可以通过下面方式获取,这种方式和c#很类似了。

NSNumber *storedNumber = array[0];

由于时间和篇幅的问题,关于OC的各种和C#对比的特性,以后继续介绍,OC里面还涉及很多相关的特点,如扩展方法,协议(类似接口),代码块等等内容,以及XCode的各种使用特性,有空继续介绍。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,456评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,370评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,337评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,583评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,596评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,572评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,936评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,595评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,850评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,601评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,685评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,371评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,951评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,934评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,167评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,636评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,411评论 2 342

推荐阅读更多精彩内容