我有一个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(子类)......
答案 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
。由于UIWebView
是UIView
的子类,并且您是子类的唯一用户,因此这应该是完全安全的。
@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方法。也许不是最优雅的解决方案,而是另一个需要考虑的解决方案。