我在尝试从JSON解析对象后填充表视图。我想将行数设置为已解析对象的计数。
这是我的viewDidLoad代码:
- (void)viewDidLoad
{
[super viewDidLoad];
// initializing data source
[self setNotifications:[[NSMutableArray alloc] init]];
NSURL *url = [NSURL URLWithString:SourceURL];
NSURLRequest *request = [NSURLRequest requestWithURL:url];
AFJSONRequestOperation *operation = [AFJSONRequestOperation JSONRequestOperationWithRequest:request success:^(NSURLRequest *request, NSHTTPURLResponse *response, id JSON) {
[self setNotifications:[JSON objectForKey:@"notifications"]];
// [[self tableView] setHidden:NO];
[[self tableView] reloadData];
} failure:^(NSURLRequest *request, NSHTTPURLResponse *response, NSError *error, id JSON) {
NSLog(@"Request Failure Because %@",[error userInfo]);
}];
[operation start];
}
这是我的一个数据源方法(我相信错误必须在这里):
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
{
if ([self notifications] && [[self notifications] count] ) {
return [[self notifications] count];
} else{
return 0;
}
}
这是我的完整错误消息:
> *** Terminating app due to uncaught exception 'NSRangeException', reason: '*** -[__NSArrayI objectAtIndex:]: index 1 beyond bounds [0 ..
> 0]'
> *** First throw call stack: (0x13c4012 0x118ce7e 0x1379b44 0x510c04 0x300ea0 0x2b5c4e 0x2b8224 0x17c952 0x17c2dc 0xc4a9d14 0x26e0 0x16ea2
> 0x18409 0x162753f 0x1639014 0x16297d5 0x136aaf5 0x1369f44 0x1369e1b
> 0x22647e3 0x2264668 0xd0ffc 0x1f1d 0x1e45) libc++abi.dylib: terminate
> called throwing an exception
更新:
这是我的cellForRowAtIndexpath方法:
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
static NSString *cellID = @"readNotification";
UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:cellID];
if (!cell){
cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleSubtitle reuseIdentifier:cellID];
}
NSDictionary *notification = [[self notifications] objectAtIndex:[indexPath row]];
[[cell textLabel] setText:[notification objectForKey:@"from"]];
[[cell detailTextLabel] setText:[notification objectForKey:@"date"]];
return cell;
}
更新2
我在项目中添加了一些日志,它提供了一些有趣的输出:
2013-08-06 23:53:03.994 NotificationReader [1193:c07] 1.请求前 2013-08-06 23:53:04.023 NotificationReader [1193:c07] 2. json解析 操作开始2013-08-06 23:53:04.031 NotificationReader [1193:c07] 3.输入numberOfRowsInSection块2013-08-06 23:53:04.040 NotificationReader [1193:c07]没有通知2013-08-06 23:53:04.319 NotificationReader [1193:c07] 3.输入numberOfRowsInSection block 2013-08-06 23:53:04.320 NotificationReader [1193:c07]有 14通知2013-08-06 23:53:08.260 NotificationReader [1193:c07] * 由于未捕获的异常而终止应用
首先,它认为我的数组是空的,然后重新加载并获得正确的通知计数。不幸的是,它永远不会进入cellForRowAtIndexPath方法..
更新3
我已经为我的项目添加了一个异常断点,并且还做了一些额外的日志记录,我假设,它在某种程度上无法重新加载表视图。到目前为止,结果如下:
2013-08-07 08:40:58.104 NotificationReader[433:c07] 1. before request
2013-08-07 08:40:58.107 NotificationReader[433:c07] 2. json parshing operation started
2013-08-07 08:40:58.113 NotificationReader[433:c07] 3. entered the numberOfRowsInSection block
2013-08-07 08:40:58.120 NotificationReader[433:c07] Found 0 notifications!
2013-08-07 08:41:10.852 NotificationReader[433:c07] Trying to reload data
2013-08-07 08:41:10.853 NotificationReader[433:c07] data needs to be reloaded
2013-08-07 08:41:10.854 NotificationReader[433:c07] 3. entered the numberOfRowsInSection block
2013-08-07 08:41:10.855 NotificationReader[433:c07] Found 14 notifications!
2013-08-07 08:41:10.858 NotificationReader[433:c07] *** Terminating app due to uncaught exception 'NSRangeException', reason: '*** -[__NSArrayI objectAtIndex:]: index 1 beyond bounds [0 .. 0]'
*** First throw call stack:
(0x13c5012 0x118de7e 0x137ab44 0x511c04 0x301ea0 0x2b6c4e 0x2b9224 0x17d952 0x17d2dc 0xc4c0d14 0x351b 0x162853f 0x163a014 0x162a7d5 0x136baf5 0x136af44 0x136ae1b 0x22657e3 0x2265668 0xd1ffc 0x2c4d 0x2b75)
libc++abi.dylib: terminate called throwing an exception}
堆栈追踪:
#0 0x0118de52 in objc_exception_throw ()
#1 0x0137ab44 in -[__NSArrayI objectAtIndex:] ()
#2 0x00511c04 in -[UITableViewDataSource tableView:heightForRowAtIndexPath:] ()
#3 0x00301ea0 in -[UITableViewController tableView:heightForRowAtIndexPath:] ()
#4 0x002b6c4e in -[UISectionRowData refreshWithSection:tableView:tableViewRowData:] ()
#5 0x002b9224 in -[UITableViewRowData numberOfRows] ()
#6 0x0017d952 in -[UITableView noteNumberOfRowsChanged] ()
#7 0x0017d2dc in -[UITableView reloadData] ()
#8 0x0c4a9d14 in -[UITableViewAccessibility(Accessibility) reloadData] ()
#9 0x0000351b in __36-[NRTableViewController viewDidLoad]_block_invoke_2 at /Users/jozsef/Development/NotificationReader/NotificationReader/NRTableViewController.m:52
我尝试将lldb与此结果一起使用:
(lldb) po [[self notifications] count]
error: Execution was interrupted, reason: Attempted to dereference an invalid pointer..
The process has been returned to the state before execution.
答案 0 :(得分:2)
这是因为尝试访问超出数组范围的索引。从外观上看,唯一的数组访问权限在-tableView:cellForRowAtIndexPath:
:
NSDictionary *notification = [[self notifications] objectAtIndex:[indexPath row]];
我希望这是引发异常的地方。
您可以通过添加"添加例外断点来确定..."来自断点导航器。
通过确认[indexPath row]
在[self notifications]
数组的范围内来解决此问题。或者如果你期望这应该总是在界限之内,那么找出它为什么会超出界限。
另外,您可以按如下方式重写其中一种方法:
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
{
return [self.notifications count];
}
如果self.notifications为nil,则返回0.如果count为0,则显然会返回0.这就是所有需要的。
编辑:最后,您还可以使用下标重写数组访问权限,如下所示:
NSDictionary *notification = self.notifications[indexPath.row];
确保在AFJSONRequestOperation的成功块中调用主线程上的UI更新,即
AFJSONRequestOperation *operation = [AFJSONRequestOperation JSONRequestOperationWithRequest:request success:^(NSURLRequest *request, NSHTTPURLResponse *response, id JSON) {
// update on the main thread!
dispatch_async(dispatch_get_main_queue(), ^{
[self setNotifications:[JSON objectForKey:@"notifications"]];
// [[self tableView] setHidden:NO];
[[self tableView] reloadData];
});
} failure:^(NSURLRequest *request, NSHTTPURLResponse *response, NSError *error, id JSON) {
NSLog(@"Request Failure Because %@",[error userInfo]);
}];
答案 1 :(得分:0)
评论者是正确的,您应该通过设置断点和单步执行代码来调试此类事情。使用日志语句补充此方法。
您还应该在Xcode中设置“All Exceptions Breakpoint”(命令-6,然后点击左下角的+按钮)。
numberOfRowsInSection
可能不会发生错误。是否可能发生在cellForRowAtIndexPath
中:
NSDictionary *notification = [[self notifications] objectAtIndex:[indexPath row]];
正如您nil
和count
检查numberOfRowsInSection
一样,您应该在此处执行相同操作:
if (self.notifications) {
if (indexPath.row < [self.notifications count]) {
NSDictionary *notification = [[self notifications] objectAtIndex:[indexPath row]];`
//...
}
}
我的猜测是,JSON响应不是您所期望的,因此self.notifications
不是您所期望的。
答案 2 :(得分:0)
在不同的机器上运行代码之后,它似乎都应该正常工作。我的配置中肯定存在一些垃圾,所以我再次添加了项目并确认它正在运行。 感谢所有回复的人! @Sam,特别感谢您的测试和耐心!
答案 3 :(得分:0)
哦Sam's answer非常好,但是在我意识到我的Table view
Content
设置为 Static Cells
之前,即使做了他提到的一切也没有成功。 而不是故事板中的 Dynamic Prototypes
!
即使您像这样定义自己的numberOfRowsInSection
:
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:
(NSInteger)section
{
return [self.notifications count];
}
如果它不是动态的,它很可能会失败;)