iOS ---不一样的NSLog打印(精准打印)

摘要:
在iOS开发过程中,调试是一个非常重要的过程。除了各种断点调试之外,NSLog似乎是最常用的调试方法。当然,这也是查找bug最简单、最简单的方法。然后问题就出现了。如何使用NSLog确定位置?--˃当宏被预编译时,它将被替换为当前函数的名称。对于Log,打印当前函数无非是_ cmd、__ func_、__ function_、PRETTY_ function懂得感恩并尊重他们的成就:iOS/Cooa:使用代码本地化程度更高的日志,以及iOS调试技能的豪华包。以上是我想介绍的关于NSLog的内容,它有助于在开发中定位打印。

  在iOS开发过程中,调试是很重要的过程,而除了各种断点调试(普通断点、条件断点、全局断点)之外,似乎NSLog是我们调试最常用的方法,当然,也是最简单朴素的寻debug方法。

  在项目中,我们常使用的NSLog的语句无外乎以下一种:  

 NSLog(@"打印字符串:%@",name);

  NSLog(@"打印整形:%i",number);//或者 %li ; %ld ; %d

  NSLog(@"打印字符:%c",c);

  NSLog(@"打印单浮点数:%f",f);

  NSLog(@"打印精度浮点数:%.2f",f);

  NSLog(@"BOOL--b-->%@",isYES?@"YES":@"NO");//打印布尔类型

  但这样并不是我们想要的结果,因为打印出来的结果,有时我们并不知道它打印的具体位置,甚至于具体信息。打个比方吧,我们经常会遇到的数组越界、网络请求数据为空等等。

  我们试一下看看:创建一个空数组-->dataArray,却打印数组的第三个元素。打印结果如下图:  iOS ---不一样的NSLog打印(精准打印)第1张

  从上图中我们可以看出,一个标准的数组越界,但我们看得出来,debug输出只是说又一个数组越界报错了,但并没有指出是哪个数组,数组在哪个位置。。。试想:如果我们在一个ViewController中有几个数组,这时数组出现越界。就会很难找出到底哪个数组出现debug。那么问题就出现了,怎样才能使用NSLog还能确定位置呢。   下面正式开始要说的话题:不一样的NSLog打印

  要使用不一样的NSlog,首先要了解 以下:预处理器在C/C++/objective-C语言提供的宏

 

  C/C++/Objective-C中用于日志输出的预处理宏.

MacroFormat SpecifierDescription
__func__%s当前函数前面
__LINE__%d源码文件中的行号
__FILE__%s源码文件完整路径
__PRETTY_FUNCTION__%s和__func__类似, 但是在 C++ 代码中包含更多的信息.

  Objective-C中用于日志输出的表达式

ExpressionFormat SpecifierDescription
NSStringFromSelector(_cmd)%@当前选择器的名字
NSStringFromClass([self class])%@当前对象类的名字
[[NSString stringWithUTF8String:__FILE__] lastPathComponent]%@源码文件的名称

 

[NSThread callStackSymbols]

 

%@

 

 

当前栈信息的刻度字符串数组。仅用于调试,不用向终端用户展示或者在代码中用作任何逻辑。

 

 

     *   __func__%s 当前函数签名

     *   __LINE__ %d 在源代码文件中当前所在行数  ---->宏在预编译时会替换成当前的行号

 

     *   __FILE__ %s 当前源代码文件全路径   -->宏在预编译时会替换成当前的源文件名

     *   __PRETTY_FUNCTION__ %s 像 __func__,但是包含了C++代码中的隐形类型信息。 ---->宏在预编译时会替换成当前的函数名称

 

  而关于Log,打印当前函数无非就是_cmd, __func__, __FUNCTION__, PRETTY_FUNCTION。

 1、打印当前的函数名,以及当前代码所在文件中得行数  

NSLog(@"%s, %d", __FUNCTION__, __LINE__);//直接定位到debug的函数名,以及当前代码所在文件中得行数。

 

  在- (void)viewDidLoad中输入以下代码。。。

iOS ---不一样的NSLog打印(精准打印)第2张

  打印结果如下:

iOS ---不一样的NSLog打印(精准打印)第3张 

 2、打印当前的函数名, NSStringFromSelector 获得参数的选择器所代表的方法的字符串

 NSLog(@"%@", NSStringFromSelector(_cmd));

打印结果如下:

iOS ---不一样的NSLog打印(精准打印)第4张

3、打印当前源代码文件全路径 

NSLog(@"%s", __FILE__);

 

  打印结果如下:

     iOS ---不一样的NSLog打印(精准打印)第5张

4、使用__PRETTY_FUNCTION__

上面介绍过:像 __func__,但是包含了C++代码中的隐形类型信息。

NSLog(@"%s", __PRETTY_FUNCTION__);

 打印结果:

iOS ---不一样的NSLog打印(精准打印)第6张

 

 在Xcode中 :

     _cmd会返回一个SEL对象,而剩下的都是来自C/C++编译器的定义所以都会返回一个C的字符串,显示结果也都差不多,可能不同编译器会有小小的差别。

         显然后面__func__系列要比_cmd好用,相对Objective-C类型内的方法调用,他不仅会显示方法名,还会显示类型,配合__LINE__,可以精确定位出Log在代码中的位置。

以下是我学习时查看的文章,也分享给大家。懂得感恩,懂得尊重它们的劳动成果:

iOS/Cocoa: 使用代码定位性更高的Log

iOS各种调试技巧豪华套餐

 

好了,以上就是我想介绍的NSLog有助于开发中定位打印的知识。都是我在接触到喜欢、再到使用的东西。iOS还有很多很多东西是我们不懂的,甚至是毫无所知的。一步一步来,摸索->学习->实践->熟练。每一步都很辛苦,坚持下,往前走会是不一样的风景。

 

免责声明:文章转载自《iOS ---不一样的NSLog打印(精准打印)》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇K8s部署构建Oracle的Docker镜像下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

C#使用FFmpeg的总结

上篇文章提到FFmpeg解决项目中视频和语音问题,说道C#和FFmpeg不得不提的2个类库。 1、Xabe.FFmpeg 简单查看了下源码和demo,发现基于ffmpeg.exe的命令行参数进行处理。 2、FFmpeg.AutoGen 把C语言对外API和类型翻译成C#的API和对象。 看了一下这个源码生成,其中使用了CppSharp来生成g.cs的文件,...

第三节:Vue3向下兼容2(v-for、数组方法、v-model、计算属性、监听器)

一. 基本指令  1. v-for 数据准备 data() { return { userInfo: { name: 'ypf', age: 18, school: '北大' }, movies: ["星际穿越",...

js forEach参数详解,forEach与for循环区别,forEach中如何删除数组元素

 壹 ❀ 引 在JS开发工作中,遍历数组的操作可谓十分常见了,那么像for循环,forEach此类方法自然也不会陌生,我个人也觉得forEach不值得写一篇博客记录,直到我遇到了一个有趣的问题,我们来看一段代码: let arr = [1, 2]; arr.forEach((item, index) => { arr.splice(inde...

C#调用C++dll,传指针释放内存问题

一、传入dll前,在C#中申请内存空间 c#里面的指针即 IntPtr 申请如下: IntPtr SrcImgData = Marshal.AllocHGlobal(length);  这种需要提前知道空间大小,否则无法确定空间大小,会导致dll内部处理时越界报错。 c#里面申请空间了,那么c++里面一般就是在这些空间里面操作了,一般不会重新分配内存,那么...

Bash数组-判断某个元素是否在数组内的几种方法

声明一个数组array,一个待测试元素var array=( element1 element2 element3 ) var="element1" 接下来用几种方法来分别测试var是否是array中的元素 判断方法1: echo "${array[@]}" | grep -wq "$var" && echo "Yes" || ech...

算法训练 删除数组零元素

算法训练 删除数组零元素   时间限制:1.0s   内存限制:512.0MB     从键盘读入n个整数放入数组中,编写函数CompactIntegers,删除数组中所有值为0的元素,其后元素向数组首端移动。注意,CompactIntegers函数需要接受数组及其元素个数作为参数,函数返回值应为删除操作执行后数组的新元素个数。输出删除后数组中...