synchronized 编译记录
2021-10-14 本文已影响0人
_一叶孤帆
源码
#import <Foundation/Foundation.h>
int main(int argc, const char * argv[]) {
@autoreleasepool {
// insert code here...
NSLog(@"Hello, World!");
NSObject *objc = [[NSObject alloc] init];
@synchronized (objc) {
NSLog(@"加了锁的 Log ");
}
}
return 0;
}
通过 xcrun -sdk iphoneos clang -arch arm64 -rewrite-objc main.m -o mainC.cpp
编译 cpp 文件
int main(int argc, const char * argv[]) {
/* @autoreleasepool */ { __AtAutoreleasePool __autoreleasepool;
NSObject *objc = ((NSObject *(*)(id, SEL))(void *)objc_msgSend)((id)((NSObject *(*)(id, SEL))(void *)objc_msgSend)((id)objc_getClass("NSObject"), sel_registerName("alloc")), sel_registerName("init"));
{ id _rethrow = 0; id _sync_obj = (id)objc; objc_sync_enter(_sync_obj);
try {
struct _SYNC_EXIT {
_SYNC_EXIT(id arg) : sync_exit(arg) {}
~_SYNC_EXIT() {objc_sync_exit(sync_exit);}
id sync_exit;
} _sync_exit(_sync_obj);
NSLog((NSString *)&__NSConstantStringImpl__var_folders_dv_nb1mrprn3kj3cnk3tts4918w0000gn_T_main_352584_mi_1);
} catch (id e) {_rethrow = e;}
{ struct _FIN { _FIN(id reth) : rethrow(reth) {}
~_FIN() { if (rethrow) objc_exception_throw(rethrow); }
id rethrow;
} _fin_force_rethow(_rethrow);}
}
}
return 0;
}
解析
truct _SYNC_EXIT {
_SYNC_EXIT(id arg) : sync_exit(arg) {}
~_SYNC_EXIT() {objc_sync_exit(sync_exit);}
id sync_exit;
} _sync_exit(_sync_obj);
声明了一个结构体 _SYNC_EXIT
_SYNC_EXIT(id arg) : sync_exit(arg) {} 是这个结构体的构造函数,sync_exit(arg) 是用arg 来初始化 sync_exit 这个成员
~_SYNC_EXIT() {objc_sync_exit(sync_exit);} 是这个对象的析构函数, 在释放时会执行 objc_sync_exit
_sync_exit(_sync_obj) 是直接声明了一个 _sync_exit 变量并用 _sync_obj 初始化
NSLog((NSString *)&__NSConstantStringImpl__var_folders_dv_nb1mrprn3kj3cnk3tts4918w0000gn_T_main_352584_mi_1); 为什么在析构函数下面,因为 _sync_exit 对象是在释放的时候销毁的所以只有这个代码块执行完才会调用 ~_SYNC_EXIT() 然后才会执行 objc_sync_exit(sync_exit)
所以整体执行顺序为 objc_sync_enter(_sync_obj) -> NSLog -> objc_sync_exit(sync_exit)
所以 synchronized 实现可以断定为调用
objc_sync_enter
objc_sync_exit
可通过 objc 源码查看 objc_sync_enter/objc_sync_exit 函数