如何提示使用UIView子类进行编译

时间:2010-11-23 09:27:32

标签: ios iphone uiviewcontroller uiwebview subclassing

我有一个UIViewController的子类,它负责一个UIWebView。

由于这是一个简单的例子,我重写-(void)loadView,实例化UIWebView并为其分配控制器的view属性:

- (void)loadView 
{
    UIWebView *wv = [[[UIWebView alloc] initWithFrame:self.frame] autorelease];
    // other configuration here...  
    self.view = wv;
}

这是好的,直到我调用UIWebView的方法。例如......

[self.view loadHTMLString:HTMLString baseURL:baseURL];

...导致编译器警告...

warning: 'UIView' may not respond to '-loadHTMLString:baseURL:'

...因为view属性声明为UIView

现在警告可以通过演员轻松解决......

[(UIWebView *)self.view loadHTMLString:HTMLString baseURL:baseURL];

...但我想做的是在界面中提供正确的类型提示。我尝试覆盖view中的MyViewController.h属性,但这也会扰乱编译器:

warning: property 'view' type does not match super class 'UIViewController' property type

有没有办法告诉编译器(和我的伙伴们)这就是我正在做的事情,而且我知道这就是我正在做的事情,这一切都没关系? (如果不是,我想我会坚持演员。)

TIA

编辑:我尝试按照marcus.ramsden的回答重新声明了view属性:这消除了警告(以及对转换的需要)但是停止了视图的显示!我不知道为什么这应该是因为控制器在被要求时仍然会返回一个UIView(子类)......

6 个答案:

答案 0 :(得分:5)

简短的回答是没有合理的方法来做到这一点。找到一种不合理的方法来解决问题会产生更多的问题。我意识到这不是你想要的答案,但为了将来读这个问题的任何人的利益,我会解释为什么我认为你不应该这样做。

view属性是UIViewController的公共接口的一部分,返回值的类型是该接口的一部分。更改已经子类化的类的公共接口是臭的。如果这对您来说不够理想,请考虑视图控制器的view属性是UIKit的基本部分;你可能搞得很少,可能会对框架产生更系统的影响。谁知道那个简单的界面吸气剂的外观下有什么样的黑暗阴谋?基于已经留在这里的一些评论,重新定义这个属性的看法以明显的方式打破了事物;我会离开。

如果你来自像ruby这样的语言,那么你不能在没有编译器警告的情况下向view属性返回的对象发送任意消息这一事实似乎并不令人满意。但是,无论好坏,Objective C都是静态(如果不是特别强烈)的类型语言。当你使用特定工具时,值得尝试使用该工具的优势,并避免其弱点,而不是试图将其变成另一种工具的习语。

另一方面,如果你来自像C ++这样的强类型语言,那个演员应该感觉非常不满意。强制转换破坏了静态类型语言严重依赖的类型系统,并且几乎总是令人讨厌的代码味道。 C型演员阵容是C型系统的核武器;它们完全覆盖了所有东西,它们几乎总是错误的使用。

如果您将来决定要在视图中添加标签以及Web视图,该怎么办?在这种情况下,您需要将Web视图移动为子视图;但如果您将视图控制器的基本视图转换为UIWebView,那么您的转换现在就是谎言。遗憾的是,编译器不会帮助您找到这些谎言,因为C样式的强制转换是未经检查的;你可以用UIView,int,块,函数指针或者rhinocerous创建一个UIWebView,编译器不会窥视。不过,你的应用程序会崩溃。

最后,你说你有兴趣避免复杂性。考虑将Web视图添加为基本视图的子视图只需要声明(和合成)单个属性并在viewDidLoad中添加子视图。考虑到这些努力,您对Web视图的所有调用都是这样的:

[self.webView loadHTMLString:HTMLString baseURL:baseURL];

或者,如果您坚持使用强制转换,那么对您的网络视图的每次调用都将如下所示:

[(UIWebView *)self.view loadHTMLString:HTMLString baseURL:baseURL];

前者对我来说似乎不太复杂,而且不那么脆弱。

答案 1 :(得分:2)

由于它是一个子类,您可以将view属性/方法覆盖为UIWebView。由于UIWebViewUIView的子类,并且您是子类的唯一用户,因此这应该是完全安全的。

@interface MyAbstractViewController : UIViewController {
    UIWebView *_webView;
}
@property (nonatomic, retain) UIWebView *view;
@end

@implementation MyAbstractViewController    
@dynamic view;
- (void)setView:(UIWebView *)webView {
  [super setView:webView];
  _webView = webView;
}

- (UIWebView *)view {
  return _webView;
}
@end

答案 2 :(得分:2)

您可以定义属性

@property (nonatomic, readonly) UIWebView *webView;

并执行此操作:

- (UIWebView*)webView {
   return (UIWebView*)self.view;
}

- (void)loadView 
{
    UIWebView *wv = [[[UIWebView alloc] initWithFrame:self.frame] autorelease];
    // other configuration here...  
    self.view = wv;
}

然后您可以self.webView随后访问您的WebView而不进行任何转换。从本质上讲,您将演员阵容集中到代码中的某个特定位置。如果您想插入另一个视图,例如标签,则只需修改代码中的单个位置即可。

答案 3 :(得分:0)

从文档中看起来可以通过使用类别来执行您想要的操作。例如;

// UIWebViewController.h
@interface UIWebViewController : UIViewController {

}

@end

@interface UIWebViewController ()

@property (nonatomic,retain) UIWebView *view;

@end

// UIViewController.m
@implementation UIWebViewController

@synthesize view;

- (void)viewDidLoad {
    UIWebView *wv = [[UIWebView alloc] initWithFrame:webViewFrame];
    self.view = wv;
    [self.view loadHTML:@"<h1>Hello World</h1>" baseURL:baseURL];
    [wv release];
}

@end

在声明属性&gt;中的http://developer.apple.com/library/ios/#documentation/Cocoa/Conceptual/ObjectiveC/Articles/ocProperties.html中有一些相关内容。使用属性&gt;财产重新申报。

试过这个,我没有得到任何编译器投诉。

答案 4 :(得分:0)

转换值或向viewController添加新的UIWebView实例变量。

UIViewController的view方法的工作不是告诉你的伙伴你使用了什么类的子类。

如果你能做你想做的事,你只需要在所有地方都回到UIView,UIKit希望UIViewController的view方法返回一个UIView。

答案 5 :(得分:-1)

正如已经提到的那样,尽管可能这样做,但它可能是明智的

您可以在UIView本身上声明一个类别,表明它可能会响应您尝试调用的选择器。这可以在自定义视图控制器的实现文件中完成。它看起来像这样:

@interface UIView (WebViewFakery)

- (void)loadHTMLString:(NSString *)string baseURL:(NSURL *)baseURL

@end

您必须添加您尝试在view上调用的所有UIWebView方法。也许不是最优雅的解决方案,而是另一个需要考虑的解决方案。