关于error:Cannot assign to 'self' outside of a method in the init family
有时候我们重写父类的init方法时不注意将init后面的第一个字母写成了小写,在这个方法里面又调用父类的初始化方法(self = [super init];)时会报错,错误信息如下:error:Cannot assign to 'self' outside of a method in the init family
原因:只能在init方法中给self赋值,Xcode判断是否为init方法规则:方法返回id,并且名字以init+大写字母开头+其他 为准则。例如:- (id) initWithXXX;
出错代码:- (id) Myinit { self = [super init]; ……}
解决方法:- (id) initWithMy { self = [super init]; }
alloc方法用于实例化一个对象,init用来将对象初始化。
在 View 中有两种方式加载 View :纯代码与XIB(SB)。这里要区分来介绍,因为这两种方式的初始化方法是两套机制。
纯代码
大家应都知道,纯代码创建 View 有两个方法提供选择,分别是:
-(instancetype)init {
self=[superinit];// TODO: 给当前视图做默认设置
NSLog(@" init ");
returnself; }
-(instancetype)initWithFrame:(CGRect)frame {
self=[superinitWithFrame:frame];/
/ TODO: 给当前视图做默认设置
NSLog(@" initWithFrame: ");
returnself; }
那么问题又来了,这两个方法在 View 中,我们应该如何选择呢?两个都要重写,还是只要重写其中一个?带着这个问题,我们创建一个继承于 UIView 的HGInitView。并且将上面的代码copy到 .m 文件中。
我们在其它地方用来年各种功能方式创建:
方式1: HGInitView*hgView=[[HGInitView alloc]initWithFrame:CGRectZero];
打印结果 证明: 仅仅触发了 initWithFrame: 方法。这结果其实没有什么可惊讶的,我们再来看看
方式2。
HGInitView*hgView=[[HGInitView alloc]init];
打印结果 证明: 即触发了 initWithFrame: 方法 又触发了init 方法。问题来了,如果我用方式2来创建视图的话,不久重复了么?
最终结论:
两种方式都能正确的创建试图。
两种方式都会触发initWithFrame:方法。
所以在这里的建议是:在 UIView 的子类中不要重写 init 方法,重写 initWithFrame: 方法就足够了。
View 层总结:
纯代码创建试图,重写 initWithFrame: 方法就足够。
XIB 加载视图,重写awakeFromNib方法就足够。
A.id:万能指针,可以指向任何对象,实质是NSObject的指针,使用的时候不用加上*
B.NSObject中得类方法new
1.完整地创建一个可用对象步骤
(1)分配存储空间 + alloc
(2)初始化 - init
// 1.调用+alloc分配存储空间
Person *p1 = [Person alloc];
// 2.调用-init进行初始化4Person *p2 = [p1 init];
// 同时进行分配存储空间和初始化
Person *p3 = [[Person alloc] init];
构造方法就是init方法
@implementation Person
// 重写-init方法
- (id) init {
// 1.一定要调用super的init方法
self = [super init];// 当前对象self
// 2.如果对象初始化成功,才能进行接下来的子类初始化
if (self != nil)
{// 初始化成功
self.age = 10;
}
// 3.返回一个已经初始化的对象
return self;
}
@end
自定义构造方法
@interface Person : NSObject
@property NSString *name;
/*
自定义构造方法的规范
1.一定是对象方法,一定以-开头
2.返回值一般是id类型
3.方法名一般以init开头
*/
- (id) initWithName:(NSString *) name;
@end
@implementation Person
- (id) initWithName:(NSString *) name {
if (self = [super init]) {
_name = name;
}
return self;
}
@end
createUI方法最好在initWithFrame中调用,外部使用init或initWithFrame均可以正常执行createUI方法。不要在自定义View中同时重写init与initWithFrame并执行相同视图布局代码。会导致布局代码(createUI)执行多次。
对于layoutSubviews的使用我们需要注意以下几点:
1、自定义视图的init方法并不会调用layoutSubviews
2、苹果声明不要直接调用layoutSubviews方法,如果需要更新,应该调用setNeedsLayout方法,视图会在下一次绘制后更新。如果需要立即更新视图,需要执行layoutIfNeeded方法
3、因为layoutSubviews调用比较频繁,因此若无特殊需求(文档所述为执行精确的子视图布局时可使用),不用重写layoutSubviews方法。
Student的alloc方法分配内存,然后再调用init方法初始化对象
* 像init这样用来初始化对象的方法,我们可以称为"构造方法"