开发者

Message sent to deallocated instance on object that should exist

开发者 https://www.devze.com 2023-03-15 11:51 出处:网络
[CFString isEqualToString:]:开发者_运维问答 message sent to deallocated instance This is the error I\'m getting. I have a View controller with a Property that is an NSMutableArray and it contains som

[CFString isEqualToString:]:开发者_运维问答 message sent to deallocated instance

This is the error I'm getting. I have a View controller with a Property that is an NSMutableArray and it contains some Player objects.

Then I have a method that switches to another view, where I can add and remove players. When I load the next view I pass a pointer to the first view, to the second view. I then updated the Array by using [previousView.playerList addObject:p]; where p is the newly created player object.

The error comes, however when I attempt to display the list of players in the second view's table view. when I try to access [previousView.playerlist objectAtIndex:[indexPath row]]; I get the error above.

here's the code:

This is in the first View, where it loads the second and passes itself to the property of the second.

- (IBAction) addPlayerButton:(id)sender {
    [self retain];
    PlayerListViewController *playerListViewController = [[PlayerListViewController alloc] init];
    playerListViewController.previousView = self;
    [UIView transitionFromView:self.view toView:playerListViewController.view duration:0.5 options:UIViewAnimationOptionTransitionFlipFromLeft completion:nil];
}

This is where I initialize the player and add it to the array.

- (IBAction)addPlayer:(id)sender {
    Player *p = [[[Player alloc] initWithPlayerName:playerNameField.text withOrder:[previousView.playerList count] withDelegate:previousView] retain];
    [previousView.playerList addObject:p];
    [playerNameField resignFirstResponder];
    [playerTableView reloadData];
}

And here is where I get my error

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
    static NSString *cellIdentifier = @"MyCell";

    UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:cellIdentifier];
    if (cell == nil) {
        cell = [[[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:cellIdentifier] autorelease];
    }
    Player* p = [previousView.playerList objectAtIndex:[indexPath row]];
    [cell.textLabel setText:[p playerName]];
    cell.accessoryType = UITableViewCellAccessoryCheckmark;

    return cell;
}

I'm getting the error on the line where I set the cell text. [cell.textLabel setText:[p playerName]];

Does anyone know where I'm screwing up? I will post more code if needed.

I trimmed Player.m to only include the Synthesis and Initalization, because the rest is A LOT of code, that doesnt really have to do with this error, because it isin't being called.

Player.h

@protocol playerProtocol <NSObject>

@optional
- (void) playerDied:(id)playerObject;
- (void) playerWasAdded:(id)playerObject;
- (void) playerLifeDidChange:(id)playerObject;
@end

@interface Player : NSObject {
    id <playerProtocol> delegate;
}
@property (nonatomic,retain) NSString *playerName;
@property (nonatomic, readwrite) int playerLife;
@property (nonatomic, readwrite) int playerPoison;
@property (nonatomic, readwrite) int order;
@property (nonatomic, readwrite) Boolean invincible;
@property (nonatomic, assign) id <playerProtocol>delegate;
@property (nonatomic, retain) UIView *viewPane;
@property (nonatomic) Boolean shown;
@property (nonatomic, readwrite) int minusButton;
@property (nonatomic, readwrite) int plusButton;
@property (nonatomic, retain) UIImageView *readout;
@property (nonatomic, retain) NSArray *playerLifeNumerals;


- (id) initWithPlayerName:(NSString *)playersName withOrder:(int)Order withDelegate:(id)d;
- (void) setPlayerLife:(int)pplayerLife;
- (void) setPlayerPoison:(int)pplayerPoison;
- (NSArray *) getLifeNumeralsFromPlayer:(Player *)playerObject;

@end

Player.m

@implementation Player

@synthesize playerLife, playerName, playerPoison, order, delegate, invincible, viewPane, readout, shown, minusButton, plusButton, playerLifeNumerals;

#pragma mark - Custom Initalizer

- (id) initWithPlayerName:(NSString *)playersName withOrder:(int)Order withDelegate:(id)d {

    [super init];
    delegate = d;
    playerLife = 20;
    playerPoison = 0;
    order = Order;
    playerName = playersName;
    invincible = NO;
    [delegate playerWasAdded:self];
    viewPane = nil;
    readout = nil;
    shown = NO;
    return self;
}

Also here's the arrays declaration @property (nonatomic, retain) NSMutableArray *playerList;


In:

- (id) initWithPlayerName:(NSString *)playersName withOrder:(int)Order withDelegate:(id)d {

    [super init];
    delegate = d;
    playerLife = 20;
    playerPoison = 0;
    order = Order;
    playerName = playersName;
    invincible = NO;
    [delegate playerWasAdded:self];
    viewPane = nil;
    readout = nil;
    shown = NO;
    return self;
}

you are not retaining playerName. Note that although you’ve declared your playerName property as retain (and it should be copy instead), you’re not using the property setter in -initWith…. Instead, you’re directly accessing the backing instance variable. Since you’re directly assigning playersName to the instance variable, you need to manually send it -retain (or, better yet, -copy).

Check whether you need this in your other declared properties. Also, you should follow the idiom:

self = [super init];
if (self) { … }
return self;

in your initialisers.

Edit: you need to carefully review your code with regard to memory management practices since you’re leaking in more than one place. For example, in:

- (IBAction) addPlayerButton:(id)sender {
    [self retain];
    PlayerListViewController *playerListViewController = [[PlayerListViewController alloc] init];
    playerListViewController.previousView = self;
    [UIView transitionFromView:self.view toView:playerListViewController.view duration:0.5 options:UIViewAnimationOptionTransitionFlipFromLeft completion:nil];
}

what’s the purpose of [self retain], and is there a balancing release?

Another example: in:

Player *p = [[[Player alloc] initWithPlayerName:playerNameField.text withOrder:[previousView.playerList count] withDelegate:previousView] retain];
[previousView.playerList addObject:p];

you get +1 ownership from +alloc, another +1 from -retain, and the array also owns p, hence another +1. You should replace -retain with -autorelease to balance the number of times you’re taking ownership of that object.

0

精彩评论

暂无评论...
验证码 换一张
取 消