Добавление делегата в AVAudioPlayer вызывает "_NSAutoreleaseNoPool (): объект 0x55e060 класса NSCFString автоматически освобожден без пула на месте - только утечка" - PullRequest
1 голос
/ 09 июня 2009

Я использую класс для воспроизведения звуков, используя AVAudioPlayer. Поскольку я хочу выпустить эти звуки сразу после их воспроизведения, я добавил делегата. Это вызывает ошибку «_NSAutoreleaseNoPool (): Object 0x55e060 класса NSCFString, автоматически освобождаемую без пула на месте - просто утечка» сразу после завершения воспроизведения звука, но до вызова my -audioPlayerDidFinishPlaying.

Вот несколько источников:

@interface MyAVAudioPlayer : NSObject <AVAudioPlayerDelegate> {
    AVAudioPlayer   *player;
    float           savedVolume;
    BOOL            releaseWhenDone;
}

Основной класс .m:

- (MyAVAudioPlayer *) initPlayerWithName: (NSString *) name;
{
    NSString *soundFilePath = [[NSBundle mainBundle] pathForResource: name ofType: @"caf"];

    NSURL *fileURL = [[NSURL alloc] initFileURLWithPath: soundFilePath];

    player = [[AVAudioPlayer alloc] initWithContentsOfURL: fileURL error: nil];
    [fileURL release];
    [player prepareToPlay];
    return (self);
}
- (MyAVAudioPlayer *)getAndPlayAndRelease:(NSString *)name withVolume:(float) vol;
{
    MyAVAudioPlayer *newMyAVPlayer = [self initPlayerWithName:name];
    player.volume = vol;
    [player play];
    releaseWhenDone = YES;
    [player setDelegate: self];
    return newMyAVPlayer;
}   
+ (void) getAndPlayAndReleaseAuto:(NSString *)name withVolume:(float) vol;
{
    MyAVAudioPlayer *newMyAVPlayer = [[MyAVAudioPlayer alloc] getAndPlayAndRelease:name withVolume:vol];
//  [newMyAVPlayer autorelease];
}

#pragma mark -
#pragma mark AVAudioPlayer Delegate Methods

- (void)audioPlayerDidFinishPlaying:(AVAudioPlayer *)playedPlayer successfully:(BOOL)flag {
    if (releaseWhenDone) {
        NSLog(@"releasing");
        [playedPlayer release];
//      [self release];
        NSLog(@"released");
    }
}

- (void)audioPlayerDecodeErrorDidOccur:(AVAudioPlayer *)player error:(NSError *)error {
    NSLog(@"Error while decoding: %@", [error localizedDescription] );
}

- (void)audioPlayerBeginInterruption:(AVAudioPlayer *)player {
    NSLog(@"Interrupted!");
}

- (void)audioPlayerEndInterruption:(AVAudioPlayer *)player {
    NSLog(@"EndInterruption!");
}

- (BOOL) play;
{   
    player.currentTime = 0.0;
    return [player play];
}

Комментирование [player setDelegate: self]; устраняет ошибку, но тогда мой audioPlayerDidFinishPlaying не вызывается.

Есть мысли? Я вдруг запускаюсь в другой ветке?

1 Ответ

0 голосов
/ 20 июня 2009

Я нашел проблему. Моя ошибка, конечно.

Во многих моих файлах классов я добавлял:

-(BOOL) respondsToSelector:(SEL) aSelector
{
    NSLog(@"Class: %@ subclass of %@, Selector: %@", [self class], [super class], NSStringFromSelector(aSelector));
    return [super respondsToSelector:aSelector];

}

В основном из любопытства.

Что ж, когда я добавил делегата к своему звуку, этот метод вызывается перед делегатом, и он вызывается из любого цикла выполнения, в котором находился AVAudioPlayer, и, вероятно, из цикла выполнения без пула авто-выпуска.

...