名前を含む文字列から変数を変更する方法はありますか?
このようなもの:
int example = 1;
NSString *foo = @"example";
foo.value++
その時点で、example
2 に等しくなります。
名前を含む文字列から変数を変更する方法はありますか?
このようなもの:
int example = 1;
NSString *foo = @"example";
foo.value++
その時点で、example
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
この例では、インデックス付きの変数のみを検索できます。それ以外はルックアップ テーブルに含まれません。
そんなことができる言語を私は知りません。
この特定のケースでは、へのポインターを渡すことができますint
。
より一般的には、名前で何かを操作したい場合は、辞書を使用します。
NSDictionary* d = @{@"example":@1};
NSString* key = @"example";
d[key] = @([d[key] intValue] + 1);