文章目录
1、写一个类不想他被子类继承 【修饰类】
__attribute((objc_subclassing_restricted))__
#import <UIKit/UIKit.h>
#import <objc/runtime.h>
NS_ASSUME_NONNULL_BEGIN
__attribute__((objc_subclassing_restricted))//禁止该类被继承
@interface AAA : UIView
-(void)mustUseMethod;
@end
NS_ASSUME_NONNULL_END
2、提示子类必须调用父类方法【修饰方法】
__attribute__((objc_requires_super))
在父类的方法后面添加,那么子类调用该方法必须实现 [super thisMethod],否则会黄色警告
@interface Father : NSObject
- (void)mustUseMethod __attribute__((objc_requires_super));
@end
@interface Child : TestObject
@end
@implementation BBB
-(void)mustUseMethod{
//[super mustUseMethod];
警告信息:(不报错)
Method possibly missing a [super mustUseMethod] call
}
@end
3、constructor / destructor 【修饰构造/析构器】
Objective-C最后还是转译成C语言,当然还有constructor / destructor
加上这两个属性的函数会在分别在可执行文件(或 shared library)
load
和unload
时被调用,可以理解为在 main()函数调用前
和 return 后执行。 实际上constructor会在+load
之后执行, 因为 dyld(动态链接器)最开始会先通知 objc runtime 去加载其中所有的类,每加载一个类时,它的+load
随之调用,全部加载完成后,dyld 才会调用所有的constructor
方法
__attribute__((constructor)) static void beforeMain() {
NSLog(@"before main");
}
__attribute__((destructor)) static void afterMain() {
NSLog(@"after main");
}
int main(int argc, const char * argv[]) {
@autoreleasepool {
NSLog(@"execute main");
}
return 0;
执行结果:
debug-objc[23391:1143291] before main
debug-objc[23391:1143291] execute main
debug-objc[23391:1143291] after main
constructor后边添加设置优先级,控制执行顺序
__attribute__((constructor(101))) static void beforeMain() {
NSLog(@"before main");
}
__attribute__((constructor(100))) static void beforeMain1() {
NSLog(@"before main1");
}
__attribute__((destructor)) static void afterMain() {
NSLog(@"after main");
}
执行结果:
2019-10-12 15:48:24.015908+0800 Food[4561:144694] before main1
2019-10-12 15:48:24.016491+0800 Food[4561:144694] before main
2019-10-12 15:48:24.017145+0800 Food[4561:144694] execute main
4、overloadable 可以允许同名函数的产生【修饰函数】
__attribute__((overloadable)) void testMethod(int age) {
NSLog(@"%@", @(age));};
__attribute__((overloadable)) void testMethod(NSString *name) {
NSLog(@"%@", name);};
__attribute__((overloadable)) void testMethod(BOOL gender) {
NSLog(@"%@", @(gender));};
int main(int argc, char * argv[]) {
@autoreleasepool {
testMethod(18);
testMethod(@"lxz");
testMethod(YES);
return UIApplicationMain(argc, argv, nil, NSStringFromClass([AppDelegate class]));
}
}
执行结果:
2019-10-12 16:01:04.491366+0800 Food[4724:150619] 18
2019-10-12 16:01:04.491510+0800 Food[4724:150619] lxz
2019-10-12 16:01:04.491632+0800 Food[4724:150619] 1
5、objc_runtime_name属性可以在编译时,将Class或者Protocol指定为另一个名字【修饰类】
__attribute__((objc_runtime_name("TestObject")))
@interface AAA : NSObject
@end
int main(int argc, char * argv[]) {
@autoreleasepool {
NSLog(@"--%@",NSStringFromClass([AAA class]));
NSLog(@"----%@",NSClassFromString(@"TestObject"));
}
}
6、通过cleanup属性,可以指定给一个变量,当变量释放之前去执行一个函数【修饰变量】
指定函数的执行,是在dealloc之前。在指定的函数中,可以传入一个形参,参数就是cleanup修饰的变量,形参是一个地址。
static void releaseBefore(NSObject **object) {
NSLog(@"AAA-------%@", *object);
}
int main(int argc, char * argv[]) {
@autoreleasepool {
{
AAA *object __attribute__((cleanup(releaseBefore))) = [AAA new];
}
return UIApplicationMain(argc, argv, nil, NSStringFromClass([AppDelegate class]));
}
}
执行结果:
2019-10-12 16:20:24.230282+0800 Food[5057:162194] AAA-------<AAA: 0x6000021c81d0>
2019-10-12 16:20:24.230633+0800 Food[5057:162194] AAA-dealloc
7、通过 unused 消除unused xxx警告【修饰变量】
如果某个变量未使用,会提示unused xxx,可以通过unused消除这个警告
int main(int argc, char * argv[]) {
@autoreleasepool {
AAA *object __attribute__((unused)) = [AAA new];
return UIApplicationMain(argc, argv, nil, NSStringFromClass([AppDelegate class]));
}
}