1

由@TX 解决。我在代码中用“// SOLUTION”标记了更改!!!

我有以下问题,如果我展示ModalViewController,NSMutableArray“projectsArray”会被重置。

这是我的代码:

这里是 .h 文件:

    // ViewController.h
#import <UIKit/UIKit.h>

@class AddProject;

@interface ViewController : UIViewController <UITableViewDelegate, UITableViewDataSource>
{
    IBOutlet UITableView *projectsTableView;
    UIAlertView *reallyDelete;

    BOOL candelButtonClicked;
    NSIndexPath *actualIndexPath;

    NSMutableArray *projectsArray;
    NSInteger ID;
    NSString *NAME;

    AddProject *addProject;
    **ViewController *viewContr; // SOLUTION**
}
- (IBAction)addNewProject:(id)sender;

- (void)addToTableView:(NSString *)projectName;

@property (nonatomic, retain) UIAlertView *reallyDelete;
@property (nonatomic) NSInteger cancelButtonIndex;

@property IBOutlet UITableView *projectsTableView;

@property (nonatomic, retain) AddProject *addProject;
**@property (nonatomic, retain) ViewController *viewContr; // SOLUTION**

@end

这里是 .m 文件:

    // ViewController.m
#import "ViewController.h"
#import "CustomTableCell.h"
#import "AddProject.h"

@interface ViewController ()

@end

@implementation ViewController

**@synthesize viewContr; // SOLUTION**

- (void)viewDidLoad
{
    [super viewDidLoad];
    // Do any additional setup after loading the view, typically from a nib.

    projectsArray = [[NSMutableArray alloc] init];
    ViewController *element = [[ViewController alloc] init];

    element->ID = 1;
    element->NAME = @"Demo project";
    NSLog(@"viewdidload");
    [projectsArray addObject:element];

    NSLog(@"1. Array length: %d", [projectsArray count]);
}

- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
{
    if ([projectsArray count] < 1) {
        NSLog(@"2. Array length: 0");
        return 0;
    } else {
        NSLog(@"2. Array length: %d", [projectsArray count]);
        return [projectsArray count];
    }
}

- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath
{
    return 78;
}

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

    CustomTableCell *cell = (CustomTableCell *)[tableView dequeueReusableCellWithIdentifier:customTableCellIdentifier];
    if (cell == nil)
    {
        NSArray *nib = [[NSBundle mainBundle] loadNibNamed:@"CustomTableCell" owner:self options:nil];
        cell = [nib objectAtIndex:0];
    }

    ViewController *element = [[ViewController alloc] init];
    element = [projectsArray objectAtIndex:indexPath.row];
    cell.nameLabel.text = element->NAME;
    cell.prepTimeLabel.text = [NSString stringWithFormat:@"%i", element->ID];

    NSLog(@"3. Array length: %d", [projectsArray count]);

    return cell;
}

- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath
{
}

- (void)tableView:(UITableView *)tableView commitEditingStyle:(UITableViewCellEditingStyle)editingStyle forRowAtIndexPath:(NSIndexPath *)indexPath {

    if (editingStyle == UITableViewCellEditingStyleDelete) {
        // Delete the row from the data source

        reallyDelete = [[UIAlertView alloc] initWithTitle:@"Deleting" message:@"Do your really want to delete that row?" delegate:self cancelButtonTitle:@"No" otherButtonTitles:@"Yes", nil];
        [reallyDelete show];

        actualIndexPath = indexPath;
    }
}

- (void)alertView:(UIAlertView *)alertView clickedButtonAtIndex:(NSInteger)buttonIndex {

    if (buttonIndex != [reallyDelete cancelButtonIndex])
    {
        //If "Yes" clicked delete
        [projectsArray removeObjectAtIndex:actualIndexPath.row];
        NSLog(@"delete row");

        [self.projectsTableView deleteRowsAtIndexPaths:[NSMutableArray arrayWithObjects:actualIndexPath, nil] withRowAnimation:YES];
        [self.projectsTableView reloadData];
    }
}

#pragma mark Edit

- (IBAction)addNewProject:(id)sender {
    NSLog(@"4. Array length: %d", [projectsArray count]);

    AddProject *addProjectView = [[AddProject alloc] initWithNibName:@"AddProject" bundle:nil];
    NSLog(@"4.1 Array length: %d", [projectsArray count]);
    addProjectView.modalTransitionStyle = UIModalTransitionStyleCoverVertical;
    NSLog(@"4.2 Array length: %d", [projectsArray count]);
    addProjectView.modalPresentationStyle = UIModalPresentationFormSheet;
    NSLog(@"4.3 Array length: %d", [projectsArray count]);
    **addProjectView.viewContr = self; // SOLUTION**
    [self presentModalViewController:addProjectView animated:YES];
    NSLog(@"4.4 Array length: %d", [projectsArray count]);
    addProjectView.view.superview.frame = CGRectMake(addProjectView.view.center.x/2.5, addProjectView.view.center.y/3, 800, 600);
    NSLog(@"4.5 Array length: %d", [projectsArray count]);
}

- (void)addToTableView:(NSString *)projectName
{
    NSLog(@"Project Array: %@", projectsArray);
    NSLog(@"addToTableView: %@", projectName);    
    NSLog(@"5. Array length: %d", [projectsArray count]);

    ViewController *element = [[ViewController alloc] init];

    element->ID = 2;
    element->NAME = projectName;

    [projectsArray addObject:element];

    NSLog(@"6. Array length: %d", [projectsArray count]);

    [self.projectsTableView reloadData];
    [self.projectsTableView setNeedsDisplay];
}

- (void)viewDidUnload
{
    [self setProjectsTableView:nil];
    self.projectsTableView = nil;
    [super viewDidUnload];
    // Release any retained subviews of the main view.
}

- (BOOL)shouldAutorotateToInterfaceOrientation:(UIInterfaceOrientation)interfaceOrientation
{
    return YES;
}

@end

这里是模态视图的 .h 文件:

    // AddProject.h
#import <UIKit/UIKit.h>

@class ViewController;

@interface AddProject : UIViewController
{    
    IBOutlet UITextField *projectName;

    ViewController *viewContr;
}

- (IBAction)back:(id)sender;
- (IBAction)next:(id)sender;

@property (nonatomic, retain) ViewController *viewContr;

@end

模态呈现文件的 .m 文件:

    // AddProject.m
#import "AddProject.h"
#import "ViewController.h"

@interface AddProject ()

@end

@implementation AddProject

**@synthesize viewContr; // SOLUTION I've mist to synthesize it**

- (id)initWithNibName:(NSString *)nibNameOrNil bundle:(NSBundle *)nibBundleOrNil
{
    self = [super initWithNibName:nibNameOrNil bundle:nibBundleOrNil];
    if (self) {
        // Custom initialization
    }
    return self;
}

- (void)viewDidLoad
{
    [super viewDidLoad];
    // Do any additional setup after loading the view from its nib.
    **// SOLUTION no new allocation/init of ViewController**
}

- (IBAction)back:(id)sender
{
    [self dismissModalViewControllerAnimated:YES];
}

- (IBAction)next:(id)sender
{
    [viewContr addToTableView:projectName.text];
    [viewContr.projectsTableView reloadData];
    [viewContr.projectsTableView setNeedsDisplay];
    [self dismissModalViewControllerAnimated:YES];
}

- (void)viewDidUnload
{
    projectName = nil;
    [super viewDidUnload];
    // Release any retained subviews of the main view.
    // e.g. self.myOutlet = nil;
}

- (BOOL)shouldAutorotateToInterfaceOrientation:(UIInterfaceOrientation)interfaceOrientation
{
    return YES;
}

@end

我已经使用断点测试了代码,并且可以在 AddProject 类“viewDidLoad”开始加载时找到问题。但我不知道为什么它会重置数组。

4

3 回答 3

1

您应该在视图控制器的 init 方法中分配 projectsArray,而不是在 viewDidLoad 中。projectsArray = [[NSMutableArray alloc] init];加载视图时分配一个新的空数组。

于 2012-08-25T12:24:05.213 回答
1

ViewController您从一个屏幕导航到另一个Add project屏幕。在Add ProjectScreen 中,您添加了一个新项目,并希望将此信息委托给ViewController您来自的 Screen。

现在,您尝试ViewController在您的 中创建一个实例,您AddProject在方法中新实例化了该实例,viewDidLoad并且在添加项目时,您通过调用- (void)addToTableView:(NSString *)projectName它来向它提供信息。这不起作用,因为 -ViewController您分配的实例是一个新实例,而不是您从中导航的实例。当你这样做时,alloc + init你会得到一个新对象。因此,尽管您在addToTableView:上被调用ViewController,但它不起作用,因为ViewController接收此消息的实例是一个新实例,而不是您从中导航的实例。

因此,您需要做的就是 - 在您的 中拥有一个ViewController属性,Add Project您可以在导航之前设置它。然后您可以向此属性发送消息 - 如脚本中所述。

我想说 - 但是这个东西有效,它提供了两个控制器之间的紧密耦合。

理想情况下,这可以通过将ViewController(您从哪里导航)作为Add Project(您导航到哪里)的代表来完成。然后Add Project可以将添加的项目委托给它的委托,例如- (void) addProjectController:(AddProjectViewController *) controller didAddProject:(NSString *) projectName. 这提供了两个视图控制器之间的丢失耦合。要了解委托概念,请参阅委托和数据源。或者,搜索其他堆栈溢出帖子。

于 2012-08-25T16:25:53.403 回答
0

好的,我的问题已通过 stackoverflow 聊天解决。最终解决方案的提示来自用户@TX。我在问题的代码中添加了提示。它们标有“// SOLUTION”。

在添加项目视图控制器的 viewDidLoad 方法中,删除行 viewContr = [[ViewController alloc] init];viewContr现在让你的类中的变量 成为ViewController一个属性。也合成它。最后在- (IBAction)addNewProject:(id)sender方法中,在呈现控制器之前,执行- addProjectView.viewContr = self;. 它将解决您的问题。

(引自chat.stackoverflow | @TX

于 2012-08-25T12:46:44.607 回答