【Objective-C】07-自定义构造方法和description方法

2024-08-29 23:38

本文主要是介绍【Objective-C】07-自定义构造方法和description方法,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

知识回顾

在第5讲中已经介绍了如何定义类和创建并初始化对象,比如有Student这个类

1.Student.h

复制代码
1 #import <Foundation/Foundation.h>
2 
3 @interface Student : NSObject {
4     int _age;
5 }
6 - (void)setAge:(int)age;
7 - (int)age;
8 @end
复制代码

 

2.Student.m

复制代码
 1 #import "Student.h"
 2 
 3 @implementation Student
 4 - (void)setAge:(int)age {
 5     _age = age;
 6 }
 7 - (int)age {
 8     return _age;
 9 }
10 @end
复制代码

 

3.在main函数中创建一个Student对象

复制代码
 1 #import "Student.h"
 2 
 3 int main(int argc, const char * argv[])
 4 {
 5 
 6     @autoreleasepool {
 7         Student *stu = [[Student alloc] init];
 8         
 9         stu.age = 10;
10         
11         [stu release];
12     }
13     return 0;
14 }
复制代码

* 在第7行调用Student的alloc方法分配内存,然后再调用init方法初始化对象

* 像init这样用来初始化对象的方法,我们可以称为"构造方法"

 

回到顶部

一、自定义构造方法

默认的构造方法,也就是init方法,它是不接收任何参数的。因此,在实际开发中,为了方便,会经常自定义构造方法。

接下来,自定义一个构造方法,可以传入一个age参数来初始化Student对象

1.在Student.h中添加方法声明

- (id)initWithAge:(int)age;

* 构造方法的方法名一般都会以init开头,返回值跟init方法一样为id类型,id可以代表任何OC对象

* 这个构造方法接收一个int类型的age参数,目的是在初始化Student对象时,顺便设置成员变量_age的值

 

2.在Student.m中实现构造方法

复制代码
1 - (id)initWithAge:(int)age {
2     self = [super init];
3     if (self != nil) {
4         _age = age;
5     }
6     return self;
7 }
复制代码

* 跟Java一样,构造方法内部首先要调用父类的构造方法,在第2行调用了父类的init方法,它会返回初始化好的Student对象,这里把返回值赋值给了self,self代表Student对象本身

* 第3~5行的意思是:如果self不为nil,也就是初始化成功,就给成员变量_age进行赋值

* 最后返回初始化过后的self,整个构造方法就结束了

 

3.简化构造方法

由于C语言和OC的语法特性,我们可以对构造方法进行简化,先简化第3行

复制代码
1 - (id)initWithAge:(int)age {
2     self = [super init];
3     if (self) {
4         _age = age;
5     }
6     return self;
7 }
复制代码

* 第3行的 if(self) 跟 if(self!=nil) 是等价的

* 还可以将第2、3行合并,继续简化

复制代码
1 - (id)initWithAge:(int)age {
2     if ( self = [super init] ) {
3         _age = age;
4     }
5     return self;
6 }
复制代码

* 第2行的总体意思是:先调用父类的构造方法init,然后将返回值赋值给self,接着判断self是否为nil

* 以后的构造方法都这样写了

 

4.调用构造方法

1 Student *stu = [[Student alloc] initWithAge:10];
2 
3 NSLog(@"age is %i", stu.age);
4 
5 [stu release];

* 在第1行调用了构造方法initWithAge:,并传入10作为参数,因此Student对象的成员变量_age会变为10

* 在第3行打印Student的成员变量_age,打印结果:

2013-04-19 21:36:47.880 构造方法[448:303] age is 10

 

回到顶部

二、description方法

1.NSLog回顾

众所周知,我们可以用NSLog函数来输出字符串和一些基本数据类

1 int age = 11;
2 NSLog(@"age is %i", age);

* 第2行的%i代表会输出一个整型数据,右边的变量age会代替%i的位置进行输出

* 输出结果:

2013-04-19 21:43:47.674 构造方法[483:303] age is 11

 

2.NSLog输出OC对象

其实,除了可以输出基本数据类型,NSLog函数还可以输出任何OC对象

1 Student *stu = [[Student alloc] initWithAge:10];
2 
3 NSLog(@"%@", stu);
4 
5 [stu release];

* 在第3行用NSLog函数输出stu对象,注意左边的格式符%@,以后想输出OC对象,就得用%@这个格式符

* NSLog函数一旦发现用%@输出某个OC对象时,就会调用这个对象的description方法(这个方法返回值是NSString类型,是OC中的字符串类型),并且将description方法返回的字符串代替%@的位置进行输出

* description方法的默认实现是返回这样的格式:<类名: 对象的内存地址>,因此上面代码的输出结果为:

2013-04-19 21:46:49.896 构造方法[492:303] <Student: 0x100109910>

Student是类名,0x100109910是对象的内存地址

* 注意了,%@只能用于输出OC对象,不能输出结构体等其他类型

* 有Java开发经验的人应该能感受到OC中的description方法就是Java中的toString方法

 

3.重写description方法

description方法的默认实现是返回类名和对象的内存地址,这样的话,使用NSLog输出OC对象,意义就不是很大,因为我们并不关心对象的内存地址,比较关心的是对象内部的一些成变量的值。因此,会经常重写description方法,覆盖description方法的默认实现

比如,重写Student的description方法,返回成员变量_age的值

1 - (NSString *)description {
2     return [NSString stringWithFormat:@"age=%i", _age];
3 }

* 在第2行调用了NSString这个类的静态方法stringWithFormat初始化一个字符串对象,并返回这个字符串

* 如果你会使用NSLog的话,那就应该能理解第2行的方法参数是什么意思了

* 假如_age是10,那么description方法返回的字符串就是@"age=10"

* 可能有人会觉得奇怪,之前创建的Student对象是需要释放的,为什么这里创建的字符串对象不用释放?要想彻底明白这个问题,需要先了解OC的内存管理,这里我们暂不做详细讨论,后面会有章节详细讨论内存管理。你可以先记住一个规则:一般情况下,静态方法返回的对象,都不用手动释放。

* 重写完description方法后,再次执行下面的代码

1 Student *stu = [[Student alloc] initWithAge:10];
2 
3 NSLog(@"%@", stu);
4 
5 [stu release];

输出结果为:

2013-04-19 22:09:56.625 构造方法[531:303] age=10

 

4.description方法的陷阱

千万不要在description方法中同时使用%@和self,下面的写法是错误的:

1 - (NSString *)description {
2     return [NSString stringWithFormat:@"%@", self];
3 }

第2行同时使用了%@和self,代表要调用self的description方法,因此最终会导致程序陷入死循环,循环调用description方法

这篇关于【Objective-C】07-自定义构造方法和description方法的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1119174

相关文章

Maven 配置中的 <mirror>绕过 HTTP 阻断机制的方法

《Maven配置中的<mirror>绕过HTTP阻断机制的方法》:本文主要介绍Maven配置中的<mirror>绕过HTTP阻断机制的方法,本文给大家分享问题原因及解决方案,感兴趣的朋友一... 目录一、问题场景:升级 Maven 后构建失败二、解决方案:通过 <mirror> 配置覆盖默认行为1. 配置示

SpringBoot排查和解决JSON解析错误(400 Bad Request)的方法

《SpringBoot排查和解决JSON解析错误(400BadRequest)的方法》在开发SpringBootRESTfulAPI时,客户端与服务端的数据交互通常使用JSON格式,然而,JSON... 目录问题背景1. 问题描述2. 错误分析解决方案1. 手动重新输入jsON2. 使用工具清理JSON3.

使用jenv工具管理多个JDK版本的方法步骤

《使用jenv工具管理多个JDK版本的方法步骤》jenv是一个开源的Java环境管理工具,旨在帮助开发者在同一台机器上轻松管理和切换多个Java版本,:本文主要介绍使用jenv工具管理多个JD... 目录一、jenv到底是干啥的?二、jenv的核心功能(一)管理多个Java版本(二)支持插件扩展(三)环境隔

Java中Map.Entry()含义及方法使用代码

《Java中Map.Entry()含义及方法使用代码》:本文主要介绍Java中Map.Entry()含义及方法使用的相关资料,Map.Entry是Java中Map的静态内部接口,用于表示键值对,其... 目录前言 Map.Entry作用核心方法常见使用场景1. 遍历 Map 的所有键值对2. 直接修改 Ma

Mybatis Plus Join使用方法示例详解

《MybatisPlusJoin使用方法示例详解》:本文主要介绍MybatisPlusJoin使用方法示例详解,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,... 目录1、pom文件2、yaml配置文件3、分页插件4、示例代码:5、测试代码6、和PageHelper结合6

Java中实现线程的创建和启动的方法

《Java中实现线程的创建和启动的方法》在Java中,实现线程的创建和启动是两个不同但紧密相关的概念,理解为什么要启动线程(调用start()方法)而非直接调用run()方法,是掌握多线程编程的关键,... 目录1. 线程的生命周期2. start() vs run() 的本质区别3. 为什么必须通过 st

如何自定义一个log适配器starter

《如何自定义一个log适配器starter》:本文主要介绍如何自定义一个log适配器starter的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录需求Starter 项目目录结构pom.XML 配置LogInitializer实现MDCInterceptor

C#之List集合去重复对象的实现方法

《C#之List集合去重复对象的实现方法》:本文主要介绍C#之List集合去重复对象的实现方法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录C# List集合去重复对象方法1、测试数据2、测试数据3、知识点补充总结C# List集合去重复对象方法1、测试数据

SpringBoot读取ZooKeeper(ZK)属性的方法实现

《SpringBoot读取ZooKeeper(ZK)属性的方法实现》本文主要介绍了SpringBoot读取ZooKeeper(ZK)属性的方法实现,强调使用@ConfigurationProperti... 目录1. 在配置文件中定义 ZK 属性application.propertiesapplicati

MyBatis设计SQL返回布尔值(Boolean)的常见方法

《MyBatis设计SQL返回布尔值(Boolean)的常见方法》这篇文章主要为大家详细介绍了MyBatis设计SQL返回布尔值(Boolean)的几种常见方法,文中的示例代码讲解详细,感兴趣的小伙伴... 目录方案一:使用COUNT查询存在性(推荐)方案二:条件表达式直接返回布尔方案三:存在性检查(EXI