「iOS」——知乎日报一二周总结

知乎日报仿写

前言

前两周内容的仿写,主要完成了首页的仿写,进度稍慢。


效果

Manager封装网络请求

知乎日报的仿写需要频繁的申请网络请求,将所有的网络请求封装成一个Manager类可以有效减少代码量,并且优化数据的使用。

具体内容在往期博客中,使用单例类进行网络请求「iOS」------AFNetworking的简单使用

此处,需要注意使用数据时,要将模型转为字典,需要在Model层自己写转换的方法。

线程冲突问题

这里我使用AFNetwokring第三方库来实现网络请求,AFNetworking 通过 NSURLSession 将所有的网络请求放入后台线程进行处理。此时就会出现线程问题。具体的问题就是

objectivec 复制代码
-(void) GetScrollerModel
{
    self.str_date = [DateModel getCurrentDateString];
    NSLog(@"%@",self.str_date);
    [[Manger sharedSingleton] NetWorkWithScroller:^(ScrollerModel *Model_Scroller) {
        
        self.dict_Scroller = [Model_Scroller ModelToDict:Model_Scroller];
        
        for(int i = 0; i < 5; i++) {
            [self.array_Scroller_ID addObject: self.dict_Scroller[@"top_stories"][i][@"id"]];
            [self.array_Scroller_URL addObject:self.dict_Scroller[@"top_stories"][i][@"url"]];
            [self.array_Scroller_Image addObject:self.dict_Scroller[@"top_stories"][i][@"image"]];
            [self.array_Scroller_title addObject:self.dict_Scroller[@"top_stories"][i][@"title"]];
            [self.array_Scroller_hint addObject:self.dict_Scroller[@"top_stories"][i][@"hint"]];
        }
        
        self.dict_data = [Model_Scroller ModelToDict:Model_Scroller];
        [self.array_data addObject:self.dict_data];
        } andError:^(NSError *error) {
            NSLog(@"GetScrollerModel错误:%@",error);
        }];
[[Manger sharedSingleton] NetWorkWithTheme:^(CellModel *Model) {
            self.dict_data = [Model ModelToDict:Model];
            NSLog(@"dice:%@",self.dict_data[@"stories"][1][@"hint"]);
            [self.array_data addObject:self.dict_data];
            
            } andError:^(NSError *error) {
                NSLog(@"CellModel错误:%@",error);
            } andNSString:(NSString *)self.str_date];
    [self setupTableView];
}

我在此进行网络数据请求时,数组的第一项可能是第一个网络请求的,也可能是第二个网络请求的。并且更新UI的实际也不确定,会导致程序崩溃。

我通过如下方法:将第二个网络申请嵌套在第一个内,并通过GCD保证线程安全。

objectivec 复制代码
-(void) GetScrollerModel
{
    self.str_date = [DateModel getCurrentDateString];
    NSLog(@"%@",self.str_date);
    [[Manger sharedSingleton] NetWorkWithScroller:^(ScrollerModel *Model_Scroller) {
        
        self.dict_Scroller = [Model_Scroller ModelToDict:Model_Scroller];
        
        for(int i = 0; i < 5; i++) {
            [self.array_Scroller_ID addObject: self.dict_Scroller[@"top_stories"][i][@"id"]];
            [self.array_Scroller_URL addObject:self.dict_Scroller[@"top_stories"][i][@"url"]];
            [self.array_Scroller_Image addObject:self.dict_Scroller[@"top_stories"][i][@"image"]];
            [self.array_Scroller_title addObject:self.dict_Scroller[@"top_stories"][i][@"title"]];
            [self.array_Scroller_hint addObject:self.dict_Scroller[@"top_stories"][i][@"hint"]];
        }
        
        self.dict_data = [Model_Scroller ModelToDict:Model_Scroller];
        [self.array_data addObject:self.dict_data];
        
        
        [[Manger sharedSingleton] NetWorkWithTheme:^(CellModel *Model) {
            self.dict_data = [Model ModelToDict:Model];
            NSLog(@"dice:%@",self.dict_data[@"stories"][1][@"hint"]);
            [self.array_data addObject:self.dict_data];
            
            dispatch_async(dispatch_get_main_queue(), ^{

                [self setupTableView];

            });
            
            } andError:^(NSError *error) {
                NSLog(@"CellModel错误:%@",error);
            } andNSString:(NSString *)self.str_date];
        } andError:^(NSError *error) {
            NSLog(@"GetScrollerModel错误:%@",error);
        }];

}

下拉刷新

上述代码强调加入数组的顺序,与此处下拉刷新有关。我下拉刷新的逻辑是通过调用-(void)scrollViewDidEndDragging:(UIScrollView *)scrollView willDecelerate:(BOOL)decelerate方法。判断当前下滑是否下滑到底。设立一个falg全局变量,如果到底就使flag++,然后根据数组上一个的date数据的Date进行网络请求。这里添加进入数组的顺序关系到网络请求的顺序,因此一定要保证是按顺序加入。

objectivec 复制代码
-(void)scrollViewDidEndDragging:(UIScrollView *)scrollView willDecelerate:(BOOL)decelerate
{
    NSString *str_new = self.array_data[1+_falg][@"date"];
    if(scrollView.contentOffset.y > self.tableView_First.contentSize.height - self.view.bounds.size.height - 50) {
        
        [[Manger sharedSingleton] NetWorkWithTheme:^(CellModel *Model) {
            
            [self.array_data addObject:[Model ModelToDict:Model]];
            
            self.falg ++;
            [self.tableView_First reloadData];
                } andError:^(NSError *error) {
                    NSLog(@"xialaerror:%@",error);
                } andNSString:(NSString *) str_new];       
    }
}

添加网络请求的图片

知乎日报的API的图片是通过URL地址保存,因此这里通过SDWebImage库来进行使用,这个库可以通过图片URL将图片加载到指定的UI控件上。此处只用到如下代码:

objectivec 复制代码
NSString *str_url = self.array_data[current][@"stories"][indexPath.row][@"images"][0];
        [cell.imagView_Small sd_setImageWithURL:[NSURL URLWithString:str_url]
                         placeholderImage:[UIImage imageNamed:@"placeholder"]];

通过时间戳和日期格式化获取时间

我们既可以通过NSDateFormatter获取时间,也可以通过时间戳来获取时间。

时间戳获取时间:将时间字符串解析为NSDate对象,并通过自1970年1月1日以来的秒数来获取时间信息。

通过时间戳获取时间的优点有:

  • 精确性:能够避免因时区变化引起的问题。
  • 通用性:时间戳可以在不同系统之间传递并且易于存储,因为它是一个简单的数字。
  • 方便进行时间计算:时间戳可以直接用于时间的加减运算,适合需要进行复杂时间计算的场景。
objectivec 复制代码
#import <Foundation/Foundation.h>

@interface DateModel : NSObject

+(NSString *)getCurrentDateString;
+(NSString *)getMonthWithDateString;

+(NSTimeInterval)getTimestampWithTimeString: (NSString *) timeString;
+(NSString *)getDateWithTimeString:(NSString *) timeString;
+(NSString *)getMouthWithTimeString:(NSString *) timeString;
+(NSString *)getDayWithTimeString:(NSString *) timeString;
+(NSString *)getBeforeDateWithTimeString:(NSString *) timeString;

@end

但是时间戳写起来略微有点麻烦,直接使用NSDateFormatter日期格式化代码更简便。

objectivec 复制代码
//时间戳形式
+(NSTimeInterval)getTimestampWithTimeString:(NSString *)timeString
{
    NSDateFormatter *dateFirmatter = [[NSDateFormatter alloc] init];
    [dateFirmatter setDateFormat:@"YYYYMMdd"];
    
    NSDate *date = [dateFirmatter dateFromString:timeString];
    
    NSTimeInterval timeStamp = [date timeIntervalSince1970];
    return timeStamp;
}

//通过日期格式化
+(NSString*)getCurrentDateString
{
    NSDateFormatter *dateFormatter = [[NSDateFormatter alloc] init];
    [dateFormatter setDateFormat:@"YYYYMMdd"];
    return [dateFormatter stringFromDate:[NSDate date]];
}

总结

首页的大部分已经完成,正在写单元格点击的响应事件。进度较慢,下星期尽量多写些。

相关推荐
2501_916008893 小时前
iOS开发者工具有哪些?Xcode、Fastlane 与 kxapp 的组合使用
ide·vscode·macos·ios·个人开发·xcode·敏捷流程
Digitally6 小时前
如何备份和恢复 iPhone:避免数据丢失(5 种方法)
ios·iphone
黑马源码库miui520868 小时前
JAVA国际版同城上门服务上门送水桶装水配送源码同城上门配送系统源码支持Android+IOS+H5
android·java·ios
2501_915921438 小时前
iPhone 定位功能测试时不越狱来修改手机位置的方法
android·ios·智能手机·小程序·uni-app·iphone·webview
spencer_tseng9 小时前
HTML5 - Android - IOS
android·ios·html·html5
90后的晨仔9 小时前
深入剖析 SDAnimatedImageView:如何优雅地在 iOS 中实现高性能动态图渲染
ios
游戏开发爱好者89 小时前
iOS 开发进阶,用 SniffMaster 实现 iPhone 抓包深度分析
android·ios·小程序·https·uni-app·iphone·webview
开心就好20251 天前
本地执行 IPA 混淆 无需上传致云端且不修改工程的方案
后端·ios
报错小能手1 天前
ios开发方向——对于实习开发的app(Robopocket)讲解
开发语言·学习·ios·swift
wechatbot8881 天前
【企业通信】基于IPAD协议的企业微信群聊管理API:群操作功能接口设计与实现
java·ios·微信·企业微信·ipad