对于程序的开发者来说,拥有一手强大的DEBUG能力,那就好比在武侠世界中拥有一种强大的内功心法一样,走到哪里都是大写的牛B。在我们DEBUG的时候,大部分情况都是要查看我们的调试日志的,这些打印日志可以帮我们精确的定位问题的位置。在OC的编程中,我们一般使用NSLog函数来进行一些打印的工作,这大致相当于C语言的printf输出语句。

1.基本使用

NSLog定义在NSObjCRuntime.h中,如下所示:

void NSLog(NSString *format, …);

省略号表示可接收多个参数。

NSLog使用起来和printf是很相似的,都是格式化的输出,不同的是printf需要的格式化字符串是char *类型,而NSLog需要的字符串是NSString型。

NSLog中会使用到的格式化占位符:

  1. %@ 对象
  2. %d, %i 整数
  3. %u 无符整形
  4. %f 浮点/双字
  5. %x, %X 二进制整数
  6. %o 八进制整数
  7. %zu size_t
  8. %p 指针
  9. %e 浮点/双字 (科学计算)
  10. %g 浮点/双字
  11. %s C字符串
  12. %.*s Pascal字符串
  13. %c 字符
  14. %C unichar
  15. %lld 64位长整数(long long)
  16. %llu 无符64位长整数
  17. %Lf 64位双字

示例:

  1. // 直接打印字符串
  2. NSLog(@"this is a string");
  3. // 打印OC对象
  4. NSString *string = @"hello world";
  5. NSLog(@"%@", string);
  6. // 打印基本数据类型
  7. NSLog(@"int : %d, float : %f", 2, 3.14);

2. description 方法

通过上面的例子我们不难发现在NSLog中使用%@占位符来打印对象的功能是非常强大的,它不仅可以打印字符串、字典、数组等OC中已经定义的对象类型,我们自定义的对象类型也可以使用%@来打印。NSLog配合%@使用时,编译器会自动调用被打印对象的description方法,如果打印的是类对象则调用类方法,实例对象则调用实例方法。继承于NSObject类的description方法默认会打印该对象的类名和它在内存中的地址。

新建一个People类,自定义description方法,调试打印。

People.h:

  1. #import <Foundation/Foundation.h>
  2. @interface People : NSObject
  3. @property (nonatomic, copy) NSString *name;
  4. @property (nonatomic, copy) NSString *sex;
  5. @property (nonatomic, assign) int age;
  6. @end

People.m:

  1. #import "People.h"
  2. @implementation People
  3. // 不要尝试在description方法中返回self,因为会出现死循环
  4. - (NSString *)description {
  5. /*
  6. // 错误示范
  7. return [NSString stringWithFormat:@"%@", self];
  8. */
  9. return [NSString stringWithFormat:@"name: %@, sex: %@, age: %d",
  10. self.name, self.sex, self.age];
  11. }
  12. + (NSString *)description {
  13. return @"People";
  14. }
  15. @end

测试代码:

  1. // 创建并初始化一个People实例p
  2. People *p = [[People alloc] init];
  3. p.name = @"jack";
  4. p.sex = @"man";
  5. p.age = 18;
  6. // 打印实例p,调用自己定义的description实例方法
  7. NSLog(@"%@", p);
  8. // 打印People类,调用自己定义的description类方法
  9. NSLog(@"%@", [p class]); // 类对象或者实例对象调用class方法会返回对象的类的所有信息,返回值为Class类型

自己实现description方法可以打印我们想要看到的内容,更加便于调试,但是在description方法中不要返回或者打印self。

3.自定义打印日志输出内容

从上面的例子我们可以看到使用NSLog打印出来的日志前面带有很长一串的时间戳,可能有时候我们根本用不到,而且还会影响查看的效率。其实对于NSLog的输出格式我们也是可以自定义的,只需要我们写一个宏定义就可以了。

如果只想让NSLog打印我们输出的内容:

  1. #define NSLog(FORMAT, ...) fprintf(stderr,"%s\n",[[NSString stringWithFormat:FORMAT, ##__VA_ARGS__] UTF8String]);

附加输出文件名和打印语句的行号:

  1. #define NSLog(FORMAT, ...) fprintf(stderr,"%s:%d\t %s\n",[[[NSString stringWithUTF8String:__FILE__] lastPathComponent] UTF8String], __LINE__, [[NSString stringWithFormat:FORMAT, ##__VA_ARGS__] UTF8String]);

由于NSLog的使用效率比较低,所以在我们的项目中非调试状态下不应该出现大量的NSLog,所以有些时候我们会在项目的.pch文件中去定义一个宏,让调试打印函数只在调试的时候有用,发布的时候就不能使用。

在Xcode6之前,项目中Supporting files文件夹下有个 “工程名-Prefix.pch”文件,你可以在该文件下声明我们的宏,这样之后工程中的任何一个文件中都可以使用该宏(也就是说可以在.pch中放一些全局的东西)。Xcode6之后,可能是因为大家把大量的头文件和宏定义放到pch里边,导致编译时间过长,苹果就去掉了pch文件,但是我们仍可以手动添加。

实用版本:

  1. #ifdef DEBUG
  2. #define NSLog(FORMAT, ...) fprintf(stderr,"%s:%d\t %s\n",[[[NSString stringWithUTF8String:__FILE__] lastPathComponent] UTF8String], __LINE__, [[NSString stringWithFormat:FORMAT, ##__VA_ARGS__] UTF8String]);
  3. #else
  4. #define NSLog(...)
  5. #endif

iOS开发--调试必备 — NSLog的更多相关文章

  1. 利用Nginx轻松实现Ajax的跨域请求(前后端分离开发调试必备神技)

    利用Nginx轻松实现浏览器中Ajax的跨域请求(前后端分离开发调试必备神技) 前言 为什么会出现跨域? 造成跨域问题的原因是因为浏览器受到同源策略的限制,也就是说js只能访问和操作自己域下的资源,不 ...

  2. iOS开发调试概览

    概述 我们都知道Xcode默认的调试器是LLDB(在此之前使用的是GDB),但是关于LLDB的debug技巧并非所有人都比较清楚,可能所有人都知道p或者po命令打印一些变量.但是实际的情况时这些还远远 ...

  3. iOS 开发调试技巧

    对于软件开发而言,调试是必须学会的技能,重要性不言而喻.对于调试的技能,基本上是可以迁移的,也就是说你以前在其他平台上掌握的很多调试技巧,很多也是可以用在iOS开发中.不同语言.不同IDE.不同平台的 ...

  4. iOS开发调试技巧总结(持续更新中)

    作者:乞力马扎罗的雪  原文 对于软件开发而言,调试是必须学会的技能,重要性不言而喻.对于调试的技能,基本上是可以迁移的,也就是说你以前在其他平台上掌握的很多调试技巧,很多也是可以用在iOS开发中.不 ...

  5. iOS开发中关于nslog的几种流行做法小结

    不管哪种方法,都必须在PCH文件中做下宏定义 DEBUG和RELEASE要分开,RELEASE时log打印要取消 方法一:简单直接,用几行代码搞定,简洁但功能少 #ifdef DEBUG #defin ...

  6. iOS开发调试Reveal使用

    推荐通过Xcode中加断点的方式集成Reveal(小缺陷,当你禁用断点时或者不用Xcode而用Appcode开发时,这个方式是不管用). 打开您的iOS工程,选择 View → Navigators ...

  7. IOS 开发调试方法

    0.警告 尽量一个警告都不要有 1.错误 1)红色提示 编译过不去的原因大部分是语法,检查括号的匹配,变量名称,作用域范围 2)编译可以通过,可以运行 a.运行过程中程序崩溃 在debug区域的右侧, ...

  8. iOS开发工程师必备技能(持续更新)

    Objective-C Objective-C语言基础 library,framework的制作 Runtime 编程 LLVM 原理和调优 操作系统 iOS内存管理和调优 iOS的文件系统和沙盒机制 ...

  9. iOS开发调试篇—Print Description of "string"

    Print Description of "string":把 string 的信息输出到控制台.Copy:复制 string 的信息,包含变量名,类名和值.View Value ...

随机推荐

  1. javascript 创建对象及对象原型链属性介绍

    我们知道javascript里定义一个普通对象的方法,如: let obj = {}; obj.num = 1; obj.string = 'string'; obj.func = function( ...

  2. 【转】 管理CPU 亲和性

    简单地说,CPU 亲和性(affinity) 就是进程要在某个给定的 CPU 上尽量长时间地运行而不被迁移到其他处理器的倾向性.Linux 内核进程调度器天生就具有被称为 软 CPU 亲和性(affi ...

  3. Leeo 智能夜灯:默默守护你的家

    http://www.ifanr.com/462377 Leeo 智能夜灯是一个低调的设备.它不需要你与之交互,也不会产生多余的费用.当你把它插到墙上插座,然后下载应用后,就不用再管它了.然后,它就开 ...

  4. centos 格式化硬盘并挂载,添加重启后生效

    [root@cloud /]# passwd 更改用户 root 的密码 . 新的 密码: 重新输入新的 密码: passwd: 所有的身份验证令牌已经成功更新. [root@cloud /]# fd ...

  5. MVC学习系列——ActionResult扩展

    首先,MVC扩展性非常强. 我从ActionResult扩展入手,因为我们知道微软ActionResult和其子类,有时候并不能满足所有返回值. 比如:我需要返回XML. 因此,现在我扩展XMLRes ...

  6. Qt实现指定线程执行回调

    说明 同线程时,直接调用回调(block参数没意义) 创建invoker所在的线程,需要有Qt的消息循环(比如UI线程) 直接上代码 typedef std::function<void()&g ...

  7. angular 服务

    在Angular里面,services作为单例对象在需要到的时候被创建,只有在应用生命周期结束的时候(关闭浏览器)才会被清除.而controllers在不需要的时候就会被销毁了.服务用于在控制器之间进 ...

  8. mongodb 的安装历程

    mongo 安装历程 mongo的安装方法有千万种,只有一种让我觉得还不错,说说安装过程中的一点心德. 方法一:源码安装,千万别用这种方法,尼马我用虚拟机编译了一下午,竟然没有编译完,强制关机,第二天 ...

  9. Ext学习-前后交互模式介绍

    在前后台交互模式的介绍中,实际上就是Store中Proxy相关的内容,比如Ajax提交. 所以详细的文档请参考: Ext学习-基础概念,核心思想介绍   中关于数据模型和MVC结构部分. 作者:sdj ...

  10. ASP.NET MVC 4 插件化架构简单实现-实例篇

    先回顾一下上篇决定的做法: 1.定义程序集搜索目录(临时目录). 2.将要使用的各种程序集(插件)复制到该目录. 3.加载临时目录中的程序集. 4.定义模板引擎的搜索路径. 5.在模板引擎的查找页面方 ...