Obj-C 零食铺

预处理

总览

宏指令 含义
#define 给一串代码指定一个常量,本质是「替换」
#include 一般用于导入头文件
#undef 将某个已经定义的宏移除
#ifdef 如果定义了某个宏,返回YES,等价于#if defined (...)
#ifndef 如果没有定义某个宏,返回YES,等价于#if !defined (...)
#if, #else, #elif, #endif 条件处理
#pragma 使用标准化方法向编译器发送某些命令,比如忽略某些警告
注意: define结尾不用带分号。因为define的本质是「替换」,如果带上分号的话,分号也会被替换到程序中。比如:
#define MAXVAL 1000;
// 报错,因为 MAXVAL 的值其实为「1;」,并不是整型
int y = MAXVAL;

预处理操作符

宏定义的换行符\

使用\可以在定义比较长的宏指令的时候换行,比如

#define RequestCompleteCheckResponseDataArrayReturnIfNil \
    [super requestCompletePreprocessor];                   \
    if (!self.responseDataArray) {                         \
        return;                                            \
    }

字符串转化

使用#可以将被替换目标转化为字符串(就是在目标前后加上双引号),比如:

#define NSStringize_helper(x) #x
#define NSStringize(x) @NSStringize_helper(x)

NSString *str = NSStringize(str);
NSLog(@"%@", str);
// 输出 str

#define  result(a, b)  \
	NSLog(@#a " and " #b " are results!")

int main(void) {
   result(aa, bb);
   return 0;
}
// 输出 aa and bb are results!

其实,上面的NSLog(@#a " and " #b " are results!")如果展开看的话,应该等价于NSLog(@"a" " and " "b" " are results!")。但是由空格分隔的字符串在编译时被连起来了,所以实际等价于NSLog(@"a and b are results!")

参考: Stringize (#)

粘贴符号##

##能把宏定义中两个参数「粘贴」为一体。比如

#define tokenpaster(n) NSLog (@"token" #n " = %d", token##n)

int main(void) {
   int token99 = 99;
   tokenpaster(99);
   return 0;
}

// 输出 token99 = 99

又比如,声明一个弱/强引用的宏:

/// weak self 定义
#define Weak(type) __weak __typeof(&*type) weak##type = type

/// strong self 定义
#define Strong(type) __strong __typeof(&*type) type = weak##type

## 即是将输入的参数 typeweak 粘在一起了

## 用到了上面的字符串转化功能

标记某个类不能被继承

// 标记ClassA不能被继承
__attribute__((objc_subclassing_restricted))
@interface ClassA : NSObject
@end

@implementation ClassA
@end

使用 __auto_type 关键字让编译器自动推断常量类型

参考:https://mp.weixin.qq.com/s/lRYYz0a0VqL9-1Dhm7aaaQ

Objc 里面的__auto_type关键字可以实现类似 Swift 类型推断的能力:

const __auto_type kAnimationDuration = 0.3;

利用这个关键字可以简化一些复杂类型的声明:

// 旧方式
NSArray<NSDictionary<NSString *, NSString *> *> *models = ...;

// 新方式
__auto_type models = ...;

还可以通过宏定义实现类似 Swift 中通过let声明常量的方式:

#if defined(__cplusplus)
#define let auto const
#else
#define let const __auto_type
#endif

#if defined(__cplusplus)
#define var auto
#else
#define var __auto_type
#endif
let kAnimationDuration = 0.3;

调用类方法时如何做到像 Swift 的 Self 那样便捷编写类名

  1. 如果是类方法相互调用的话,可以用 self 替代类名
  2. 如果是实例方法调用类方法的话,需要 [[self class] classMethod] 或者 [self.class classMethod]
  3. 也可以将类名定义为一个宏。例如:
#define xxx YourClassName
[xxx classMethod];

访问控制

NSEnum forward declaration

// Forward declaration for XYZCharacterType in other header say XYZCharacter.h
typedef NS_ENUM(NSUInteger, XYZCharacterType);


// Enum declaration header: "XYZEnumType.h"
#ifndef XYZCharacterType_h
#define XYZCharacterType_h

typedef NS_ENUM(NSUInteger, XYZEnumType) {
    XYZCharacterTypeNotSet,
    XYZCharacterTypeAgent,
    XYZCharacterTypeKiller,
};

#endif /* XYZCharacterType_h */`
参考:https://stackoverflow.com/a/42009056/16991379

Read more

2025 年度总结

2025 年度总结

今天是 2026 年 1 月 1 号,又是新的一年。这个元旦没有安排出行任务,就在家里休息休息,或者出门溜达溜达。昨天休了一天全薪病假,做了体检,写了年终绩效总结,晚上干了一顿烤肉,没有时间写个人的年度总结。今天起早写写总结。 以下「今年」指 2025 年。 职业发展 算起来,我已经毕业工作四年多了。职业发展整体上还算稳定,没有碰到过糟心事,遇到的领导们也都对我关怀有加。今年又晋升一次,薪资迈上新的台阶。越往上升,越觉得离职业生涯的终点越近,逼迫自己赶紧找个靠谱稳定的副业,到 35 岁没人要的时候能养活自己。 最近两年 AI 大模型的崛起,提高了许多行业的可替代性。码农虽然不是首当其冲的,但危机感已经弥漫在各个论坛博客公共平台上面。没有人能准确预测到未来发展,但做好两手准备是很有必要的。码农不能再只低着头守着自己的键盘和屏幕,也要往外看,接触社会上的各种信息,打破信息壁垒。掌握的信息越多,出路就越多。

By Gray
联通 FTTR 宽带从路由器设置自动重启和穿墙功率

联通 FTTR 宽带从路由器设置自动重启和穿墙功率

几个月前把家里宽带换成了联通的千兆 FTTR 宽带,包含一主一从两个点位。配套光猫设备是华为的星光 F50 尊享版。 主点位放置在客厅茶几上,方便连接电视。从点位放在卧室门口,那里恰好有一个不耽误过路的小拐角可以放路由器。平常我们基本不在客厅活动,其他区域最近的 Wi-Fi 信号源是从路由器,因此我们大多数的设备连接的都是从路由器。从路由器的工作负荷很大。 从路由器个头小主路由器很多,散热不咋地。工作时间久了发热就容易发生数据包堵塞,丢包延迟高。需要把它电源拔掉重启。从宽带开通到现在,数据包堵塞影响网络的情况每个月会发生一次。有一次还影响了居家办公的视频会议。宽带维修师傅也给不出有效的法子,建议就是定期插拔从路由器电源。 从路由器和书房之间隔了两堵墙。信号到我书桌那个位置时,千兆网速已经衰减到只有 400-500Mbps 了,折损将近一半。叠加路由器发热的 debuff,书桌位置的网速最差的时候几乎和百兆宽带差不多。 我尝试过在光猫后台管理将路由器功率设置到「穿墙」模式,但没有任何作用。今天在后台研究了一番发现,原来我之前设置的功率是仅对主路由器生效,从路由器还是标准功率。要修

By Gray