NSLog类我们经常用,我整理了一下NSLog相关的东西,就在这里写一个总结吧。仅为个人见解,有谬误处望大神指正~
基本介绍:
在Objective-C中,NSLog相当于C语言中的printf,常用于文字输出。NSLog很像printf,同样会在console中输出显示结果。不同的是,传递进去的格式化字符是NSString的对象,而不是char *这种字符串指针。
NSLog的参数跟printf一样,是不定长参数。这个在下面会进行介绍。
NSLog的格式如下所示:
%@ 对象
-
%d, %i 整数
-
%u 无符整形
-
%f 浮点/双字
-
%x, %X 二进制整数
-
%o 八进制整数
-
%zu size_t
-
%p 指针
-
%e 浮点/双字 (科学计算)
-
%g 浮点/双字
-
%s C 字符串
-
%.*s Pascal字符串
-
%c 字符
-
%C unichar
-
%lld 64位长整数(long long)
-
%llu 无符64位长整数
-
%Lf 64位双字
这里简单介绍一下%@。
%@是对象的格式替换符,它可以替换所有的对象。包括各种自定义对象。
%@的基本原理是调用每个对象的description方法并显示结果。description方法是NSObject的方法,每个Objective-C的自定义类都可以重写这个方法。如果你的自定义类并没有重写这个方法,直接用%@输出自定义类的对象,输出到控制台上面的内容将会是<类名:对象地址>
-(NSString*)description;
下面说一下Objective-C中关于NSLog的实现机制。
FOUNDATION_EXPORT void NSLog(NSString *format, ...) NS_FORMAT_FUNCTION(1,2);
#define NS_FORMAT_FUNCTION(F,A) __attribute__((format(__NSString__, F, A)))
上面一行代码是NSObjCRuntime.h中关于NSLog的定义。
下面一行是NS_FORMAT_FUNCTION(F,A)的定义
我们先来看前面的部分
FOUNDATION_EXPORT
这一行在NSObjCRuntime.h中定义如下:
#define FOUNDATION_EXPORT FOUNDATION_EXTERN
这一行,extern的作用不必细说,百度百科有云,extern置于变量或者函数前,以表示变量或者函数的定义在别的文件中,提示编译器遇到此变量和函数时在其他模块中寻找其定义。另外,extern也可用来进行链接指定。
#define NS_FORMAT_FUNCTION(F,A) __attribute__((format(__NSString__, F, A)))
这个定义中,该__attribute__属性可以给被声明的函数加上类似printf或者scanf的特征,它可以使编译器检查函数声明和函数实际调用参数之间的格式化字符串是否匹配。该功能十分有用,尤其是处理一些很难发现的bug。
void NSLog(NSString *format, ...)
然后来看NSLog的参数部分。说到这儿,要简单介绍一下不定长参数的一些知识。
所谓不定长参数,就是函数的形参数量不定,类型也可能是不定的。我们把像上面的函数sum中如“int num”这样的参数叫做“有名参数”,后面用“…”代表的都是“匿名参数”,有名参数是可以在函数中通过变量名直接访问的,匿名函数则无法通过变量名直接访问,只能是通过相对有名参数的位置(地址)来访问了。
关键在于:
(1)匿名参数的个数和类型必须通过有名参数传递给被调函数
如NSLog中的第一个参数NSString* format,在format中不仅告诉了NSLog参数的个数,还必须指定正确的类型,二者缺一不可。
(2)被调函数本身有办法直接或间接定位参数的个数和类型
即描述参数个数和类型的参数的位置应该是固定的,函数有办法定位它们,而不是如匿名参数那般不确定的。
也就是说,后面的参数列表中的内容,跟前面的format中声明的参数的类型和数量必须一一匹配,而且因为后面的NS_FORMAT_FUNCTION(1,2),xcode会自动检测不匹配的行为并且予以警告。