当前位置: 首页 > news >正文

Objective-C之成魔之路【9-类构造方法和成员变量作用域、以及变量】

郝萌主倾心贡献,尊重作者的劳动成果。请勿转载。

假设文章对您有所帮助,欢迎给作者捐赠,支持郝萌主,捐赠数额任意,重在心意^_^ 

我要捐赠: 点击捐赠

Cocos2d-X源代码下载:点我传送


构造方法
出于初始化类中的成员变量的须要。 能够提供一个方法用于 此目的, 
这种方法就叫构造方法或构造方法(Constructor)。 
与C++和Java不同。 Objective-C命名是没有限制的, 而且有返回值本身类型指针。
以音乐类举例:
Song.h文件
@interface Song : NSObject {
NSString *title;
NSString *artist;
long int duration;
}
//操作方法
- (void)start;
- (void)stop;
- (void)seek:(long int)time;
//訪问成员变量方法
@property(nonatomic,retain) NSString *title;
@property(nonatomic,retain) NSString *artist;
@property(readwrite) long int duration;
//构造方法
-(Song*) initWithTitle: (NSString *) newTitle
andArtist: (NSString *) newArtist
andDuration:( long int ) newDuration;
@end
在Song类的定义中加入了一个方法。 它一般用 init开头命名, 
它的返回值非常特殊, 是返回值本身类型指针。 而且有返回值本身类型指针。


实现代码例如以下:

@implementation Song
@synthesize title;
@synthesize artist;
@synthesize duration;
//构造方法
-(Song*) initWithTitle: (NSString *) newTitle
andArtist: (NSString *) newArtist
andDuration:(long int) newDuration {
self = [super init];
if ( self ) {
self.title = newTitle;
self.artist = newArtist;
self.duration = newDuration;
}
return self;
}
... ...
@end
代码说明:
1、头文件引用规则
导入的文件要用一对引號引起来,而不是<Foundation/Foundation.h>中的“<”和">"字符。

双引號适用于本地文件(你自己创建的文件),而不是系统文件,这样就通知编译器在哪里可以找到指定的文件。

2、构造方法的实现代码差点儿就是模式代码, 基本上都是例如以下写法: 
-(id)init
{
    self = [super init];
    if (self) {
        //初始化代码
    }
    return self;
}
当中使用 [super init] 来调用父类默认构造方法。

 

对象的初始化的常见的编程习惯是类中全部的初始化方法都以init开头。

假设希望在类对象初始化时做一些事情。能够通过覆写init方法达到这个目的。

这种方法返回的实例对象指派给还有一新个关键词: self。

 

self非常像 C++ 和 Java 的 this。 
3、还有if ( self ) 跟 ( self != nil ) 一样, 是为了确定调用父类构造方法成功返回了一个新对象。 
当初始化变量以后。 用返回self 的方式来获得自己的地址。 
4、父类默认构造方法 -(id) init。 
技术上来说, Objective-C中的构造方法就是一个 "init" 开头的方法。 
而不像 C++ 与Java 有特殊的结构。


5、

必须将父类init方法的运行结果赋值给self,
由于初始化过程改变了对象在内存中的位置(意味着引用将要改变)。

假设父类的初始化过程成功,返回的值是非空的。通过if语句能够验证,
凝视说明能够在这个代码块的位置放入自己定义的初始化代码。
通常能够在这个位置创建并初始化实例变量。

注意

init被定义为返回id类型,这是编写可能会被继承的类init方法的一般规则。

程序開始运行时。它向全部的类发送initialize调用方法。

假设存在一个类及相关的子类,则父类首先得到这条消息。

该消息仅仅向每一个类发送一次。而且向该类发送其他不论什么消息之前,保证向其发送初始化消息。


实例成员变量作用域限定符

在接口中声明的实例变量能够通过子类进行继承。

能够把以下指令放在实例变量之前。以便更精确地控制其作用域:

即便从封装的角度出发, 实例成员变量应该定义为@private, 
但作为一种面向对象的语言, Objective-C支持@public、 @private和@protected作用域限定。 
假设一个实例变量没有不论什么的作用域限定的话, 那么缺省就是@protected。

•@public作用域限定的实例变量,能够被本类、其它类或者模块中定义的方法直接訪问,即可在不论什么情况下訪问。 
•@private作用域限定的实例变量。仅仅能在这个类里面才可以訪问;
在实现部分定义的实例变量默认属于这样的作用域。
•@protected作用域限定的实例变量。 能够在这个类里面和这个类的派生子类里面能够訪问这个变量,
在类外的訪问是不推荐的, 但也能够訪问。

在接口部分定义的实例变量默认是这样的作用域。

@package,对于64位映像。能够在实现该类的映像中的不论什么地方訪问这个实例变量。


@public指令使得其它方法或函数能够通过使用指针运算符(->)訪问实例变量。
但实例变量声明为public并非良好的编程习惯,
由于这违背了数据封装的思想(即一个类须要隐藏它的实例变量)。

以一个简单的样例说明作用域:
訪问定义例如以下:
#import <Foundation/NSObject.h>

@interface Access: NSObject {
@public
	int publicVar;
@private
	int privateVar;
@protected
	int protectedVar;
}

@end
调用的main函数例如以下:
#import <Foundation/Foundation.h>
#import "Access.h"

int main (int argc, const char * argv[]) {

	Access *a = [[Access alloc] init];
	
	a->publicVar = 5;
	NSLog(@"public var: %i\n", a->publicVar);
	
	
	a->protectedVar = 6;
	NSLog(@"protectedVar var: %i\n", a->protectedVar);
	
	
	//不能编译
	//a->privateVar = 10;
	//NSLog(@"private var: %i\n", a->privateVar);
	
    return 0;
}
注意:
@public、 @private和@protected作用域限定仅仅能修饰的实例成员变量, 不能修饰类变量, 更不能修饰方法。



什么是类变量和类方法

再以一个简单的样例说明下:

ClassA.h文件

#import <Foundation/NSObject.h>

static int count;

@interface ClassA: NSObject {
	int aaa;
	
}
+(int) initCount;
+(void) initialize;
@end

ClassA.m文件

#import "ClassA.h"

@implementation ClassA

-(id) init {
	self = [super init];
	count++;
	return self;
}

+(int) initCount {
	return count;
}

+(void) initialize {
	count = 0;
}
@end

代码说明:
init方法是默认构造方法, 
在这个构造方法中累计类变量count, 
在实例方法中能够訪问类变量的, 可是类方法不能訪问实例变量。 
initCount 方法是一个普通的类方法, 用于返回类变量count。 
initialize方法是很特殊的类方法,它是在类第一次訪问时候被自己主动调用 , 
因此它一般用来初始化类变量的。 
类似于C#中的静态构造方法。

调用的main函数
#import <Foundation/Foundation.h>
#import "ClassA.h"

int main( int argc, const char *argv[] ) {
	ClassA *c1 = [[ClassA alloc] init];
	ClassA *c2 = [[ClassA alloc] init];
	
	// print count
	NSLog(@"ClassA count: %i", [ClassA initCount] );
	
	ClassA *c3 = [[ClassA alloc] init];
	NSLog(@"ClassA count: %i", [ClassA initCount] );
	
	[c1 release];
	[c2 release];
	[c3 release];
	
	return 0;
}
代码说明:

在第一次实例化ClassA时候会调用两个方法: 

initialize类方法和实例构造方法init, 

然后再次实例化ClassA时候仅仅是调用实例构造方法init。 而没有调用 initialize类方法。 
这样类变量count被一直累加。 
它隶属类。

因此c1 实例能够訪问, c2和c3都能够訪问。

关于属性、存储方法和实例变量:

编码规范(Xcode4已经採用的)眼下的趋势是使用下划线(_)作为实例变量名的起始字符。

@synthesize window = _window;

表明合成(synthesize)属性window的取值方法和设置方法,

并将属性与实例变量_window(实例变量并没有显性声明)关联起来。

这对差别属性和实例变量的使用是有帮助的。

[window makeKeyAndVisible]; //错误

[_window makeKeyAndVisible];//正确

[self.window makeKeyAndVisible];//正确


全局变量:

在程序的開始处(全部的方法、类定义)编写一下语句:

int gMoveNumber = 0;

那么这个模块中的不论什么位置都能够引用这个变量的值。

这样的情况下,我们说gMoveNumber被定义为全局变量。

依照惯例。用小写的g作为全局变量的首字母。

外部变量是可被其它不论什么方法或函数訪问和更改其值的变量。

在须要訪问外部变量的模块中,变量声明和普通方式一样。仅仅是须要在声明前加上keywordextern。


使用外部变量时。必须遵循以下这条重要的原则:变量必须定义在源文件里的某个位置。

即在全部的方法和函数之外声明变量,而且前面不加keywordextern。如:int gMoveNumber;

确定外部变量的另外一种方式是在全部的函数之外声明变量。

在声明前面加上keywordextern,同一时候显式地为变量指派初始值。


记住。声明不会引起分配变量的内存空间,而定义会引起变量内存空间分配。

处理外部变量时,变量能够在很多地方声明为extern,可是仅仅能定义一次。

注意,假设变量定义在包括訪问该变量的文件里,那么不须要单独进行extern声明。


静态变量:

在方法之外定义的变量不仅是全局变量,并且是外部变量。

假设希望定义全局变量且仅仅在特定模块(文件)中是全局的,就能够使用static来修饰。


注意。重载alloc并非好的编程实践,由于这种方法处理内存的物理分配。


枚举数据类型:

枚举数据类型的定义以keywordenum开头。之后是枚举数据类型的名称,然后是标识符序列(包括在一对花括号内),它们定义了能够给该类型指派的所以的同意值。

在代码中定义的枚举类型的作用域限于块的内部。

另外,在程序的開始及全部块之外定义的枚举数据类型对于该文件是全局的。

定义枚举数据类型时,必须确保枚举标识符与定义在同样作用域之内的变量名和其它标识符不同。

相关文章:

  • MyBatis学习总结(11)——MyBatis动态Sql语句
  • Android应用正确使用扩展SD卡,特别是安卓4.4以后的版本
  • 开源HYBUnicodeReadable日志显示Unicode中文
  • CKEditor在线编辑器增加一个自定义插件
  • 【转载】Tomcat 7.0.3x 启动慢并且遇到StackOverflowError的异常的解决办
  • ubuntu14.04 使用kvm安装win7系统
  • A successful Git branching model
  • ELK日志系统:Elasticsearch + Logstash + Kibana 搭建教程
  • centos制作yum源
  • android在假设绘制自己定义的bitmap,然后返回给ImageView
  • 通过读取excel文件生成sql语句
  • 指针知识梳理5-字符串与指针,程序内存总结
  • 使用360浏览器访问字体逆时针旋转90度的问题
  • Shell脚本实现自动修改IP地址
  • 参数的排列组合
  • 230. Kth Smallest Element in a BST
  • Java新版本的开发已正式进入轨道,版本号18.3
  • JS正则表达式精简教程(JavaScript RegExp 对象)
  • Koa2 之文件上传下载
  • mysql innodb 索引使用指南
  • ng6--错误信息小结(持续更新)
  • Nodejs和JavaWeb协助开发
  • rc-form之最单纯情况
  • Redis在Web项目中的应用与实践
  • uni-app项目数字滚动
  • Webpack入门之遇到的那些坑,系列示例Demo
  • 发布国内首个无服务器容器服务,运维效率从未如此高效
  • 后端_MYSQL
  • 聊聊spring cloud的LoadBalancerAutoConfiguration
  • 七牛云 DV OV EV SSL 证书上线,限时折扣低至 6.75 折!
  • 前言-如何学习区块链
  • 网页视频流m3u8/ts视频下载
  • 微信小程序--------语音识别(前端自己也能玩)
  • 为什么要用IPython/Jupyter?
  • ​sqlite3 --- SQLite 数据库 DB-API 2.0 接口模块​
  • ​Z时代时尚SUV新宠:起亚赛图斯值不值得年轻人买?
  • #LLM入门|Prompt#1.7_文本拓展_Expanding
  • $redis-setphp_redis Set命令,php操作Redis Set函数介绍
  • (cos^2 X)的定积分,求积分 ∫sin^2(x) dx
  • (附源码)spring boot网络空间安全实验教学示范中心网站 毕业设计 111454
  • (附源码)spring boot智能服药提醒app 毕业设计 102151
  • (太强大了) - Linux 性能监控、测试、优化工具
  • (一)80c52学习之旅-起始篇
  • (转)AS3正则:元子符,元序列,标志,数量表达符
  • (转)Linux NTP配置详解 (Network Time Protocol)
  • (转载)在C#用WM_COPYDATA消息来实现两个进程之间传递数据
  • (轉貼) UML中文FAQ (OO) (UML)
  • .NET / MSBuild 扩展编译时什么时候用 BeforeTargets / AfterTargets 什么时候用 DependsOnTargets?
  • .Net CoreRabbitMQ消息存储可靠机制
  • .net MVC中使用angularJs刷新页面数据列表
  • .NET 除了用 Task 之外,如何自己写一个可以 await 的对象?
  • .NET 的静态构造函数是否线程安全?答案是肯定的!
  • .net 后台导出excel ,word
  • .NET 回调、接口回调、 委托
  • .NET(C#) Internals: as a developer, .net framework in my eyes