我有什么方法可以修改包含它名字的字符串中的变量吗?
这样的事情:
int example = 1;
NSString *foo = @"example";
foo.value++
此时,example
将等于2.
答案 0 :(得分:4)
你不能用这样的局部变量做到这一点。您可以使用KVC来实现实例变量。
答案 1 :(得分:2)
与其他答案相反,如果你愿意采用完全黑客模式,这确实是可能的。利用C中宏和指针的强大功能,我完成了这项任务(它还包括有关对象的类型的信息,以防您在运行时需要知道该做什么)。
使用它作为一个有趣的例子,但要小心 - 它会增加使用的内存量。
首先,让我们展示一下如何使用这些API:
<强>的main.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);
}
<强> IndexedVariable.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
<强> IndexedVariable.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
您可以仅使用此示例查找索引变量,其他任何内容都不会出现在查找表中。
答案 2 :(得分:1)
我不知道任何语言可以让你这样做。
在这种特殊情况下,您可以将指针传递给int
。
更一般地说,如果您想按名称操纵某些内容,请使用字典:
NSDictionary* d = @{@"example":@1};
NSString* key = @"example";
d[key] = @([d[key] intValue] + 1);