携手创作,一起成长!这是我参与「日新方案 8 月更文应战」的第14天,点击查看活动概况
导言
在 C 语言中,程序内变量或函数的效果域和寿数是由其存储类确认的,比如static、extern。当 static 使得一个特定的文件中的函数和变量大局可见,extern 则使它们对一切文件可见。
I 运用static结合线程安全形式来创立同享实例
单例方针应该运用线程安全形式来创立同享实例。
+ (instancetype)sharedInstance {
static id sharedInstance = nil;
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{ sharedInstance = [[self alloc] init]; });
return sharedInstance;
}
1.1 运用场景
- 将登陆之后的token信息作为单利方针的属性进行存储
- 存储一些本次app内存销毁的变量,比如操控接口的恳求标志、蓝牙打印机的衔接信息。
/**
>登录账号得到的token信息。最好不要作为一个独立的单利方针存储;而是将它作为单例方针的属性userInfo,这样便于切换账号存储token和其他账号信息
*/
@property (strong, nonatomic) UserInfoModel *userInfo;
/**
操控接口的恳求标志
*/
@property (assign, nonatomic) BOOL IsreqGetCurrentSysUsering;
/**
蓝牙打印机的衔接信息
*/
@property(nonatomic,strong)CBPeripheral *SelectPeripheral;
运用事例
kunnan.blog.csdn.net/article/det…
1.2 单例形式的根本完成
- 一个类只要一个方针
/**
单例形式:一个类只要一个方针
*/
@implementation KNMusicTool
extern id _musicTool;//大局变量
/**
alloc办法内部会调用这个办法
*/
+ (instancetype)allocWithZone:(struct _NSZone *)zone{
NSLog(@"%s",__func__);
@synchronized(self) {//多线程
//创立方针的内存空间
if (nil == _musicTool) {
_musicTool = [super allocWithZone:zone];
}
}
return _musicTool;
}
/** 便利拜访单例方针*/
+ (instancetype)sharedMusicTool{
@synchronized(self) {
//操作同享资源
if (nil == _musicTool) {
_musicTool = [[self alloc]init];
}
}
return _musicTool;
}
/** NSLog(@"%@",[[[HLMusicTool alloc]init] copy]);//Returns the object returned by copyWithZone:. copyWithZone 恪守协议NSCopying
*/
- (id)copyWithZone:(nullable NSZone *)zone{
return _musicTool;
}
1.3 单例形式的完善
问题1: extern 对大局变量的引证 会在全程序中查找 _musicTool,在其他类引证大局变量,其他类就可以修正大局变量的值,导致单例方针存在被修正的危险。
extern id _musicTool; //引证大局变量,会在全程序中查找 _musicTool,在其他类引证大局变量,其他类就可以修正大局变量的值,导致单例方针存在被修正的危险。
// 解决办法:static 润饰的大局变量,这样其他类就无法引证
解决办法:运用static进行润饰,效果域仅限于当时的文件。
static id _musicTool;
问题2: 屡次加锁
+ (instancetype)sharedMusicTool{
@synchronized(self) {
//操作同享资源
if (nil == _musicTool) {
_musicTool = [[self alloc]init];
}
}
return _musicTool;
}
解决: 只要在确实需求创立方针的时分,才进行加锁。
if (nil == _musicTool) {//加锁之前先进行是否满足创立方针的条件
@synchronized(self) {
//操作同享资源
if (nil == _musicTool) {//避免创立屡次
_musicTool = [[self alloc]init];
}
}
}
例子:运用static替代extern,加锁之前先进行是否满足创立方针的条件。
/* static 润饰变量:
1)static的局部变量:确保只初始化一次,在程序运转过程中只要一份内容;--局部变量的生命周期和大局变量相似,可是不能改动效果域
2)static 润饰的大局变量:不允许本类的h文件拜访。即效果域仅限于当时的文件
*/
static id _musicTool;//大局变量
/**
alloc办法内部会调用这个办法
*/
+ (instancetype)allocWithZone:(struct _NSZone *)zone{
NSLog(@"%s",__func__);
if (nil == _musicTool) {
@synchronized(self) {//多线程
//创立方针的内存空间
if (nil == _musicTool) {
_musicTool = [super allocWithZone:zone];
}
}
}
return _musicTool;
}
/** 便利放回单例方针*/
+ (instancetype)sharedMusicTool{
if (_musicTool == nil) {//避免频频加锁
//加锁
@synchronized(self) {
//操作同享资源
if (nil == _musicTool) {//避免创立屡次
_musicTool = [[self alloc]init];
}
}
}
return _musicTool;
}
/** NSLog(@"%@",[[[HLMusicTool alloc]init] copy]);//Returns the object returned by copyWithZone:. copyWithZone 恪守协议NSCopying
*/
- (id)copyWithZone:(nullable NSZone *)zone{
return _musicTool;
}
1.4 饿汉式
/*
ninitailize、load办法的区别:
initailize、load都是类办法
当一个类 或者分类被装载进内存时,就会调用一次load办法(当时这个类还不可用)
当第一次运用这个类时,就会调用一次initailize办法
*/
/** Invoked whenever a class or category is added to the Objective-C runtime; 因此不存在多线程问题*/
+ (void)load{
_instance = [[KNSoundTool alloc]init];
NSLog(@"%s----instance%@",__func__,_instance);
}
//+ (void)initialize{
// NSLog(@"%s",__func__);
//}
+ (instancetype)allocWithZone:(struct _NSZone *)zone{
if (_instance == nil) {
_instance = [super allocWithZone:zone];
}
return _instance;
}
- (id)copyWithZone:(NSZone *)zone{
return _instance;
}
+(instancetype)shareSoundTool{
return _instance;
}
II 运用GCD结合宏来完成单例
运用static结合线程安全形式dispatch_once
来创立同享实例,并运用条件编译#if进行ARC、MRC的适配。
缺陷:宏界说的代码欠好调试
2.1 ARC 环境下的GCD完成单例
/*
GCD 完成单例
*/
static id _dataTool;
+ (instancetype)shareDataTool{
//GCD
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{//一次性代码
_dataTool = [[self alloc]init];
});
return _dataTool;
}
+ (instancetype)allocWithZone:(struct _NSZone *)zone{
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
_dataTool = [super allocWithZone:zone];
});
return _dataTool;
}
- (id)copyWithZone:(NSZone *)zone{
return _dataTool;
}
2.2 非ARC的单例形式
- release重写,进行阻止单例方针的开释
@implementation HSDataTool
static id _dataTool;
+(instancetype)shareDataTool{
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
_dataTool = [[HSDataTool alloc]init];
});
return _dataTool;
}
- (id)copyWithZone:(NSZone *)zone{
return _dataTool;
}
+ (instancetype)allocWithZone:(struct _NSZone *)zone{
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
_dataTool = [super allocWithZone:zone];
NSLog(@"%s",__func__);
});
return _dataTool;
}
#pragma mark - MRC环境的适配:重写
/*
Decrements the receiver’s reference count.
The receiver is sent a dealloc message when its reference count reaches 0.
*/
- (oneway void)release{
}
- (instancetype)retain{
return self;
}
- (NSUInteger)retainCount{
return 1;
}
2.3 GCD和宏来完成单例的详细代码
- 单例内容
#ifndef HSSingleton_h
#define HSSingleton_h
//头文件的单例内容
#define HSSingletonH(classname) +(instancetype)share##classname
//.m文件的单例代码
#define HSSingletonM(classname) \
static id _instance;\
+(instancetype)share##classname{\
static dispatch_once_t onceToken;\
dispatch_once(&onceToken, ^{\
_instance = [[self alloc]init];\
});\
return _instance;\
}\
- (id)copyWithZone:(NSZone *)zone{\
return _instance;\
}\
+ (instancetype)allocWithZone:(struct _NSZone *)zone{\
static dispatch_once_t onceToken;\
dispatch_once(&onceToken, ^{\
_instance = [super allocWithZone:zone];\
});\
return _instance;\
}\
- (oneway void)release{\
}\
- (instancetype)retain{\
return self;\
}\
- (NSUInteger)retainCount{\
return 1;\
}
#endif /* HSSingleton_h */
2.4 ARC、MRC的适配(条件编译)
//头文件的单例内容
#define HSSingletonH(classname) +(instancetype)share##classname
//.m文件的单例代码
#if __has_feature(objc_arc)
#define HSSingletonM(classname) \
static id _instance;\
+(instancetype)share##classname{\
static dispatch_once_t onceToken;\
dispatch_once(&onceToken, ^{\
_instance = [[self alloc]init];\
});\
return _instance;\
}\
- (id)copyWithZone:(NSZone *)zone{\
return _instance;\
}\
+ (instancetype)allocWithZone:(struct _NSZone *)zone{\
static dispatch_once_t onceToken;\
dispatch_once(&onceToken, ^{\
_instance = [super allocWithZone:zone];\
});\
return _instance;\
}
#else
#define HSSingletonM(classname) \
static id _instance;\
+(instancetype)share##classname{\
static dispatch_once_t onceToken;\
dispatch_once(&onceToken, ^{\
_instance = [[self alloc]init];\
});\
return _instance;\
}\
- (id)copyWithZone:(NSZone *)zone{\
return _instance;\
}\
+ (instancetype)allocWithZone:(struct _NSZone *)zone{\
static dispatch_once_t onceToken;\
dispatch_once(&onceToken, ^{\
_instance = [super allocWithZone:zone];\
});\
return _instance;\
}\
- (oneway void)release{\
}\
- (instancetype)retain{\
return self;\
}\
- (NSUInteger)retainCount{\
return 1;\
}\
- (instancetype)autorelease{\
return self;\
}
#endif
III 运用extern声明公共办法、大局字符串常量
3.1 大局字符串常量
大局的字符串常量替代宏常量,节约内存空间。
blog.csdn.net/z929118967/…
运用场景:在声明比如 userInfo 字典,NSNotification 称号和 NSError 域的时分。
/// The domain for errors originating within `RACCommand`.
extern NSErrorDomain const RACCommandErrorDomain;
typedef NS_ERROR_ENUM(RACCommandErrorDomain, RACCommandError) {
/// -execute: was invoked while the command was disabled.
RACCommandErrorNotEnabled = 1,
};
完成办法:在公共头文件里声明一个 extern 的 NSString * const,并在完成文件里界说该 NSString * const:
- 公共头文件: Consts.h
extern NSString * _Nonnull const KNClientId;//
//key 存储到 NSUserDefaults 里(类型为 NSDictionary<NSString *, NSNumber *> *)
- 完成文件:界说大局字符串常量
#define KNClientId @""//宏会在编译时,将一切引证宏变量的当地,进行值的替换,造成许多相同的临时字面量,糟蹋内存
NSString * const KNClientId = @"";// 大局的const常量替代宏常量,节约内存空间。内存只要一份
运用字符串常量来替代宏的运用:
- 界说const 大局常量 ,确保只在一处界说,多处进行引证。
- 大局的const常量替代宏常量,节约内存空间(内存只要一份)。
当 static 使得一个特定的文件中的函数和变量大局可见,extern 则使它们对一切文件可见。
3.2 公共办法
运用场景:
- 仅提供辅佐而与详细状态无关的办法:枚举类型转字符、蓝牙衔接状态的判别、转化方针经纬度为高德坐标系、是否为海外用户、铲除缓存
//TransactionStateMachine.h
extern NSString * NSStringFromTransactionState(TransactionStateENUM state);
//TransactionStateMachine.m
NSString * NSStringFromTransactionState(TransactionState state) {//枚举类型转字符
switch (state) {
case TransactionOpened:
return @"Opened";
case TransactionPending:
return @"Pending";
case TransactionClosed:
return @"Closed";
default:
return nil;
}
}
- 法并没有被公开声明,所以你必需要自己声明:
extern uint64_t dispatch_benchmark(size_t count, void (^block)(void));//The dispatch_benchmark function returns the average number of nanoseconds the given block takes to execute.
extern "C" MGCopyAnswer(CFStringRef prop);//可利用MGCopyAnswer(“UniqueDeviceID”)读取设备GUID
- 转化方针经纬度为高德坐标系
FOUNDATION_EXTERN
#if defined(__cplusplus)
#define FOUNDATION_EXTERN extern "C"
#else
#define FOUNDATION_EXTERN extern
#endif
///AMapLocation CoordinateType
typedef NS_ENUM(NSUInteger, AMapLocationCoordinateType)
{
AMapLocationCoordinateTypeBaidu = 0, ///<Baidu
AMapLocationCoordinateTypeMapBar, ///<MapBar
AMapLocationCoordinateTypeMapABC, ///<MapABC
AMapLocationCoordinateTypeSoSoMap, ///<SoSoMap
AMapLocationCoordinateTypeAliYun, ///<AliYun
AMapLocationCoordinateTypeGoogle, ///<Google
AMapLocationCoordinateTypeGPS, ///<GPS
};
/**
* @brief 转化方针经纬度为高德坐标系
* @param coordinate 待转化的经纬度
* @param type 坐标系类型
* @return 高德坐标系经纬度
*/
FOUNDATION_EXTERN CLLocationCoordinate2D AMapLocationCoordinateConvert(CLLocationCoordinate2D coordinate, AMapLocationCoordinateType type);
FOUNDATION_EXTERN BOOL AMapLocationDataAvailableForCoordinate(CLLocationCoordinate2D coordinate);
- UIRectClip
#ifdef __cplusplus//假如界说了__cplusplus,那么当时源代码被当中C++源代码处理,
#define UIKIT_EXTERN extern "C" __attribute__((visibility ("default")))
#else//假如没有界说__cplusplus, 那么当时源代码被当作C源代码处理
#define UIKIT_EXTERN extern __attribute__((visibility ("default")))
#endif
//对指定符号增加visibility(“default”)来导出符号
UIKIT_EXTERN void UIRectClip(CGRect rect);
- 是否为海外用户
blog.csdn.net/z929118967/…
iOS设备限制境外买卖(付出买卖危险操控)
/**
* 是否为海外用户...海外用户,SDK内部会屏蔽一些操作 默以为NO.
* @warning AMapServices初始化之前,设置才能生效
*/
extern BOOL _amapLocationOverseas;
- 删去数据库文件
//BGFMDBConfig_h
/**
删去数据库文件
*/
extern BOOL bg_deleteSqlite(NSString*_Nonnull sqliteName);
//BGTool.m
/**
删去数据库文件
*/
BOOL bg_deleteSqlite(NSString*_Nonnull sqliteName){
return [BGDB deleteSqlite:sqliteName];
}
//BGDB.m
/**
删去数据库文件.
*/
+(BOOL)deleteSqlite:(NSString*)sqliteName{
NSString* filePath = CachePath(([NSString stringWithFormat:@"%@.db",sqliteName]));
NSFileManager * file_manager = [NSFileManager defaultManager];
NSError* error;
if ([file_manager fileExistsAtPath:filePath]) {
[file_manager removeItemAtPath:filePath error:&error];
}
return error==nil;
}
- 铲除缓存
//BGFMDBConfig_h
/**
铲除缓存
*/
extern void bg_cleanCache();
//BGTool.m
/**
铲除缓存
*/
void bg_cleanCache(){
[[NSCache bg_cache] removeAllObjects];
}
- 蓝牙衔接状态的判别
//QCTPrinterManager.h
extern BOOL checkisBluetoothConnected();
//QCTPrinterManager.m
BOOL checkisBluetoothConnected(){
NSLog(@"SelectPeripheral.state %ld", [QCTSession shareQCTSession].SelectPeripheral.state);
switch ([QCTSession shareQCTSession].SelectPeripheral.state) {
case CBPeripheralStateConnected:
return YES;
break;
default:
return NO;
break;
}
}
see also
小程序:iOS逆向
只为你出现有价值的信息