Фальшивый va_list в ARC
Мне нужно создать в приложении iOS подделку va_list
перейти на NSString initWithFormat:arguments:
функция, это мой код:
NSArray *fixedArguments = [[NSArray alloc] initWithArray:arguments];
NSRange range = NSMakeRange(0, [fixedArguments count]);
va_list fakeArgList = (va_list)malloc(sizeof(NSString *) * [fixedArguments count]);
__unsafe_unretained id *ptr = (__unsafe_unretained id *)fakeArgList;
[fixedArguments getObjects:ptr range:range];
content = [[NSString alloc] initWithFormat:outputFormat
arguments:(va_list)fakeArgList];
free(fakeArgList);
Компилятор жалуется на это сообщение в строке приведения:
error: cast of a non-Objective-C pointer type 'va_list' (aka 'char *') to '__unsafe_unretained id *' is disallowed with ARC
getObjects:range:
Функция определяется следующим образом:
- (void)getObjects:(id __unsafe_unretained [])objects range:(NSRange)range;
Я перепробовал все, но до сих пор не могу избавиться от этой ошибки...
Есть ли решение для создания подделки va_list
с включенной ARC? Что я делаю неправильно?
2 ответа
РЕДАКТИРОВАТЬ: это больше не работает. Как и предполагалось в первоначальном ответе, ABI, по-видимому, изменился из-под этого ответа
Немного поигрался и заставил его работать - дважды проверил на утечки или оставленную память и ничего не увидел.
NSArray *fixedArguments = [[NSArray alloc] initWithObjects: @"foo", @"bar", @"baz", nil];
NSRange range = NSMakeRange(0, [fixedArguments count]);
NSMutableData* data = [NSMutableData dataWithLength: sizeof(id) * [fixedArguments count]];
[fixedArguments getObjects: (__unsafe_unretained id *)data.mutableBytes range:range];
NSString* content = [[NSString alloc] initWithFormat: @"1: %@ 2: %@ 3: %@" arguments: data.mutableBytes];
NSLog(@"%@", content);
Мне нравится (ab) использовать NSMutableData вот так, чтобы получить семантику сохранения / освобождения для произвольного куска памяти - это не обязательно относится к рассматриваемой проблеме, но это изящный маленький трюк.
Примечание для будущих читателей: фальсификация подобного va_list может работать с текущим ABI для MacOS и iOS, но в целом он не переносимый и не очень хороший подход.
Это возможно, если вы захотите добавить немного быстроты в свой проект!
Важным моментом является отображение NSArray на [CVarArgType]
который является быстрым эквивалентом для va_list
, Если вы попытаетесь разыграть [AnyObject]
в [CVarArgType]
Вы вызываете сбои во время выполнения, но с map
мы можем явно составить необходимый список.
Остальная часть кода - это оболочка, которую я сделал, чтобы я мог вызывать ее из obj-c. Вы можете создать оболочку для любой функции obj-c, которую вы хотите вызвать таким образом.
@objc class StringFormat: NSObject {
class func format(key: String, args: [AnyObject]) -> String {
let locArgs: [CVarArgType] = args.map({ (arg: AnyObject) -> CVarArgType in
if let iArg = (arg is NSNumber ? arg.intValue : nil) {
return iArg
}
return arg as! CVarArgType
});
return String(format: key, arguments: locArgs)
}
}