gpt4 book ai didi

objective-c - 带有用户提供的上下文和 [self autorelease] 的 UIAlertView

转载 作者:塔克拉玛干 更新时间:2023-11-02 09:32:37 24 4
gpt4 key购买 nike

我查看了一些有关如何为 UIAlertView 提供上下文的想法。常见的答案是将其保存在字典或子类 UIAlertView 中。我不喜欢将上下文保存在字典中的想法,这是数据的错误位置。 Apple 不支持子类化 UIAlertView,因此按照我的标准,这不是一个好的解决方案。

我想出了一个主意,但我不确定该怎么做。创建作为 UIAlertView 委托(delegate)的上下文对象的实例。反过来,警报 View 上下文有它自己的委托(delegate),即 View Controller 。

问题是释放内存。我将 alertView.delegate 设置为 nil 并调用 [self autorelease] 以释放 -alertView:didDismissWithButtonIndex: 中的上下文对象。

问题是:我给自己带来了什么问题?我怀疑我正在为一个微妙的内存错误做好准备。

这里是简单的版本,只支持 -alertView:clickedButtonAtIndex:

使用

- (void)askUserIfTheyWantToSeeRemoteNotification:(NSDictionary *)userInfo
{
[[[[UIAlertView alloc] initWithTitle:[userInfo valueForKey:@"action"]
message:[userInfo valueForKeyPath:@"aps.alert"]
delegate:[[WantAlertViewContext alloc] initWithDelegate:self context:userInfo]
cancelButtonTitle:@"Dismiss"
otherButtonTitles:@"View", nil] autorelease] show];
}
- (void)alertView:(UIAlertView *)alertView clickedButtonAtIndex:(NSInteger)buttonIndex withContext:(id)context
{
if (buttonIndex != alertView.cancelButtonIndex)
[self presentViewForRemoteNotification:context];
}

界面

@protocol WantAlertViewContextDelegate <NSObject>
- (void)alertView:(UIAlertView *)alertView clickedButtonAtIndex:(NSInteger)buttonIndex withContext:(id)context;
@end

@interface WantAlertViewContext : NSObject <UIAlertViewDelegate>
- (id)initWithDelegate:(id<WantAlertViewContextDelegate>)delegate context:(id)context;
@property (assign, nonatomic) id<WantAlertViewContextDelegate> delegate;
@property (retain, nonatomic) id context;
@end

实现

@implementation WantAlertViewContext
- (id)initWithDelegate:(id<WantAlertViewContextDelegate>)delegate context:(id)context
{
self = [super init];
if (self) {
_delegate = delegate;
_context = [context retain];
}
return self;
}
- (void)dealloc
{
[_context release];
[super dealloc];
}
- (void)alertView:(UIAlertView *)alertView clickedButtonAtIndex:(NSInteger)buttonIndex
{
[self.delegate alertView:alertView clickedButtonAtIndex:buttonIndex withContext:self.context];
}
- (void)alertView:(UIAlertView *)alertView didDismissWithButtonIndex:(NSInteger)buttonIndex
{
alertView.delegate = nil;
[self autorelease];
}
@synthesize delegate = _delegate;
@synthesize context = _context;
@end

最佳答案

您可以使用关联对象的概念。使用函数 objc_setAssociatedObject()objc_getAssociatedObject() .您可以使用这些属性实质上添加一个新属性,在您的情况下可以保存一个 NSDictionary。 , 通过类别到对象。

这是一个 UIAlertView 的例子类别。这些文件应该在没有 ARC 的情况下编译, -fno-objc-arc如果项目正在使用 ARC,则设置标志。

UIAlertView+WithContext.h:

#import <UIKit/UIKit.h>
@interface UIAlertView (Context)
@property (nonatomic, copy) NSDictionary *userInfo;
@end

UIAlertView+WithContext.m:

#import "UIAlertView+WithContext.h"
// This enum is actually declared elseware
enum {
OBJC_ASSOCIATION_ASSIGN = 0,
OBJC_ASSOCIATION_RETAIN_NONATOMIC = 1,
OBJC_ASSOCIATION_COPY_NONATOMIC = 3,
OBJC_ASSOCIATION_RETAIN = 01401,
OBJC_ASSOCIATION_COPY = 01403
};
@implementation UIAlertView (Context)
static char ContextPrivateKey;
-(void)setUserInfo:(NSDictionary *)userInfo{
objc_setAssociatedObject(self, &ContextPrivateKey, userInfo, 3);
}
-(NSDictionary *)userInfo{
return objc_getAssociatedObject(self, &ContextPrivateKey);
}
@end

这个类别很容易使用。

SomeViewController.m: UIAlertViewDelegate是否使用 ARC。

-(void)viewDidAppear:(BOOL)animated{
[super viewDidAppear:animated];
UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"Title" message:@"Message" delegate:self cancelButtonTitle:@"OK" otherButtonTitles:nil];
alert.userInfo = [NSDictionary dictionaryWithObject:@"Hello" forKey:@"Greeting"];// autorelease if MRC
[alert show]; // release if MRC
}

-(void)alertView:(UIAlertView *)alertView didDismissWithButtonIndex:(NSInteger)buttonIndex{
NSLog(@"userInfo:%@",alertView.userInfo);
}

当您按下 alertview 的 OK 按钮时,您将看到:

userInfo:{
Greeting = Hello;
}

一些注意事项:

1) 确保关联类型与属性声明相匹配,以便一切按预期运行。

2) 你可能不应该使用 userInfo对于属性(property)/协会,因为 Apple 很可能决定添加 userInfo属性(property)UIAlertView future 。

编辑 解决您对 [self autorelease]; 的担忧

您必须平衡隐含的 alloc从这一行保留:delegate:[[WantAlertViewContext alloc] initWithDelegate:self context:userInfo] .您可以通过调用 [self autorelease]; 来实现此平衡。在决赛中 UIAlertView委托(delegate)方法。

诚然,这确实让人感觉不对。主要是因为在看这个时没有办法乍一看看起来不像内存管理不善。但是有一种简单的方法可以避免您正在创建的这种“受控泄漏”API;拥有 WantAlertViewContext 的实例明确保留自己。例如:

-(id)initWithDelegate:(id<WantAlertViewContextDelegate>)delegate context:(id)context{
self = [super init];
if (self) {
_delegate = delegate;
_context = [context retain];
}
return [self retain]; // Explicitly retain self
}

-(void)alertView:(UIAlertView *)alertView didDismissWithButtonIndex:(NSInteger)buttonIndex{
alertView.delegate = nil;
[self autorelease]; // Or just [self release]; doesn't make much difference at this point
}

现在你的类(class)内部有些和谐了。我说一些是因为这仍然不完美。例如,如果一个实例永远不是警报 View 委托(delegate),它就永远不会被释放。它仍然只是一个“半控制”的内存泄漏。

无论如何,现在您的实例化调用看起来更合乎逻辑:

delegate:[[[WantAlertViewContext alloc] initWithDelegate:self context:userInfo] autorelease];

我认为这种特殊的设计模式充满了危险。如果您最终使用它,请密切关注它。

关于objective-c - 带有用户提供的上下文和 [self autorelease] 的 UIAlertView,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10304273/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com