0

別のピア (自己ではない) に対して displayNameForPeer を呼び出した後に GKSession を解放するときに、これでシミュレーターを確実にクラッシュさせることができます。 4.2 以降ではなく、4.0 および 4.1 でのみクラッシュが見られるため、心配する必要があります)。

出力は次のとおりです。

found available peer; checking name and ID... m4, 26176566
*** -[GKSessionInternal lock]: message sent to deallocated instance 0x7508900

再現可能な最小限のコード セットを次に示します。クラッシュをトリガーするには、ネットワーク上で別の GKSession が表示されている必要があります (displayNameForPeer を呼び出すために使用できるピアが見つかるようにする必要があります)。これと同じコードを別のデバイスで実行しますが、makeUnavailable と killSession の呼び出しは必要ありません。

- (void)viewDidLoad
{

[self createSession];
[self makeAvailable];

peerListAvailable = [[NSMutableArray alloc] initWithArray:[currentSession peersWithConnectionState:GKPeerStateAvailable]];
for (NSString *peer in peerListAvailable)
{
    // this method guarantees the crash on session release
    NSLog(@"found available peer; checking name and ID... %@, %@",[currentSession displayNameForPeer:peer], peer);
}
[peerListAvailable release];
peerListAvailable = nil;


[self makeUnavailable];
[self killSession];

[super viewDidLoad];
}


- (void) createSession
{
if (!currentSession)
{
    currentSession = [[GKSession alloc] initWithSessionID:@"GKTester" displayName:nil sessionMode:GKSessionModePeer];
    currentSession.delegate = self;
    currentSession.disconnectTimeout = 30;
    [currentSession setDataReceiveHandler: self withContext:nil];
}

}

-(void) killSession
{
if (currentSession) 
{
    [currentSession disconnectFromAllPeers];
    [currentSession setDelegate:nil];
    [currentSession setDataReceiveHandler:nil withContext:nil];
    [currentSession release]; // crash occurs after this
    currentSession = nil;   
}
}

-(void) makeAvailable
{
while (currentSession && !currentSession.available)
{
    [currentSession setAvailable:YES];
    [NSThread sleepForTimeInterval:.5];
}
}

-(void) makeUnavailable
{
while (currentSession && currentSession.available)
{
    [NSThread sleepForTimeInterval:.5];
    [currentSession setAvailable:NO];
}

}

4

1 に答える 1

0

コードにオーバーリリースがあります:

[currentSession disconnectFromAllPeers];
[currentSession setDelegate:nil];
[currentSession setDataReceiveHandler:nil withContext:nil];
[currentSession release]; // This is an over-release
currentSession = nil; // You are trying to access a variable after it's been released

次のように、dealloc でのみ currentSession メンバー変数を解放する必要があります。

- (void)dealloc
{
    [currentSession release];
    [super dealloc];
}
于 2012-01-16T01:05:56.043 に答える