reason: '-[UITableViewController loadView] loaded the "GWHomeViewController" nib but didn't get a UITableView.'

uncaught exception 'NSInternalInconsistencyException, reason:[UITableViewController loadView] loaded the "Controller" nib but didn't get a UITableView - 暴走路人甲

http://blog.csdn.net/ryantang03/article/details/7941058#reply

上面那篇文章是我查找的ios实现下拉刷新功能,在我下载完代码运行的过程中发现会报下面的这个错误。经过半下午的研究我终于找到了原因。

EGOTableViewPullRefreshTest[2150:907] *** Terminating app due to uncaught exception 'NSInternalInconsistencyException', reason: '-[UITableViewController loadView] loaded the "MainViewController" nib but didn't get a UITableView.'

1.我找到的第一个解决办法就是在该网站下的一个评论:http://t6883.codeinpro.us/q/50810b8b4f1eba38a4f10fe6

If you subclass a UITableViewController on purpose, you need to change it's default view load behaviour just by overriding it's load view method:

- (void)loadView {
    [super loadView];
}

加到代码中果然可以了,由于不知道原因,又查找了一下loadView的作用,现附文章如下:

http://www.dreamingwish.com/dream-2011/correct-online-information-error-loadview-viewdidload-viewdidunload.html

一、loadView

永远不要主动调用这个函数。view controller会在view的property被请求并且当前view值为nil时调用这个函数。如果你手动创建view,你应该重载这个函数。如 果你用IB创建view并初始化view controller,那就意味着你使用initWithNibName:bundle:方法,这时,你不应该重载loadView函数。

这个方法的默认实现是这样:先寻找有关可用的nib文件的信息,根据这个信息来加载nib文件,如果没有有关nib文件的信息,默认实现会创建一个空白的UIView对象,然后让这个对象成为controller的主view。

所以,重载这个函数时,你也应该这么做。并把子类的view赋给view属性(property)(你create的view必须是唯一的实例,并且不被其他任何controller共享),而且你重载的这个函数不应该调用super。

如果你要进行进一步初始化你的views,你应该在viewDidLoad函数中去做。在iOS 3.0以及更高版本中,你应该重载viewDidUnload函数来释放任何对view的引用或者它里面的内容(子view等等)。

这个网上的资料都说的很不全面,尤其是蓝色字部分。

二、viewDidLoad

这个函数在controller加载了相关的views后被调用,而不论这些views存储在nib文件里还是在loadView函数中生成。而多数情况下是做nib文件的后续工作。

网上资料对这个函数的描述则完全不对。

三、viewDidUnload

这个函数是viewDidLoad的对立函数。在程序内存欠缺时,这个函数被controller调用()。由于controller通常保存着与 view (这里黑体的view指controller的view属性)相关的对象(一般是 view 的子view)或者其他运行时创建的对象的引用,所以你必须使用这个函数来放弃这些对象的所有权以便内存回收。但不要释放那些难以重建的数据(不要在这个函数中释放 view )。

通常controller会保存nib文件建立的views的引用,但是也可能会保存着loadView函数创建的对象的引用。最完美的方法是使用合成器方法:

self.myCertainView = nil;

这样合成器会release这个view,如果你没有使用property,那么你得自己显式释放这个view。

网上对这个函数的描述含含糊糊,看了等于没看。

另外:如果controller存储了其他object和view的引用,你还得在dealloc方法中释放这些内存。对于iOS2.x,你还必须在调用super dealloc方法前将这些引用置为nil。

四、结论

所以流程应该是这样:

(loadView/nib文件)来加载view到内存 ——>viewDidLoad函数进一步初始化这些view ——>内存不足时,调用viewDidUnload函数释放views

—->当需要使用view时有回到第一步

如此循环

2.后来我又无意中看到我我前几天写的下面这篇文章是,带颜色的部分是我着重标出的(前几天刚写,里面的重要问题就忘记了,看来非得被问题拌一脚才能长记性��)

http://www.cnblogs.com/baozou/p/3314578.html

UITableViewController继承自 UIViewController,但是initWithNibName:bundle:方法的行为是不一样的。普通的UIViewController 如果nibName参数是nil,则自动载入和自己类名相同的xib文件。而UITableViewController遇到nibName为nil时, 却不加载xib文件,而是创建一个空的table view。所以,对于UITableViewController来说,如果使用了xib文件,则必须写出完整xib文件名,才能正确创建。

因此,当使用了xib时,UITableViewController 不能 使用这样的方式创建:

[[TableViewController alloc] init];
[[TableViewController alloc] initWithNibName:nil bundle:nil];

然后我在appdelegate里面果然找到这样的代码:

MainViewController *mainView = [[ MainViewController alloc ] init ];看来果真是这个原因。

3.还有一种方法

下拉刷新的那个例子是定义的类直接继承自UITableViewController 而 我以前遇到的情况都是继承自UIViewController 然后在类中定义一个UITableView。

其实要理解这几个类之间的关系就好办了,我就是不明白这几个类之间的关系才出错。

UITableViewDelegate 

UITableViewDataSource这两个代理是在UITableView中定义的,然后放到了UITableView类中来使用,UITableViewController类包含UITableView类变量。

 

摘自:http://www.tuicool.com/articles/67VN3m

posted @ 2016-03-16 17:09  筱简單  阅读(428)  评论(0编辑  收藏  举报