iOS开发 程序后台上传位置CLLocationManager

2024-03-05 04:18

本文主要是介绍iOS开发 程序后台上传位置CLLocationManager,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

之前开发一款配送员用的APP时,用到了在程序在后台时,可以不断上传位置的功能,今天略微整理了一下,

主要用到系统的CoreLocation

代码:

#import <Foundation/Foundation.h>
#import <UIKit/UIKit.h>@interface XSDLocationTools : NSObject
+ (XSDLocationTools *)shareInstance;
// 开启定位
- (void)startLocationService;
@end

.m

//
//  XGLocationTool.m
//  XGPayDemo
//
//  Created by 小广 on 16/4/25.
//  Copyright © 2016年 小广. All rights reserved.
//#import "XSDLocationTools.h"
#import <CoreLocation/CoreLocation.h>
//#import "WGS84ToGCJ02.h"
#import "BaiduMapDefine.h"
#import "XSDBaiduMapTools.h"#define LAST_LONG  @"last_longitude"  // 上次上传位置的经度
#define LAST_LATI  @"last_latitude"  // 上次上传位置的纬度@interface XSDLocationTools ()<CLLocationManagerDelegate>
{//dispatch_source_t _timer;CLLocationCoordinate2D _newCoor;
}
// 1.设置位置管理者属性
@property (nonatomic, strong) CLLocationManager *lcManager;
//@property (nonatomic, assign) BOOL isRequest;
@property (nonatomic, strong) NSTimer *uploadTimer;@end@implementation XSDLocationTools+ (XSDLocationTools *)shareInstance {static XSDLocationTools *instance = nil;static dispatch_once_t onceToken;dispatch_once(&onceToken, ^{instance = [[XSDLocationTools alloc] init];[instance p_addNSNotificationObserver];});return instance;
}// 开启定位
- (void)startLocationService {if ([CLLocationManager locationServicesEnabled]) {// 创建位置管理者对象self.lcManager = [[CLLocationManager alloc] init];self.lcManager.delegate = self; // 设置代理// 设置定位距离过滤参数 (当本次定位和上次定位之间的距离大于或等于这个值时,调用代理方法)self.lcManager.distanceFilter = 50;self.lcManager.desiredAccuracy = kCLLocationAccuracyBest; // 设置定位精度(精度越高越耗电)// 2、在Info.plist文件中添加如下配置://(1)NSLocationAlwaysUsageDescription 授权使应用在前台后台都能使用定位服务//(2)NSLocationWhenInUseUsageDescription 授权使应用只能在前台使用定位服务// 两者也可以都写if ([[UIDevice currentDevice].systemVersion floatValue] >=8.0 ) {// iOS0.0:如果当前的授权状态是使用是授权,那么App退到后台后,将不能获取用户位置,即使勾选后台模式:location[self.lcManager requestAlwaysAuthorization];[self.lcManager requestWhenInUseAuthorization];}// iOS9.0+ 要想继续获取位置,需要使用以下属性进行设置(注意勾选后台模式:location)但会出现蓝条if ([self.lcManager respondsToSelector:@selector(allowsBackgroundLocationUpdates)]) {//self.lcManager.allowsBackgroundLocationUpdates = YES;}[self.lcManager startUpdatingLocation]; // 开始更新位置[self.uploadTimer setFireDate:[NSDate distantPast]]; // 开启定时器}
}/** 获取到新的位置信息时调用*/
-(void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray *)locations {CLLocation *tempLocation = locations[0];// 将坐标转化为百度坐标 方法来源于百度sdkNSDictionary *temp = BMKConvertBaiduCoorFrom(tempLocation.coordinate, BMK_COORDTYPE_GPS);CLLocationCoordinate2D nowLocation = BMKCoorDictionaryDecode(temp);//if (self.isRequest) return;//self.isRequest = YES;//[self uploadUserLocationHandle:nowLocation];//[self uploadLocationTimer:nowLocation];_newCoor = nowLocation;
}
/** 不能获取位置信息时调用*/
-(void)locationManager:(CLLocationManager *)manager didFailWithError:(NSError *)error {NSLog(@"获取定位失败");
}/** 定位服务状态改变时调用*/
- (void)locationManager:(CLLocationManager *)manager didChangeAuthorizationStatus:(CLAuthorizationStatus)status
{switch (status) {case kCLAuthorizationStatusNotDetermined:{NSLog(@"用户还未决定授权");break;}case kCLAuthorizationStatusRestricted:{NSLog(@"访问受限");break;}case kCLAuthorizationStatusDenied:{// 类方法,判断是否开启定位服务if ([CLLocationManager locationServicesEnabled]) {NSLog(@"定位服务开启,被拒绝");} else {NSLog(@"定位服务关闭,不可用");}break;}case kCLAuthorizationStatusAuthorizedAlways:{NSLog(@"获得前后台授权");break;}case kCLAuthorizationStatusAuthorizedWhenInUse:{NSLog(@"获得前台授权");break;}default:break;}
}// 直接上传用户位置
static NSInteger uploadCount = 1;
- (void)uploadUserLocationHandle:(CLLocationCoordinate2D)coor {NSDictionary *dic = @{@"longitude":@(coor.longitude),@"latitude":@(coor.latitude)};__weak typeof(self)weakSelf = self;[[UserManager shareInstance] uploadUserLocation:dic block:^(BOOL success) {if (!success) {if (uploadCount > 3) return ;uploadCount ++;[weakSelf uploadUserLocationHandle:coor];XSDLog(@"上传位置不ok");return;}// if (uploadCount != 1) uploadCount = 1;XSDLog(@"上传位置ok");}];}// 定时上传位置
- (void)uploadLocationTimer {BOOL canUpload = [self isCanUpload:_newCoor];if (canUpload) {NSDictionary *dic = @{@"longitude":@(_newCoor.longitude),@"latitude":@(_newCoor.latitude)};__weak typeof(self)weakSelf = self;// 和后台服务器进行交互 上传位置[[UserManager shareInstance] uploadUserLocation:dic block:^(BOOL success) {if (success) {XSDLog(@"上传位置ok");return ;}XSDLog(@"上传位置不ok");[weakSelf uploadUserLocationHandle:_newCoor];}];}}// 监听用户登录的通知
- (void)p_addNSNotificationObserver {[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(p_loginNotification) name:kLoginNotification object:nil];
}- (void)p_loginNotification {// 用户登录, 就开始上传位置NSString *latitude = [XSDTools objectForKey:TCPFLocationlatitude];NSString *longitude = [XSDTools objectForKey:TCPFLocationlongitude];if (!latitude || !longitude) return;// 读取本地的经纬度CLLocationCoordinate2D location = CLLocationCoordinate2DMake(latitude.doubleValue, longitude.doubleValue);[self uploadUserLocationHandle:location];
}// 是否达到条件(判断距离 大于一定距离)上传位置
- (BOOL)isCanUpload:(CLLocationCoordinate2D)coor {//  下面代码相当于 NSUserDefaults 存取数据NSString *latitude = [XSDTools objectForKey:LAST_LATI];NSString *longitude = [XSDTools objectForKey:LAST_LONG];if (!latitude || !longitude) {//  下面代码相当于 NSUserDefaults 存取数据[XSDTools setValue:[NSString stringWithFormat:@"%f",coor.latitude] forKey:LAST_LATI];[XSDTools setValue:[NSString stringWithFormat:@"%f",coor.longitude] forKey:LAST_LONG];return YES;}//  下面代码来源于百度sdk 计算两点间的距离 NSNumber *distence = [XSDBaiduMapTools calculateTwoPointLongWithStart:CLLocationCoordinate2DMake(latitude.doubleValue, longitude.doubleValue) end:coor];if (distence.integerValue >= 100) {//  下面代码相当于 NSUserDefaults 存取数据[XSDTools setValue:[NSString stringWithFormat:@"%f",coor.latitude] forKey:LAST_LATI];[XSDTools setValue:[NSString stringWithFormat:@"%f",coor.longitude] forKey:LAST_LONG];return YES;}return NO;
}// 懒加载
- (NSTimer *)uploadTimer {if (!_uploadTimer) {_uploadTimer = [NSTimer scheduledTimerWithTimeInterval:10.0 target:self selector:@selector(uploadLocationTimer) userInfo:nil repeats:YES];}return _uploadTimer;
}@end
里面用到了一些自定义的类,不过不影响,各位可以根据需求修改,挺简单的;

在AppDelegate的- (BOOL)application:(UIApplication *)application willFinishLaunchingWithOptions:(NSDictionary *)launchOptions 方法里

直接调用即可:[[XSDLocationTools shareInstance] startLocationService];

还有,是必须在如图所示,勾选location updates;

图:


最后,审核的时候,一定要说明清楚,为啥要用这个后台上传位置功能;这个审核被拒概率很大,也没有好的解决方法;

这篇关于iOS开发 程序后台上传位置CLLocationManager的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/775309

相关文章

Python38个游戏开发库整理汇总

《Python38个游戏开发库整理汇总》文章介绍了多种Python游戏开发库,涵盖2D/3D游戏开发、多人游戏框架及视觉小说引擎,适合不同需求的开发者入门,强调跨平台支持与易用性,并鼓励读者交流反馈以... 目录PyGameCocos2dPySoyPyOgrepygletPanda3DBlenderFife

使用Python开发一个Ditto剪贴板数据导出工具

《使用Python开发一个Ditto剪贴板数据导出工具》在日常工作中,我们经常需要处理大量的剪贴板数据,下面将介绍如何使用Python的wxPython库开发一个图形化工具,实现从Ditto数据库中读... 目录前言运行结果项目需求分析技术选型核心功能实现1. Ditto数据库结构分析2. 数据库自动定位3

Django开发时如何避免频繁发送短信验证码(python图文代码)

《Django开发时如何避免频繁发送短信验证码(python图文代码)》Django开发时,为防止频繁发送验证码,后端需用Redis限制请求频率,结合管道技术提升效率,通过生产者消费者模式解耦业务逻辑... 目录避免频繁发送 验证码1. www.chinasem.cn避免频繁发送 验证码逻辑分析2. 避免频繁

Spring Boot集成/输出/日志级别控制/持久化开发实践

《SpringBoot集成/输出/日志级别控制/持久化开发实践》SpringBoot默认集成Logback,支持灵活日志级别配置(INFO/DEBUG等),输出包含时间戳、级别、类名等信息,并可通过... 目录一、日志概述1.1、Spring Boot日志简介1.2、日志框架与默认配置1.3、日志的核心作用

PyQt5 GUI 开发的基础知识

《PyQt5GUI开发的基础知识》Qt是一个跨平台的C++图形用户界面开发框架,支持GUI和非GUI程序开发,本文介绍了使用PyQt5进行界面开发的基础知识,包括创建简单窗口、常用控件、窗口属性设... 目录简介第一个PyQt程序最常用的三个功能模块控件QPushButton(按钮)控件QLable(纯文本

基于Python开发一个图像水印批量添加工具

《基于Python开发一个图像水印批量添加工具》在当今数字化内容爆炸式增长的时代,图像版权保护已成为创作者和企业的核心需求,本方案将详细介绍一个基于PythonPIL库的工业级图像水印解决方案,有需要... 目录一、系统架构设计1.1 整体处理流程1.2 类结构设计(扩展版本)二、核心算法深入解析2.1 自

Spring Boot 结合 WxJava 实现文章上传微信公众号草稿箱与群发

《SpringBoot结合WxJava实现文章上传微信公众号草稿箱与群发》本文将详细介绍如何使用SpringBoot框架结合WxJava开发工具包,实现文章上传到微信公众号草稿箱以及群发功能,... 目录一、项目环境准备1.1 开发环境1.2 微信公众号准备二、Spring Boot 项目搭建2.1 创建

golang程序打包成脚本部署到Linux系统方式

《golang程序打包成脚本部署到Linux系统方式》Golang程序通过本地编译(设置GOOS为linux生成无后缀二进制文件),上传至Linux服务器后赋权执行,使用nohup命令实现后台运行,完... 目录本地编译golang程序上传Golang二进制文件到linux服务器总结本地编译Golang程序

使用Docker构建Python Flask程序的详细教程

《使用Docker构建PythonFlask程序的详细教程》在当今的软件开发领域,容器化技术正变得越来越流行,而Docker无疑是其中的佼佼者,本文我们就来聊聊如何使用Docker构建一个简单的Py... 目录引言一、准备工作二、创建 Flask 应用程序三、创建 dockerfile四、构建 Docker

springboot项目打jar制作成镜像并指定配置文件位置方式

《springboot项目打jar制作成镜像并指定配置文件位置方式》:本文主要介绍springboot项目打jar制作成镜像并指定配置文件位置方式,具有很好的参考价值,希望对大家有所帮助,如有错误... 目录一、上传jar到服务器二、编写dockerfile三、新建对应配置文件所存放的数据卷目录四、将配置文