iOS 深度好文macOS开发备忘录

OC枚举值到字符串转换

2017-08-09  本文已影响1918人  fever105

OC枚举值到字符串转换

Swift枚举类型引入了Raw Value的概念,每个枚举case的Raw Value可以是其case name。假设有case king,则king.rawValue = "king"。相比之下OC枚举类型就弱爆了,只能绑定整数;要"反射"自己的case name,必须手动实现反射函数。不过,宏替换可以在一定程度上帮助我们自动实现。

本文基于Easy way to use variables of enum types as string in C?
,是笔者无意间看到的。短短几十行宏代码,看了好大一会儿也不知所以然,不得不重新捧起K&R复习Marcos;晚上回家接着细细研读,不禁感叹:竟然还有这种操作!


A. 使用

这套宏的用法分为两步:

A.1 在.h文件中定义枚举,声明反射函数

// M1
#define RAP_DIRECTION(XX) \
XX(RAPDirectionEast, ) \
XX(RAPDirectionSouth, ) \
XX(RAPDirectionWest, = 50) \
XX(RAPDirectionNorth, = 100) \

// M2
DECLARE_ENUM(RAPDirection, RAP_DIRECTION)

A.2 在.m文件中实现反射函数

DEFINE_ENUM(RAPDirection, RAP_DIRECTION)

A.3 例子

  1. 在任意.h文件中定义枚举,声明反射函数:

    #import "EnumMarcos.h"
    
    #define RAP_DIRECTION(XX) \
    XX(RAPDirectionEast, ) \
    XX(RAPDirectionSouth, ) \
    XX(RAPDirectionWest, = 50) \
    XX(RAPDirectionNorth, = 100) \
    
    DECLARE_ENUM(RAPDirection, RAP_DIRECTION)
    
  2. 在相应的.m文件中实现反射函数:

    DEFINE_ENUM(RAPDirection, RAP_DIRECTION)
    
  3. 尝试使用:

    NSString *str = NSStringFromRAPDirection(RAPDirectionEast);
    NSLog(@"RAPDirectionEast has case name: %@", str);
    
    
    RAPDirection dir = RAPDirectionFromNSString(@"RAPDirectionNorth");
    NSLog(@"RAPDirectionNorth has case value: %zd", dir);
    

输出结果:

RAPDirectionEast has case name: RAPDirectionEast
RAPDirectionNorth has case value: 100

B. 实现

B.1 DECLARE_ENUM(EnumType, ENUM_DEF)


#define DECLARE_ENUM(EnumType, ENUM_DEF) \
typedef NS_ENUM(NSInteger, EnumType) { \
ENUM_DEF(ENUM_VALUE) \
}; \
NSString *NSStringFrom##EnumType(EnumType value); \
EnumType EnumType##FromNSString(NSString *string); \

B.1.1 ENUM_VALUE(name, assign)

#define ENUM_VALUE(name, assign) name assign,

B.2 DEFINE_ENUM(EnumType, ENUM_DEF)

#define DEFINE_ENUM(EnumType, ENUM_DEF) \
NSString *NSStringFrom##EnumType(EnumType value) \
{ \
switch(value) \
{ \
ENUM_DEF(ENUM_CASE) \
default: return @""; \
} \
} \
EnumType EnumType##FromNSString(NSString *string) \
{ \
ENUM_DEF(ENUM_STRCMP) \
return (EnumType)0; \
}

B.2.1 ENUM_CASE(name, assign)

#define ENUM_CASE(name, assign) case name: return @#name;

B.2.1 ENUM_STRCMP(name, assign)

#define ENUM_STRCMP(name, assign) if ([string isEqualToString:@#name]) return name;

参考资料

  1. 示例代码
  2. Easy way to use variables of enum types as string in C?
上一篇下一篇

猜你喜欢

热点阅读