#import "MainViewController.h" #import "Video.h" #define kBaseURL @"http://192.168.3.252/~apple" @interface MainViewController ()<UITableViewDataSource, UITableViewDelegate> @property (strong, nonatomic) NSArray *dataList; @property (weak, nonatomic) UITableView *tableView; @end @implementation MainViewController
</pre><p class="p1">/*</p><p class="p2"><span class="s1"> </span>在網絡應用開發中,</p><p class="p2"><span class="s1"> 1 </span>數據是同步載入的,可以保證用戶有的看</p><p class="p2"><span class="s1"> 2 </span>圖像。音頻,視頻是異步載入的。可以保證在不堵塞主線程的使用的前提下。用戶可以漸漸的看到多媒體信息。</p><p class="p1"> */</p><pre code_snippet_id="412158" snippet_file_name="blog_20140630_1_3481337" name="code" class="objc"> #pragma mark 實例化視圖 - (void)loadView { self.view = [[UIView alloc]initWithFrame:[UIScreen mainScreen].applicationFrame]; //1 tableview CGRect frame = self.view.bounds; UITableView *tableView = [[UITableView alloc]initWithFrame:CGRectMake(0, 0, frame.size.width, frame.size.height - 44) style:UITableViewStylePlain]; //1)數據源 [tableView setDataSource:self]; //2)代理 [tableView setDelegate:self]; //)設置表格高度 [tableView setRowHeight:80]; [self.view addSubview:tableView]; self.tableView = tableView; //toolBar UIToolbar *toolBar = [[UIToolbar alloc]initWithFrame:CGRectMake(0, tableView.bounds.size.height, 320, 44)]; [self.view addSubview:toolBar]; //加入toolbarbutton UIBarButtonItem *item1 = [[UIBarButtonItem alloc]initWithTitle:@"load json" style:UIBarButtonItemStyleDone target:self action:@selector(loadJson)]; UIBarButtonItem *item2 = [[UIBarButtonItem alloc]initWithTitle:@"load xml" style:UIBarButtonItemStyleDone target:self action:@selector(loadXML)]; UIBarButtonItem *item3 = [[UIBarButtonItem alloc]initWithBarButtonSystemItem:UIBarButtonSystemItemFlexibleSpace target:nil action:nil]; [toolBar setItems:@[item3, item1, item3, item2, item3]]; } #pragma mark -uitableview數據源方法 對於uitableview如下這兩個方法是必須實現的。 - (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section { return self.dataList.count; } //每填充一行都調用一次這種方法。知道界面上的所有行都填充完畢。, - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath { //使用可充用標示符查詢可重用單元格 static NSString *ID = @"MyCell"; UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:ID]; if (cell == nil) { cell = [[UITableViewCell alloc]initWithStyle:UITableViewCellStyleSubtitle reuseIdentifier:ID]; } //設置單元格內容 Video *v = self.dataList[indexPath.row]; cell.textLabel.text = v.name; cell.detailTextLabel.text = v.teacher; //載入圖片 //1)同步載入網絡圖片,同步方法覺得這這些指令在完畢以前。興許指令都沒法運行。 //注意:在開發網絡應用時,不要使用同步方法載入圖片,不然會嚴重影響用戶體驗 // NSString *imagePath = [NSString stringWithFormat:@"%@%@", kBaseURL, v.imageURL]; // NSURL *imageUrl = [NSURL URLWithString:imagePath]; // NSData *imageData = [NSData dataWithContentsOfURL:imageUrl]; // UIImage *image = [UIImage imageWithData:imageData]; // // //2)異步載入網絡圖片 // //網絡鏈接自己就有異步命令 sendAsync // [cell.imageView setImage:image]; //假設緩存圖像不存在 if (v.cacheImage == nil) { //使用默認圖像佔位,即可以保證有圖像。又可以保證有地方。 UIImage *image = [UIImage imageNamed:@"user_default.png"]; [cell.imageView setImage:image]; //使用默認圖像佔位 //開啓異步鏈接,載入圖像。因爲載入完畢以後,需要刷新相應的表格航 [self loadImageAsyncWithIndexPath:indexPath]; }else { [cell.imageView setImage:v.cacheImage]; } //[self loadImageAsyncWithUrl:imageUrl imageView:cell.imageView]; return cell; } #pragma mark 異步載入網絡圖片 //因爲uitableview是可重用的,爲了不用戶高速頻繁刷新表格,形成數據衝突。不能直接將uiimageview傳入異步方法 //正確的解決方法是:將表格行的indexpath傳入異步方法。載入完畢圖像之後,直接刷新指定的行。 - (void)loadImageAsyncWithIndexPath:(NSIndexPath *)indexPath { Video *v = self.dataList[indexPath.row]; //取出當前要填充的行 NSString *imagePath = [NSString stringWithFormat:@"%@%@", kBaseURL, v.imageURL]; NSURL *imageUrl = [NSURL URLWithString:imagePath]; //NSLog(@"%@ %@", url, imageView); //1 request NSURLRequest *request = [NSURLRequest requestWithURL:imageUrl]; //2 connection sendasync異步請求 [NSURLConnection sendAsynchronousRequest:request queue:[NSOperationQueue mainQueue] completionHandler:^(NSURLResponse *response, NSData *data, NSError *error) { //UIImage *image = [UIImage imageWithData:data]; //[imageView setImage:image]; //將網絡數據保存至緩存圖像。 v.cacheImage = [UIImage imageWithData:data]; //刷新表格 [self.tableView reloadRowsAtIndexPaths:@[indexPath] withRowAnimation:UITableViewRowAnimationLeft]; }]; } #pragma mark 處理json數據 - (void)handlerJSONData:(NSData *)data { //json文件裏的[]表示一個數據。 //反序列化json數據 /* 序列化: 將一個nsboject轉換成序列數據,以便經過互聯網進行傳輸。 反序列化:將網絡上獲取的數據反向生成咱們需要的對象。web
*/ //第二個參數是解析方式。通常用NSJSONReadingAllowFragments NSArray *array = [NSJSONSerialization JSONObjectWithData:data options:NSJSONReadingAllowFragments error:nil]; NSLog(@"%@", array); //json解析之後是nsarray格式的數據。 //提示:假設開發網絡應用。可以將反序列化出來的對象,保存至沙箱。以便興許開發使用。 NSArray *docs = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES); NSString *path = [docs[0]stringByAppendingPathComponent:@"json.plist"]; [array writeToFile:path atomically:YES]; //把array裏面的數據寫入沙箱中的jspn.plist中。 //給數據列表賦值 NSMutableArray *arrayM = [NSMutableArray array]; for (NSDictionary *dict in array) { Video *video = [[Video alloc]init]; //給video賦值 [video setValuesForKeysWithDictionary:dict]; [arrayM addObject:video]; } self.dataList = arrayM; //刷新表格 [self.tableView reloadData]; NSLog(@"%@", arrayM); //這句話將調用video裏面的description和nsarray+log裏面的descriptionWithLocale } #pragma mark 載入json - (void)loadJson { NSLog(@"load json"); //從webserver載入數據 NSString *str = @"http://www.baidu.com?format=json"; //這裏是亂寫的 //提示:NSData自己具備同步方法。但是在實際開發中。不要使用次方法 //在使用NSData的同步方法時,沒法指定超時時間。假設server鏈接不正常,會影響用戶體驗。json
//NSData *data = [NSData dataWithContentsOfURL:[NSURL URLWithString:str]]; //簡歷NSURL NSURL *url = [NSURL URLWithString:str]; //創建NSURLRequest NSURLRequest *request = [NSURLRequest requestWithURL:url cachePolicy:NSURLRequestUseProtocolCachePolicy timeoutInterval:2.0f]; //創建NSURLConnect的同步方法載入數據 NSURLResponse *response = nil; NSError *error = nil; //同步載入數據 NSData *data = [NSURLConnection sendSynchronousRequest:request returningResponse:&response error:&error]; //錯誤處理 if (data != nil) { //如下這兩句話自己沒有什麼意義,僅用於跟蹤調試。 NSString *result = [[NSString alloc]initWithData:data encoding:NSUTF8StringEncoding]; NSLog(@"%@", result); //在處理網絡數據的時候,不要將NSData轉換成nsstring。 [self handlerJSONData:data]; }else if (data == nil && error == nil){ NSLog(@"空數據"); }else { NSLog(@"%@", error.localizedDescription); } } #pragma mark 載入xml - (void)loadXML { NSLog(@"load xml"); } //- (void)viewDidLoad //{ // [super viewDidLoad]; //} @end 緩存