iPhone: - [NSConcreteMutableData fastestEncoding]: нераспознанный селектор - PullRequest
2 голосов
/ 21 марта 2009

Я пытаюсь сделать это:

self.somestring = [@"hardcodedstring" stringByAppendingString:someotherstring];

Но я продолжаю получать:

*** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '*** -[NSConcreteMutableData fastestEncoding]: unrecognized selector sent to instance 0x1fb930'

Я нигде не вызываю этот метод, но вижу, что stringByAppendingString: вызывает его в моем стеке:

Stack: (
808221155,
806100816,
808224837,
807957033,
807851552,
812064725,
812064413,
18085, <== -[NSString stringByAppendingString:] + 109 in section LC_SEGMENT.__TEXT.__text of /Developer/Platforms/iPhoneOS.platform/Developer/SDKs/iPhoneOS2.2.1.sdk/System/Library/Frameworks/Foundation.framework/Foundation
817945012,
821160740,
821157652,
821149552,
807852041,
812070519,
807836299,
807834407,
827752032,
816118388,
816157144,
8381,
8244

) Как бы я это исправить, чтобы добавить строку, как и положено. Спасибо,
Исаак
Редактировать : Чтобы получить доступ к некоторой строке, я делаю это:

@property (nonatomic,retain) NSString *somestring;

В моем ViewController.h и:

@synthesize somestring;

В моем ViewController.m.
Редактировать : отсюда приходит некая строка:

- (void)imagePickerController:(UIImagePickerController *)picker didFinishPickingImage:(UIImage *)img editingInfo:(NSDictionary *)editInfo {
    self.somestring = [@"hardcodedstring" stringByAppendingString:[UIImagePNGRepresentation(img) encodeBase64]]; //encodeBase64 encodes it to base64
}

Подробнее Редактировать : Я расстался:

2009-03-20 15:14:21.389 myproject[3467:20b] *** -[NSConcreteMutableData getCharacters:range:]: unrecognized selector sent to instance 0x1fa630
2009-03-20 15:14:21.403 myproject[3467:20b] *** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '*** -[NSConcreteMutableData getCharacters:range:]: unrecognized selector sent to instance 0x1fa630'
2009-03-20 15:14:21.412 myproject[3467:20b] Stack: (
808221155,
806100816,
808224837,
807957033,
807851552,
807660389,
807733601,
807733297,
807891629,
812155873,
812293801,
18081,
817945012,
821160740,
821157652,
821149552,
807852041,
812070519,
807836299,
807834407,
827752032,
816118388,
816157144,
8381,
8244
)
terminate called after throwing an instance of 'NSException'
(gdb) info symbol 18081
-[InternalChoicesController imagePickerController:didFinishPickingImage:editingInfo:] + 73 in section LC_SEGMENT.__TEXT.__text of /Users/isaacwaller/Documents/myproject/build/Debug-iphoneos/myproject.app/myproject
(gdb) info symbol 812293801
NSLog + 25 in section LC_SEGMENT.__TEXT.__text of /Developer/Platforms/iPhoneOS.platform/Developer/SDKs/iPhoneOS2.2.1.sdk/System/Library/Frameworks/Foundation.framework/Foundation

Так что я думаю, что это проблема с кодированием данных Base64? Код, который я использую:

static const char encodingTable[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";

@implementation NSData (Base64)

- (NSString *)encodeBase64;
{
if ([self length] == 0)
    return @"";

char *characters = malloc((([self length] + 2) / 3) * 4);
if (characters == NULL)
    return nil;
NSUInteger length = 0;

NSUInteger i = 0;
while (i < [self length])
{
    char buffer[3] = {0,0,0};
    short bufferLength = 0;
    while (bufferLength < 3 && i < [self length])
        buffer[bufferLength++] = ((char *)[self bytes])[i++];

    //  Encode the bytes in the buffer to four characters, including padding "=" characters if necessary.
    characters[length++] = encodingTable[(buffer[0] & 0xFC) >> 2];
    characters[length++] = encodingTable[((buffer[0] & 0x03) << 4) | ((buffer[1] & 0xF0) >> 4)];
    if (bufferLength > 1)
        characters[length++] = encodingTable[((buffer[1] & 0x0F) << 2) | ((buffer[2] & 0xC0) >> 6)];
    else characters[length++] = '=';
    if (bufferLength > 2)
        characters[length++] = encodingTable[buffer[2] & 0x3F];
    else characters[length++] = '=';    
}

return [[[NSString alloc] initWithBytesNoCopy:characters length:length encoding:NSASCIIStringEncoding freeWhenDone:YES] autorelease];
}

@end

Спасибо, Исаак Уоллер

Ответы [ 2 ]

4 голосов
/ 21 марта 2009

Я бы проверил код на наличие опечаток. Ошибка довольно ясно указывает на то, что экземпляр NSMutableData используется там, где ожидается NSString. Может показаться, что либо encodeBase64 каким-то образом возвращает self, либо необработанный UIImagePNGRepresentation (img) передается stringByAppendingString:.

Если это не очевидно, просто разбейте его и проверяйте на каждом шаге. (Я использую NSLog для этого примера, но, конечно, сработал бы и с отладчиком).

- (void)imagePickerController:(UIImagePickerController *)picker didFinishPickingImage:(UIImage *)img editingInfo:(NSDictionary *)editInfo {
    NSData *rep = UIImagePNGRepresentation(img);
    NSLog(@"Rep: %@", rep);
    NSString *base64 = [rep encodeBase64];
    NSLog(@"Base 64 is a %@", NSStringFromClass([base64 class]));
    self.somestring = [@"hardcodedstring" stringByAppendingString:base64]; //encodeBase64 encodes it to base64

}

Мое лучшее предположение заключается в том, что вы случайно передаете необработанные NSData вместо NSString - но если это не удастся, приведенный выше код должен либо работать правильно, либо точно показывать, где что-то ломается.

2 голосов
/ 21 марта 2009

@"Some String" это не объект, это строковый литерал. Вы не можете отправлять сообщения на него. Вам нужно будет сделать что-то еще, чтобы присоединиться к этим строкам.

Что-то вроде:

 [NSString stringWithFormat:@"%@%@", @"String 1", @"String 2"];

Видимо, это не правильно. Строковые литералы рассматриваются как объекты.

Как уже упоминалось в комментариях, у вас также может быть проблема с self.somestring. Если вы не объявили свойство или не синтезировали somestring, то доступ к нему через self. неверен. Вы должны просто использовать

somestring = [NSString stringWithFormat:@"%@%@", @"String 1", @"String 2"];

Если вы сделали:

@interface myClass {
  NSString *somestring;
}
@property(nonatomic, retain) NSString *somestring;
@end

И

@implementation myClass
@synthesize somestring;
@end

Затем вы можете получить доступ к переменной, используя self.somestring, которая на самом деле является просто синтаксическим сахаром для [self somestring].

Важно также отметить, что @property и @synthesize на самом деле являются синтаксическими кусочками сахара. Они делают что-то похожее на это.

@interface myClass {
  NSString *somestring;
}

-(NSString *)somestring;
-(void)setSomestring:(NSString *)value;
@end

И

@implentation myClass
-(NSString *)somestring {
  return somestring;
}
-(void)setSomestring:(NSString *)value {
  somestring = value;
}
@end

Так что если вы не объявили somestring как свойство и не синтезировали его, то у вас нет этих методов для ответа на передаваемое сообщение.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...