关于ios命名空间(NameSpacing)

为什么Objecive-C中的很多类名都是NS开头的呢?

我保证在你第一次给别人介绍Objective-C的时候肯定会听到这句话。

就像父母要向孩子解释什么是死亡或者圣诞老人是不存在的问题一样,父母总是寄希望时间会让孩子自己找到答案。

你既然这么问了,实际上NS代表了NeXTSTEP(好吧,其实是代表NeXTSTEP/Sun,我们只是做个简单的介绍),它被用于...

你越解释,你会发现对方越失望*,接下来,他们不在只是随便问问了,他们开始问一些你更难解释的问题--在Objective-C中@是什么?

命名一直是Objective-C的硬伤,和那些优雅的语言相比,Objective-C缺乏标识符容器这点引来了很多不切实际的批评家。

他们总是说:Objective-C不像其他流行语言一样提供模块化机制来避免类名和方法名的冲突。

相反地,Objective-C 依靠前缀来确保APP中的方法名不会影响其他有相同名字的代码。

插入一个关于类型系统的题外话之后我们会继续进入关于命名的讨论。

C和Objective-C中的类型

我曾在这个博客上多次提过Objective-C是直接建立在C语言之上的,一个重要的原因是Objective-C和C语言共用一个类型系统,他们都要求标识符是全局唯一的。

你可以自己定义一个和@interface同名的静态变量,编译之后你会得到一个错误:

Objective-C

@interfaceXXObject:NSObject@endstaticchar*XXObject;//将“XXObject”重新定义为不同的符号

也就是说,Objective-C的runtime在C语言的类型系统上又创建了一个抽象层,它甚至可以允许下面这段代码被编译:

Objective-C

@protocolFoo@end@interfaceFoo:NSObject{idFoo;}@propertyidFoo;+(id)Foo;-(id)Foo;@end@interfaceFoo(Foo)@end@implementationFoo@synthesizeFoo;+(id)Foo{idFoo=@"Foo";returnFoo;}@end

通过Objective-C的环境,程序能区别所有相同名字的类,协议,类别,实例变量,实例方法和类方法。

一个变量能重新调整一个已经存在的方法也是得益与C语言的类型系统(这个有点像一个变量能够隐藏它的隐藏功能)

前缀

在Objective-C应用中的所有类名都必须是全局唯一的。由于很多不同的框架中会有一些相似的功能,所以在名字上也可能会有重复(users, views, requests / responses 等等),所以苹果官方文档规定类名需要有2-3个字母作为前缀。

类前缀

苹果官方建议两个字母作为前缀的类名是为官方的库和框架准备的,而对于作为第三方开发者的我们,官方建议使用3个或者更多的字母作为前缀去命名我们的类。

一个资深的Mac或iOS开发者可能会记得下面大部分的缩写标识符:

PrefixFrameworks

ABAddressBook / AddressBookUI

ACAccounts

ADiAd

ALAssetsLibrary

AUAudioUnit

AVAVFoundation

CACoreAnimation

CBCoreBluetooth

CFCoreFoundation / CFNetwork

CGCoreGraphics / QuartzCore / ImageIO

CICoreImage

CLCoreLocation

CMCoreMedia / CoreMotion

CVCoreVideo

EAExternalAccessory

EKEventKit / EventKitUI

GCGameController

GLK*GLKit

JSJavaScriptCore

MAMediaAccessibility

MCMultipeerConnectivity

MFMessageUI*

MIDI*CoreMIDI

MKMapKit

MPMediaPlayer

NKNewsstandKit

NSFoundation, AppKit, CoreData

PKPassKit

QLQuickLook

SCSystemConfiguration

Sec*Security*

SKStoreKit / SpriteKit

SLSocial

SSSafari Services

TWTwitter

UIUIKit

UTMobileCoreServices

第三方类前缀

直到最近,由于CocoaPods的出现和大量新的iOS开发者的涌现,开源代码的遍布,第三方代码在很大程度上对苹果和其余的Objective-C开发社区来说已经不是问题了。最近苹果官方的命名指南也发生了变化,它将三个字母作为前缀的建议只是做为一个习惯做法。

正因为这样,那些已经存在的第三方库依然使用2个字母作为前缀,你可以参考一些那些在GitHub上得到很多start的Objective-C的仓库

PrefixFrameworks

AFAFNetworking("Alamofire")

RKRestKit

GPUGPUImage

SDSDWebImage

MBMBProgressHUD

FBFacebook SDK

FMFMDB("Flying Meat")

JKJSONKit

FUIFlatUI

NINimbus

RACReactive Cocoa

我们已经看到在在这个第三方库的前缀已经和我的AFNetworking一样了,所以最好还是要在你的代码中遵守要三个字母以上的作为类前缀的规定(https://github.com/AshFurrow/AFTabledCollectionView)。

对于那些针对特殊功能而写的第三方库的作者,可以考虑在下一次主要升级时使用@compatibility_alias来为那些使用者们提供一个天衣无缝的转移路径。

方法前缀

不仅是类容易造成命名冲突,selectors也很容易造成命名冲突,甚至方法比类会有更多的问题。 考虑一下这个category:

Objective-C

@interfaceNSString(PigLatin)-(NSString*)pigLatinString;@end

如果-pigLatinString方法被另一个category实现了(或者以后版本的iOS或者OS X 在NSString类中也添加了同样名字的方法),那么调用这个方法就会得到未定义的行为错误,因为我们不能保证在runtime中哪个方法会先被定义。

我们可以通过在方法名前加前缀来避免这个问题,就像加这个类名一样(在类别名前加前缀也是个好办法):

Objective-C

@interfaceNSString(XXXPigLatin)-(NSString*)xxx_pigLatinString;@end

苹果官方建议所有category方法都要使用前缀,这个建议比类名需要加前缀的规定更加广为人知和接受。

很多开发者都在热情地讨论着这个规定的某一方面。然而,无论是通过成本角度还是效益角度来衡量命名冲突风险的可能性都是是不全面的:

category的主要功能是通过语法糖将一些有用的功能包裹进原来的类中。任何一个category方法都可以被选择性实现,你也可以把他当做是self的一个隐型功能方法。

当我在编译器的环境参数中将OBJC_PRINT_REPLACED_METHODS这个参数设置为YES,那我们就能在编译的时候检测方法名是否有冲突。实际上,方法名的冲突是很少发生的,而且在发生的时候,他们通常会得到一个needlessly duplicated across dependencies的提示。即使发生最坏的情况,程序在运行是出现异常,那么很可能是两个方法名一样,那么他们的事情也是一样的,所以结果也不会有什么变化。就像Swiss Army Knife写了一个category,他定义了NSArray中的-firstObject这个方法,那么只要苹果官方没有在NSArray中加这个方法的话,那么这个类别方法一直有效的。

在苹果官方的编程指南中有很多严肃又松散的解释。这里没有固定的文档,他们可能一直变化。看到这里,如果你还是悬而未决,那么你只需要把的category方法名加上前缀,如果你还是选择不去做任何改变,那么你就等着自食其果吧。

Swizzling

在Swizzling时,方法名加前缀或者后缀也是非常有必要的,这个我在上周关于swizzling的文章中提到过。

Objective-C

@implementationUIViewController(Swizzling)-(void)xxx_viewDidLoad{[selfxxx_viewDidLoad];// Swizzled implementation}

我们真的需要命名空间么?

在最近关于Objective-C替换、改造和重塑的讨论中,我可以明显地发现命名空间是未来的一个趋势。但是它到底给我们带来了什么呢?

美学?除了IETE成员和军事人员,我想没有人会喜欢CLAs的视觉审美,但是用::,/或者另外的.这些符号真的能让我们觉得更好么?你真的想要以后把NSArray叫做"Foundation Array"?(那我这个NSHipster.com这个博客不是也得改名字了?!)

语义学?让我们比较一下其他的语言,看看他们是怎么用命名空间的,那么你就会意识到命名空间不能解决所有不明确的问题。可能在某些额外环境的情况下,那些命名空间会出现更多问题。

你还是不赞同,那么你想象一下Objective-C的命名空间的实现可能会像这个样子,你会觉得怎么样:

Objective-C

@namespaceXX@implementationObject@usingF:Foundation;-(void)foo{F:Array*array=@[@1,@2,@3];// ...}@end@end

虽然Objective-C有繁琐的代码但也有容易理解的明显优点。我们作为开发者去讨论NSString的时候,我们不会把它理解成别的意思,编译器也是一样。当我们在阅读代码时,我们不需要过多地去考虑这些代码是什么作用的。并且最重要的是,NSString这个类名在google这些搜索引擎中很容易就可以找到, 你不会得到其他结果

不管怎样,如果你对这个讨论感兴趣的话,我强烈建议你看一下Kyle Sluderthis namespace feature proposal。非常值得一看。

转:原文链接

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

推荐阅读更多精彩内容

  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,698评论 0 9
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,647评论 18 139
  • 1.项目经验 2.基础问题 3.指南认识 4.解决思路 ios开发三大块: 1.Oc基础 2.CocoaTouch...
    阳光的大男孩儿阅读 4,980评论 0 13
  • 翻译论文上课大姨妈很痛不能喝咖啡
    dq920813阅读 168评论 0 0
  • 读猫说之前,Zcash对于我来讲是一个比较陌生的币种。之前只是知道云币网首页第三个币种是ZEC,现在多了个BCC后...
    陈晓东_阅读 296评论 0 2