这些天,折腾ios的那些事

来源:互联网 发布:博物馆设计软件 编辑:程序博客网 时间:2024/06/09 15:26

序言

从清明假回来,部门老板开始给我布置实习的第二个任务,叫我按照一个很蛋疼的登陆流程做一个iPad应用的登陆模块,也许说起登陆模块大家都会,就是验证用户名和密码的过程,但是实际的产品的登陆流程却远比这个复杂,这里不对登陆过程具体介绍了,主要说说折腾ios具体开发过程的事情,作为一个初入行的码农,以后一定要记住这些教训。

(1)工作任务很简单,先看懂PC版的登陆流程,然后照葫芦画瓢的将其流程在ipad上运行起来。pc版的是一个同事写的,用的C#,vc08版和10版的版本转化就折腾了半天,不过前两周一直再整C#,所以代码看起来还比较快,花了一天的事情对蛋疼的流程终于有了一个清晰的了解,不过后来发现还是有个地方没有理解老板的意图,还好补救起来不是很麻烦。题注:以后接任务的时候一定要弄清楚老板的最终目的,一定要拿个小本记清楚,长期堆代码的人脑子不够用,和同事的交流一定要不嫌麻烦,交流的过程实际上就是交流感情的过程,以后的圈子就是这样建立的。

(2)ipad的开发当然要用到mac,xcode,还好机子环境都已经搭建好了,ipad版本的已有代码拷贝下来,开始整啦。第一个问题,就是ios开发的模拟器调试和真机联机调试(device)的区别,听同事说的,模拟器的运行处理过程其实是用的机器的intel处理器,而ipad上是用的arm处理器,部分代码在两个调试平台会有些区别,具体什么区别现在我还没有接触到。而真机调试需要去苹果官方申请开发者权限,下载ios开发的私钥和许可证才行。私钥其实就相当于现在各大开放平台的appID和secretKey,而许可证就相当于官方注册你要调试机器的固件编号,对其分发一个和私钥配对的解密字符串。 搞了半天下载的私钥都出不来,最后还是直接从同事的另一台mac上把私钥拷贝出来,环境这才算搭好。题注:虽然很多问题要自己想办法解决,但是同事以前做过的话,过去喊一声,总比你在那里研究半天的强;在给你解决问题的过程中还能对你言传身教的,可以让你学到不少东西,至少可以少走弯路,更快的完成老板交代的任务。

(3)总算可以闲下来写代码了,ipad和iphone的开发没有什么区别,虽然以前自己也折腾过几个小实例,看过那个所有初学者都会看的objective-c基础教程,但真正自己写代码还真没有怎么接触过。所以就当边学边练了。

技术问题

讨论一些工作工程中花比较多时间解决的问题:

(1)iOS内存分配和释放的问题

开始在window的appDaleget中写了一些UIAlertview,调试代码的时候执行到Alertview show的时候在主线程里就提示SIGBART错误,一开始以为是Daleget中不能显示AlertVIew,以前看书是说最好不要在dale get中去显示界面的东西,就有这样的误会。 不过在同事确认可以后,将问题确认为代码漏洞。于是将Alertview 置于函数入口,直接return,发现在Daleget是可以显示的,于是将alertview 一步步的后移,终于确定了出问题的函数,进入函数之后,在一部分一部分的代码注释,最后终于确定了问题代码行:

 NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];[defaults release];

问题就出在release 上,defaults的初始化并没有调用init分配内存。刚写代码,所以一直谨记ios开发的init release原则,所以每遇到一个*对象的时候就去release它,结果这样是不对的。后面又遇到这样的内存释放问题,由于谨记上一次教训,开始以为alloc就要release,其实也不是这样的,只有init 你才需release;以后遇到这样的问题,宁愿少release,也不要多release。因为有些对象你不release是不会出问题的。


(2)iOS联机调试的文件拷贝问题

iOS联机调试时需要将程序中的资源文件先拷贝到调试机器里,我一开始不知道,将文件放到xcode的机器上,检查文件是否存在时发现路径是机器的路径。这下就滑稽了,不知道怎么拷贝一个文件到ipad的应用目录下去,mac下也查不到ipad的磁盘,最后询问同事才知道拷贝资源文件到程序是在程序里做的,给我的版本是只要目录存在就不拷贝,我的个去呀,原来时这样的。所以只能在机器上先把原有的应用删掉,然后command+R,拷贝文件,OK!


(3) iOS判断是否有网络的程序

在判断网络是否存在的情况下,我开始的做法时send一个登录百度的http请求,如果请求的返回码是200-300之间,我就判定有网,而pc版的做法也是开一个ping百度的线程,看是否能够ping成功,可是ipad上没法用ping呀。所以只有另寻办法,在网上找了找,发现下面这段代码能用,嘿,贴下来,供以后使用。

//check if net OK-(BOOL) checkNet {    bool ans = NO;        // Create zero addy    struct sockaddr_in zeroAddress;    bzero(&zeroAddress, sizeof(zeroAddress));    zeroAddress.sin_len = sizeof(zeroAddress);    zeroAddress.sin_family = AF_INET;        // Recover reachability flags    SCNetworkReachabilityRef defaultRouteReachability = SCNetworkReachabilityCreateWithAddress(NULL, (struct sockaddr *)&zeroAddress);    SCNetworkReachabilityFlags flags;        BOOL didRetrieveFlags = SCNetworkReachabilityGetFlags(defaultRouteReachability, &flags);    CFRelease(defaultRouteReachability);        if (!didRetrieveFlags)    {        NSLog(@"Error. Could not recover network reachability flags");        ans = NO;    }        BOOL isReachable = flags & kSCNetworkFlagsReachable;    BOOL needsConnection = flags & kSCNetworkFlagsConnectionRequired;    ans = (isReachable && !needsConnection) ? YES : NO;        return ans;}

(4) iOS xml解析问题

这个问题困扰了我很久,ios有一个解析XML的标准库,叫NSXMLParser,网上还说,解析XML的方法很多,各种各样的办法很多,我不想包含第三方的库,所以我还是选择自带的NSXMLParser。这个库是用了xml 代理的方式,具体解释一下这个代理,这个代理就是自己新建一个xml解析的类,在这个类里创建一个NSXMLParser的对象,然后将代理设为这个新建的xml类。另外新建的xml解析类必须继承NSXMLParserDelegate协议,然后在新建的xml类中自己实现这些协议的方法。具体看代码吧。

首先自己定义存放解析数据的数据结构:

//存储用户名和密码的数据结构@interface  UserPwd: NSObject {    NSString *_username;    NSString *_passwd;}@property (nonatomic, retain) NSString *username;@property (nonatomic, retain) NSString *passwd;@end@implementation UserPwd@synthesize username = _username;@synthesize passwd = _passwd;-(void) dealloc{    [_username release];    [_passwd release];}@end

然后实现自己创建的xml解析类:

@implementation XMLParser- (void) parseXMLFile:(NSString *)xmlFilename{    NSData *data = [[NSData alloc] initWithContentsOfFile:xmlFilename];    NSXMLParser *parser = [[NSXMLParser alloc] initWithData:data]; //    NSXMLParser *parser = [[NSXMLParser alloc] initWithContentsOfURL:[[NSURL alloc] initWithString:xmlFilename]];    [parser setShouldProcessNamespaces:NO];        [parser setShouldReportNamespacePrefixes:NO];    [parser setShouldResolveExternalEntities:NO];    [parser setDelegate:self];    [parser parse];} -(void)parser:(NSXMLParser *)parser didStartElement:(NSString *) elementName namespaceURI:(NSString *) namespaceURI qualifiedName:(NSString *) qualifiedName attributes:(NSDictionary *)attributeDict{     NSLog(@"111begin");     if([elementName isEqualToString:@"user"]) {         NSLog(@"User");         s_UserPwd = [[UserPwd alloc] init];        //读每个elemet的属性值 
[s_UserPwd setUsername:[attributeDict objectForKey:@"username"]];         [s_UserPwd setPasswd:[attributeDict objectForKey:@"password"]];                  NSLog(@"%@", [s_UserPwd username]);     }     }}
//解析每个element的value- (void)parser:(NSXMLParser *)parser foundCharacters:(NSString *)string {    NSLog(@"%@", string);}//element结束时的动作- (void)parser:(NSXMLParser *)parser didEndElement:(NSString *)elementName  namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName {    NSLog(@"111end");}//报告解析的结束-(void)parserDidEndDocument:(NSXMLParser*)parser{    }//报告不可恢复的解析错误-(void)paser:parserErrorOccured{    }@end

(5)iOS UIWebview获取屏幕touch按钮事件问题的解决

这个问题给困扰了我一天,最后投机用了一个方法才搞定,问题是这样的,在一个UIWebview中加载了一个登录页,页面中有一个登录按钮(html元素),问题就是我如何在viewcontroller中获取这个页面元素的的点击事件,网上说ios里的webview中无法获取touch事件,这可怎么半?

查资料发现,IOS可以和javascript交互,你在ios中可以获取页面中的标题、元素、提交表单,插入js,但是js中元素的onclick事件中却无法插入ios的代码。而ios的UIWebview中只能截获url跳转事件,开始的登录按钮没有onclick事件,是为了安全性的考虑,参考同事安桌板的做法,给登录按钮设置了一个onclick事件,让其调转到一个有指定标记(登录标记)的url,然后再在UIwebview中截获
这个事件,就OK了。

- (BOOL)webView:(UIWebView *)webView shouldStartLoadWithRequest:(NSURLRequest *)request navigationType:(UIWebViewNavigationType)navigationType{    //获取当前调转页面的URL    NSString *_loginURL = [[request URL] absoluteString];    //如果当前URL有当前登录标记,则进行登录验证    if([[_loginURL lowercaseString] rangeOfString:@"login标记"].length > 0)    {        userName = [webView stringByEvaluatingJavaScriptFromString:@"document.getElementsByName('loginId')[0].value"];        passWd = [webView stringByEvaluatingJavaScriptFromString:@"document.getElementsByName('passwd')[0].value"];        [self iPADLogin];    }     return YES;}