2

我今天在想这个,现在我测试了我有点困惑......</p>

当通过将视图控制器推送到导航堆栈或以模态方式呈现视图控制器来使用视图控制器时,我想知道内存管理。

让我们使用模态示例作为思想实验,这里是创建和呈现视图的源代码,在我的示例中,ARC 与否无关紧要,所以两者都是:

使用 ARC:

ViewController *myViewController = [[ViewController alloc] init];  
myViewController.delegate = self;
[self presentViewController:myViewController animated:YES completion:NULL];

没有 ARC:

ViewController *myViewController = [[ViewController alloc] init];  
myViewController.delegate = self;
[self presentViewController:myViewController animated:YES completion:NULL];
[myViewController release];                   //As it's now 'owned' by the presenting View controller

这将是我对如何在现有 ViewController 上以模态方式呈现 viewController 的理解。

对于我们的示例,假设上面的代码位于一个方法中,当触摸按钮以呈现 ViewController 时调用该方法。

现在回答我的问题,

我正在做的是每次触摸按钮时调用此代码,在使用 Instruments 进行测试期间,我似乎没有任何泄漏。-但是,因为我在 myViewController deallocviewDidLoad方法中有 NSLog 语句,所以我知道每次我触摸按钮时它都会被实例化,但从未解除分配。

所以...

A)为什么我没有在仪器中显示泄漏或 Live Bytes 增加)(无论是否使用 ARC),因为我似乎在创建一个新的 viewController 并且每次我去展示它时都会泄漏旧的。

B)如果这不是内存安全的,那么编写上述代码的正确方法是什么?我在 Apple 的示例代码和互联网上都看到了这种代码片段。我(和他们)是否应该不将 alloc init 行包装在if语句中以检查对象是否已创建?

IE

if(!myViewController)
{
    ViewController *myViewController = [[ViewController alloc] init];  
}
myViewController.delegate = self;
[self presentViewController:myViewController animated:YES completion:NULL];

感谢您花时间阅读和回答,我真的很想知道这一点,因为我一直在使用上面的代码创建、推送和呈现 ViewController,并且从未注意到泄漏!- 可能不得不回去重写它!

为避免混淆,请注意:委托属性是我的 UIViewController 子类的自定义属性(我在其中实现了委托协议),需要正确关闭模态呈现的 Viewcontroller。根据编码指南。

问候,约翰

根据要求编辑,创建代表:

。H

@protocol NotificationManagementViewControllerDelegate;

@interface NotificationManagementController : 
{
    __weak NSObject <NotificationManagementViewControllerDelegate> *delegate;
}
@property (nonatomic, weak) NSObject <NotificationManagementViewControllerDelegate> *delegate;
@protocol NotificationManagementViewControllerDelegate <NSObject>

@optional
- (void)didFinishSettingNotification:(NotificationManagementController *)notificationManagementController;

.m

- (void)sendMessageToDismiss {
    if ([[self delegate] respondsToSelector:@selector(didFinishSettingNotification:)]) {
        [self.delegate didFinishSettingNotification:self];
    }
}

最后在代表 .m 中:

- (void)didFinishSettingNotification:(NotificationManagementController *)notificationManagementController
{
    [self dismissViewControllerAnimated:YES completion:NULL];
}
4

1 回答 1

1

你没有得到泄漏,因为你创建了一个新的控制器并且 ARC 会为你释放这个分配。

但是,最好@property为您的新视图控制器创建一个。并修改您的 ie 实现,如:

@property (nonatomic, strong) ViewController *myViewController;

if (!_myViewController)
    self.myViewController = [[ViewController alloc] init];  

self.myViewController.delegate = self;
[self presentViewController:_myViewController animated:YES completion:nil];

在这里,您有一个惰性属性,并且在第一次创建后不会创建新的 ViewController。但是,您需要在测试之外通过您的委托(或任何属性)。

此外,如果您使用您的第一个实现并将此控制器添加到当前控制器的子视图中而没有属性,这将起作用,但您会得到泄漏。我通过以下代码获得了这种经验:

根视图控制器

- (void)viewDidLoad
{
    [super viewDidLoad];

    ViewController *myViewController = [[ViewController alloc] init]; 
    [self.view addSubview:myViewController.view];
}

myViewController将被添加到屏幕上但立即释放而不保留对象的任何引用,因此如果您在“ViewController”中添加操作,您的应用程序将在没有 XCode 解释的情况下崩溃。

所以,写这个不泄漏的正确方法是:

- (void)viewDidLoad
{
    [super viewDidLoad];

    if (!_myViewController)
        self.myViewController = [[ViewController alloc] init];

    [self.view addSubview:self.myViewController.view];
}

答案有点长,可以改进,所以不要犹豫!希望它会帮助一些人。

于 2013-11-07T12:53:17.600 回答