__attribute__

懒得找出处了~!

    In fact, when __attribute__ was first introduced to GCC, it was faced with some resistance by some who suggested that #pragma be used exclusively for the same purposes.

    There were, however, two very good reasons why __attribute__ was added:

    It was impossible to generate #pragma commands from a macro (before the C99 _Pragma operator).
    There is no telling what the same #pragma might mean in another compiler.
    Quoth the GCC Documentation for Function Attributes:

    These two reasons applied to almost any application that might have been proposed for #pragma. It was basically a mistake to use #pragma for anything.

    Indeed, if you look at modern Objective-C–in the headers of Apple frameworks and well-engineered open-source projects–__attribute__ is used for myriad purposes. (By contrast, #pragma’s main claim to fame these days is decoration: #pragma mark)
  • 子类调用该方法的时候需要用 super
__attribute__((objc_requires_super))
//eg:略
  • 定义的的结构体也可以使用语法糖
__attribute__((objc_boxable))

typedef struct __attribute__((objc_boxable)) {
    CGFloat x, y, width, height;
} YYGRect;

YYGRect rect2 = {1, 2, 3, 4};
NSValue *value2 = @(rect2);      //编译通过

  • 在程序的主函数之前运行
#include 
__attribute__((constructor)) void before_main() { 
   printf("before main\n"); 
} 

__attribute__((destructor)) void after_main() { 
   printf("after main\n"); 
} 

int main(int argc, char **argv) { 
   printf("in main\n"); 
   return 0; 
}
  • 修饰一个变量在该变量作用域结束后,自动调用一个指定的方法
    所谓作用域结束,包括大括号结束、return、goto、break、exception等各种情况。可以修饰的变量不止NSString,包括自定义Class或基本类型都是可以的。以压栈的方式调用,在函数调用dealloc之前执行:
    __attribute__((cleanup))
// 指定一个cleanup方法,注意入参是所修饰变量的地址,类型要一样
// 对于指向objc对象的指针(id *),如果不强制声明__strong默认是
// __autoreleasing,造成类型不匹配
__strong NSString *string __attribute__((cleanup(Show))) = @"Yin Yu 思密达";

...//这里写N多代码,都是最后执行下面的函数

static void Show(__strong NSString **string)
{   
    NSLog(@"代码以为我为标志结束!%@",*string);
}

//eg:

// 函数里面:
// 加了个`unused`的attribute用来消除`unused variable`的warning
__strong void(^block)(void) __attribute__((cleanup(blockCleanUp), unused)) = ^{
        NSLog(@"代码以为我标志结束!");
    };

//所调用的函数
// void(^block)(void)的指针是void(^*block)(void)
static void blockCleanUp(__strong void(^*block)(void)) 
{
    (*block)();
}
//定义宏
#define onExit __strong void(^block)(void) __attribute__((cleanup(blockCleanUp), unused)) = ^

//使用的地方
onExit {
        NSLog(@"代码以为为结束");
    };
//下面~~~~~~~~N多代码



# 所调用的函数
// void(^block)(void)的指针是void(^*block)(void)
static void blockCleanUp(__strong void(^*block)(void)) 
{
    (*block)();
}
    typedef void (^mtl_cleanupBlock_t)();

    #define metamacro_concat_(A, B) A ## B

    #define metamacro_concat(A, B) \
            metamacro_concat_(A, B)

        #define onExit \
        try {} @finally {} \
        __strong mtl_cleanupBlock_t metamacro_concat(mtl_exitBlock_, __LINE__) __attribute__((cleanup(mtl_executeCleanupBlock), unused)) = ^


        + (void)enumeratePropertiesUsingBlock:(void (^)(objc_property_t property, BOOL *stop))block {
        Class cls = self;
        BOOL stop = NO;

        while (!stop && ![cls isEqual:MTLModel.class]) {
            unsigned count = 0;
            objc_property_t *properties = class_copyPropertyList(cls, &count);

            cls = cls.superclass;
            if (properties == NULL) continue;
    //注意这里的用法
            @onExit {
                free(properties);
            };

            for (unsigned i = 0; i < count; i++) {
                block(properties[i], &stop);
                if (stop) break;
            }
        }
    }
    //@onExit 宏展开之后
    @try {} @finally {}
            __strong mtl_cleanupBlock_t mtl_exitBlock___LINE__ __attribute__((cleanup(mtl_executeCleanupBlock), unused)) = ^{
                free(properties);
            };
//可以保证 程序在即将运行出 propertties的作用时释放  properties
  • 定义若干函数,函数名字相同但是方法不同,编译器能够自动识别调用的函数
__attribute__((overloadable)) void logAnything(id arr) 
{
    NSLog(@"%@", arr);
}
__attribute__((overloadable)) void logAnything(int num) 
{
    NSLog(@"%@", @(num));
}
__attribute__((overloadable)) void logAnything(CGRect rect)
{
    NSLog(@"%@", NSStringFromCGRect(rect));
}
#调用
//传递数值
    logAnything(@[@"hello",@"world"]);  
//int 数值 
    logAnything(66);                               
//Rect
    logAnything(CGRectMake(0, 0, 0, 0));
  • 混淆
_attribute__((objc_runtime_name("FalseEunuch")))    
//相当于给Eunuch起个学名
@interface Eunuch : NSObject

+(void)show;

@end
//示例:
 NSLog(@"%@",[Eunuch class]);
//得到的打印消息
//2016-08-03 14:16:46.882 0803-黑魔法[24329:1921649] FalseEunuch
  • attribute((availability))
- (CGSize)sizeWithFont:(UIFont *)font NS_DEPRECATED_IOS(2_0, 7_0, "Use -sizeWithAttributes:") __TVOS_PROHIBITED;
//来看一下 后边的宏
#define NS_DEPRECATED_IOS(_iosIntro, _iosDep, ...) CF_DEPRECATED_IOS(_iosIntro, _iosDep, __VA_ARGS__)

define CF_DEPRECATED_IOS(_iosIntro, _iosDep, ...) __attribute__((availability(ios,introduced=_iosIntro,deprecated=_iosDep,message="" __VA_ARGS__)))
//宏展开以后如下
__attribute__((availability(ios,introduced=2_0,deprecated=7_0,message=""__VA_ARGS__)));
//iOS即是iOS平台
//introduced 从哪个版本开始使用
//deprecated 从哪个版本开始弃用
//警告的消息
//其实还可以再加一个参数例如
void f(void) __attribute__((availability(macosx,introduced=10.4,deprecated=10.6,obsoleted=10.7)));
//obsoleted完全禁止使用的版本

你可能感兴趣的:(__attribute__)