我的iOS入门第三课

简介

承上两篇我的iOS入门第一课,我的iOS入门第二课,本次记录的是一年前写的MyLocations,当时是用swift,前几天用OC重写了一遍,感触良多。主要功能是记录个人走过的地方,加以描述和照片在手机里留下美好的记忆。如下:

我的iOS入门第三课_第1张图片
主要界面.png

下面主要讲我学到了什么:

一、基本控件
UITableView,MKMapView
二、框架
CoreLocation、CoreData、MapKit、CoreGraphics、CoreAnimation、AudioToolbox

重点知识:

  • 本app尝试使用Autoresizing做布局
  • UITabBarController管理的体系
  • TabBar第一个界面主要做一件事,获取用户GPS坐标并转换成地址
    使用CoreLocation获取地址
    • CLLocationManager 获取GPS坐标(经纬度)通过代理通知
    • startUpdatingLocation 调用此方法得到、更新坐标
    • desiredAccuracy 坐标精确度 (精确到十米:kCLLocationAccuracyNearestTenMeters)
      CLLocationManagerDelegate 代理方法
    • locationManager:didFailWithError (error -> CLError)
      • kCLErrorLocationUnknown 没找着,但CL仍继续在找
      • kCLErrorDenied 没开定位服务
      • kCLErrorNetwork 网络错误
    • locationManager:didUpdateLocations 此方法会调用多次,每次得到的location坐标都会更精确。
      关于CLLocation:
      • timestamp location创建的时间
      • horizontalAccuracy 精确度,小于0则代表invalid,不准的
  • 在获取坐标前必须获得用户权限,使用定位
CLAuthorizationStatus authStatus = [CLLocationManager authorizationStatus];
    if (authStatus == kCLAuthorizationStatusNotDetermined) {
        [_locationManager requestWhenInUseAuthorization];
        return;
    }
    if (authStatus == kCLAuthorizationStatusDenied || authStatus == kCLAuthorizationStatusRestricted) {
        [self showLocationServicesDeniedAlert];
        return;
    }

并在info.plist中加上

key:NSLocationWhenInUseUsageDescription 
type:String 
value:This app lets you keep track of interesting places. It needs access to the GPS coordinates for your location. 
  • CLLocation得到的是坐标,最终还得用CLGeocoder
    转换成地址 -> CLPlacemark

    • geocoder reverseGeocodeLocation:completionHandler 得到CLPlacemark (按以下顺序写地址)
      • subThoroughfare
      • thoroughfare
      • locality
      • administrativeArea
      • postalCode
  • 第一个界面得到的地址希望被描述并存储起来,则进入到Location detail界面。关于Core Data:

    • 新建DataModel,生成一个.xcdatamodeld的文件
    • 新建实体,增加属性,类型是其它类型则选Transformable,根据环境选择是否是optional(右边third tab)
    • 从Core Data中取出的对象都是NSManagedObject
      我们通常继承NSManagedObject实现自己的子类。Editor → Create NSManagedObject Subclass
    • 选择要生成的实体。 生成两种文件,当数据实体发生改变时只需要再次Create NSManagedObject Subclass,只会改变CoreDataProperties这个分类文件。在另一钟文件中写模型方法。
    • 在CoreDataProperties分类文件中把已知的属性类型修正过来(id -> CLPlaceMark)
    • 若一开始没有用Core Data,则新建一个使用Core Data的项目,把在AppDelegate中两个文件多出来的Core Data代码全部复制到本项目中。
      我们只需要知道这些代码做了两件事
      • 加载数据模型
      • 连接SQLite数据库

    具体来说就是:

    1. 找到bundle中先前创建的“DataModel.momd”路径,转化成URL
    2. 根据URL创建NSManagedObjectModel,这个对象代表所有数据模型,我们可以向它要实体,但一般我们不需要直接使用这个对象
    3. 创建指向SQLite数据库的URL. SQLite文件在Documents文件夹中
    4. 创建NSPersistentStoreCoordinator,这个对象管理SQLite数据库
    5. 把SQLite数据库加到coordinator
    6. 最后创建NSManagedObjectContext
  • NSManagedObjectContext 这个对象用来与Core Data沟通,我们通常在context上做改变,然后调用它的save方法就可以存储到数据库中。
    向数据库中插入一条数据

    • 对象 = [NSEntityDescription insert…]
    • 设置对象属性
    • 调用context save()
      删除
    1. 向fetchedResultsController 拿到要删除的对象
    2. 调用context deleteObject:方法
    3. 调用context save方法
      查询
    4. 创建NSFetchRequest
    5. 创建要查询的实体 [NSEntityDescription entityForName…], 并设置request.entity = entity
    6. 排序,根据字段key排序 [NSSortDescriptor key: ascending:], 并设置request.sortDescriptors = @[]
    7. 调用context executeFetchRequest:执行查询得到结果
  • 在tableView中,通常不用以上这种查询方式,CoreData提供了NSFetchedResultsController来与tableView更加密切合作。(TabBar第二个界面)
    使用懒加载NSFetchedResultsController,前3步都与以上相同。
    4.选择性设置fetchRequest.fetchBatchSize = 20
    5.创建NSFetchedResultsController,sectionNameKeyPath为j结果分组的key,cacheName为缓存的名字(每次performFetch都要把cache删除)
    6.设置代理NSFetchedResultsControllerDelegate,4个代理方法用来更新TableView,具体看github代码。

    • 调用fetchedResultsController performFetch: 从数据库中取数据
      调用fetchedResultsController objectAtIndexPath:得到对应的某个数据
  • 调用相册和照相机

UIImagePickerController *imagePicker = [[MyImagePickerController alloc] init];
    imagePicker.sourceType = UIImagePickerControllerSourceTypeCamera; // PhotoLibrary为相册
    imagePicker.delegate = self;
    imagePicker.allowsEditing = YES;
    imagePicker.view.tintColor = self.view.tintColor;
    [self presentViewController:imagePicker animated:YES completion:nil];
  • 必须要遵循UIImagePickerControllerDelegate,UINavigationControllerDelegate,只需要实现ImagePicker的两个代理方法
-(void)imagePickerController:(UIImagePickerController *)picker didFinishPickingMediaWithInfo:(NSDictionary *)info {
    _image = info[UIImagePickerControllerEditedImage];//取得照片
    if (_image) {
        [self showImage:_image];
    }
    [self.tableView reloadData];
    [self dismissViewControllerAnimated:YES completion:nil];   
}
-(void)imagePickerControllerDidCancel:(UIImagePickerController *)picker {
    [self dismissViewControllerAnimated:YES completion:nil];
}
  • MKMapView (TabBar第三个界面)
    • 显示自己的位置
MKCoordinateRegion region = MKCoordinateRegionMakeWithDistance(self.mapView.userLocation.coordinate, 1000, 1000); // 1000米的度量
    [self.mapView setRegion:[self.mapView regionThatFits:region] animated:YES];
  • 显示其它坐标位置,其它坐标必须遵守MKAnnotation协议,并实现三个属性。为了把所有的坐标都显示在地图上,首先遍历坐标得到最小最大经度和最小最大纬度,从而计算出所有坐标的中心center,再用最大经度-最小经度,最大纬度-最小纬度得到要显示的范围span,为了不让坐标显示在屏幕边缘,把范围再乘以1.1,最后调用MKCoordinateRegionMake(center, span)得到最终显示的region. 再调用mapView setRegion:即可。具体看github代码
  • 自定义MKPinAnnotationView(就是那个插在地图上的竹竿)
    类似自定义tableView上的Cell,可复用。遵循MKMapViewDelegate代理协议,实现mapView:viewForAnnotation:方法。

总结

此app是对tableView使用的进一步加强,还有对CoreData的认识和使用,并且还有对CoreLocation和MapView的理解,初涉CoreAnimation之CABasicAnimation,等等。细节的处理、熟悉api,精密的逻辑,让我对iOS开发又上升了一个层次。不说了,继续加油!
源代码传送门

卧薪藏胆,三千越甲可吞吴。

你可能感兴趣的:(我的iOS入门第三课)