我有什么办法可以从包含变量名的字符串中修改变量吗?
像这样的东西:
int example = 1;
NSString *foo = @"example";
foo.value++
此时,example
将等于 2。
我有什么办法可以从包含变量名的字符串中修改变量吗?
像这样的东西:
int example = 1;
NSString *foo = @"example";
foo.value++
此时,example
将等于 2。
你不能用这样的局部变量来做到这一点。您可以使用KVC对实例变量进行操作。
与其他答案相反,如果您愿意完全破解模式,这确实是可能的。使用 C 中的宏和指针的力量,我完成了这项任务(它还包括有关对象类型的信息,以防您需要知道在运行时做什么)。
使用这个作为一个有趣的例子,但要小心 - 它会增加相当多的内存使用量。
首先,让我们展示您将如何使用这些 API:
主.m:
#import <Foundation/Foundation.h>
#import "IndexedVariable.h"
int main() {
INDEXED_VAR(int, example, 1);
NSString *foo = @"example";
GET_INDEXED_VAR_AS(foo, int)++;
NSLog(@"%i", example);
}
索引变量.h:
#import <Foundation/Foundation.h>
// feel free to not use these macros, but I feel they make it much easier to read
#define choose_if __builtin_choose_expr
#define compatible __builtin_types_compatible_p
#define INDEXED_TYPE_FROM_NAME(p_type) \
choose_if(compatible(signed char , p_type), INDEXED_TYPE_SIGNED_CHAR,\
choose_if(compatible(unsigned char , p_type), INDEXED_TYPE_UNSIGNED_CHAR,\
choose_if(compatible(signed short , p_type), INDEXED_TYPE_SIGNED_SHORT,\
choose_if(compatible(unsigned short, p_type), INDEXED_TYPE_UNSIGNED_SHORT,\
choose_if(compatible(signed int , p_type), INDEXED_TYPE_SIGNED_INT,\
choose_if(compatible(unsigned int , p_type), INDEXED_TYPE_UNSIGNED_INT,\
choose_if(compatible(signed long , p_type), INDEXED_TYPE_SIGNED_LONG,\
choose_if(compatible(unsigned long , p_type), INDEXED_TYPE_UNSIGNED_LONG,\
choose_if(compatible(float , p_type), INDEXED_TYPE_FLOAT,\
choose_if(compatible(double , p_type), INDEXED_TYPE_DOUBLE,\
choose_if(compatible(id , p_type), INDEXED_TYPE_OBJC_OBJECT,\
choose_if(compatible(void * , p_type), INDEXED_TYPE_GENERIC_POINTER,\
INDEXED_TYPE_UNKNOWN\
))))))))))))
#define INDEXED_VAR(p_type, p_name, p_initial_value)\
p_type p_name = p_initial_value;\
IndexedVariable *__indexed_ ## p_name = [IndexedVariable index:INDEXED_TYPE_FROM_NAME(p_type) :@#p_name :&p_name];\
(void) __indexed_ ## p_name
#define GET_INDEXED_VAR_AS(p_name, type) (*((type *) [[IndexedVariable lookupWithName:p_name] address]))
// represents the type of an indexed variable
enum INDEXED_TYPE {
INDEXED_TYPE_SIGNED_CHAR,
INDEXED_TYPE_UNSIGNED_CHAR,
INDEXED_TYPE_SIGNED_SHORT,
INDEXED_TYPE_UNSIGNED_SHORT,
INDEXED_TYPE_SIGNED_INT,
INDEXED_TYPE_UNSIGNED_INT,
INDEXED_TYPE_SIGNED_LONG,
INDEXED_TYPE_UNSIGNED_LONG,
INDEXED_TYPE_FLOAT,
INDEXED_TYPE_DOUBLE,
INDEXED_TYPE_OBJC_OBJECT,
INDEXED_TYPE_GENERIC_POINTER,
INDEXED_TYPE_UNKNOWN,
};
@interface IndexedVariable : NSObject
+(id) index:(enum INDEXED_TYPE) indexedType :(NSString *) varName :(void *) ptr;
+(IndexedVariable *) lookupWithName:(NSString *) name;
-(enum INDEXED_TYPE) indexedType;
-(void *) address;
@end
索引变量.m:
#import "IndexedVariable.h"
static NSMutableDictionary *indexedVariableDictionary;
@implementation IndexedVariable {
enum INDEXED_TYPE _type;
void *_address;
}
+(id) index:(enum INDEXED_TYPE)indexedType :(NSString *)varName :(void *)ptr
{
IndexedVariable *var = [IndexedVariable new];
var->_type = indexedType;
var->_address = ptr;
if (indexedVariableDictionary == nil) {
indexedVariableDictionary = [NSMutableDictionary new];
}
indexedVariableDictionary[varName] = [NSValue valueWithNonretainedObject:var];
return var;
}
+(IndexedVariable *) lookupWithName:(NSString *)name
{
return [indexedVariableDictionary[name] nonretainedObjectValue];
}
-(enum INDEXED_TYPE) indexedType {
return _type;
}
-(void *) address {
return _address;
}
-(void) dealloc {
NSArray *keys = [indexedVariableDictionary allKeysForObject:[NSValue valueWithNonretainedObject:self]];
[indexedVariableDictionary removeObjectsForKeys:keys];
}
@end
您只能使用此示例查找索引变量,其他任何内容都不会在查找表中。
我不知道任何语言可以让你这样做。
在这种特殊情况下,您可以将指针传递给int
.
更一般地说,如果您想通过名称来操作某些东西,请使用字典:
NSDictionary* d = @{@"example":@1};
NSString* key = @"example";
d[key] = @([d[key] intValue] + 1);