效果图
主要的逻辑
Manager封装网络请求
- 首先,对于获取网络请求,我是将这些方法封装成了一个类Manager,后续在获取以往的内容时又封装了一个beforeManager类用于网络请求。这里不多赘述,Manager封装网络请求的知识参考我的以往博客:iOS——Manager封装网络请求
- 获取到网络请求之后,使用Model层的类和JSONModel来获取接收到的内容,JSONModel的知识可以参考我以往的博客:iOS——JSONModel的使用与JSONModel的嵌套,在Model层我使用了两个类,一个mainModel用于接收一开始启动程序接收的内容,一个beforeModel用于接收以往的内容。
线程的管理
在写知乎日报的时候,遇见了线程的问题,比如说在viewController中获取Manager网络请求的内容时,因为在viewControllert中viewDidLoad执行的很早,所以如果将View层的初始化放在viewDidLoad的话,就会先去布局好UI,等网络请求好时无法将请求到的数据赋给UI控件。所以这时候需要将View层的初始化重新放在一个实例方法loadUI中,在完成网络请求之后再去调用该loadUI方法。这时候就会发现,如果我们只是将其不加修饰写在网络请求完回调的方法时,就会报错,因为View的初始化不在主线程进行。这时我们就需要使用:
dispatch_async(dispatch_get_main_queue(), ^{
[self loadUI];
});
使其在主线程中进行,才能解决问题。
同样的,在后面刷新tableView时的reloadData也要使用这个方法,是因为在iOS中,reloadData方法必须在主线程上调用。
加载网络图片
在进行网络请求时,我发现请求到的图片内容都是url,此时我们没办法直接将其转化为图片形式,所以就要使用一个第三方库:SDWebImage库,这个库可以将我们请求到的url转为图片,其用法如下:
首先,我们要导入该库:和Masonry、JSONModel这些的方法一样:pod ‘SDWebImage’ 即可
然后获取我们通过网络请求到的图片的url,并且导入SDWebImage的头文件。然后使用 SDWebImage 中的 sd_setImageWithURL: 方法将网络图片加载到 UIImageView 中。
示例:
[yourImageView sd_setImageWithURL:[NSURL URLWithString:@"图片的url"]
placeholderImage:[UIImage imageNamed:@"placeholder"]];
这将下载位于指定 URL 的图片并将其设置为 yourImageView,如果图片下载失败,将会使用 placeholder 图片作为占位符。
左上角时间的获取
这里我使用了NSDate来获取当前时间,并将时间转化为字符串,然后赋值给View层。
#import <Foundation/Foundation.h>
NS_ASSUME_NONNULL_BEGIN
@interface timeModel : NSObject
- (NSArray*)titleTimeLabel;
@end
NS_ASSUME_NONNULL_END
#import "timeModel.h"
@implementation timeModel
- (NSArray *)titleTimeLabel {
NSDate *timeDate = [NSDate date];
NSCalendar *gregorian = [[NSCalendar alloc] initWithCalendarIdentifier: NSCalendarIdentifierGregorian];
unsigned unitFlags = NSCalendarUnitYear | NSCalendarUnitMonth | NSCalendarUnitDay | NSCalendarUnitHour | NSCalendarUnitMinute | NSCalendarUnitSecond | NSCalendarUnitWeekday;
NSDateComponents *comp = [gregorian components: unitFlags fromDate: timeDate];
NSString *month = [[NSString alloc] init];
if (comp.month == 1) {
month = @"一";
}
if (comp.month == 2) {
month = @"二";
}
if (comp.month == 3) {
month = @"三";
}
if (comp.month == 4) {
month = @"四";
}
if (comp.month == 5) {
month = @"五";
}
if (comp.month == 6) {
month = @"六";
}
if (comp.month == 7) {
month = @"七";
}
if (comp.month == 8) {
month = @"八";
}
if (comp.month == 9) {
month = @"九";
}
if (comp.month == 10) {
month = @"十";
}
if (comp.month == 11) {
month = @"十一";
}
if (comp.month == 12) {
month = @"十二";
}
NSString *day = [NSString stringWithFormat:@"%ld", (long)comp.day];
NSArray * timeArr = [NSArray arrayWithObjects:month, day, nil];
return timeArr;
}
@end
单元格的刷新
这块我的代码还有问题,但是我初步写出了这个逻辑。我使用了- (void)tableView:(UITableView *)tableView willDisplayCell:(UITableViewCell *)cell forRowAtIndexPath:(NSIndexPath *)indexPath;协议方法,当即将出现某indexPath.row位置的单元格时,就调用该方法。我使用了一个全局变量numberOfCell,其初始值为1,我的单元格的行数的返回值就是5 * numberOfCell,每当调用到该方法时,numberOfCell就会加一,因此我的单元格数量刷新后就会增加5个。当触发该方法的时候,就获取存在ManagerModel类中date属性(该属性表示当天的日期的字符串),然后将该date-1就得到前一天的时间,我还定义了一个全局变量n用于表示刷新了多少天,每当刷新一次就让n+1,因此使用date-n就能得到刷新的对应天数的字符串,再将该字符串传给beforeManager的timeStr属性,该属性用来补全https://news-at.zhihu.com/api/4/news/before/%@的url,然后进行网络请求,这样我们就获得到了刷新后的内容,再将该内容赋给对应的beforeStoriesModel类的实例的stories属性,再将该属性给单元格并刷新单元格,就实现了单元格的刷新。
但是目前有个获取到的stories数组的越界问题,因此我只能刷新两次就崩了,这周我改正了这个问题会将解决方法写在下周的博客中。