文档章节

程序员转型指南 当Java遇见了Objective-C(1)

Ryon_king
 Ryon_king
发布于 2014/08/22 12:05
字数 4340
阅读 32
收藏 0

目前在移动开发领域最重要的两个平台分别为Android平台和iOS,在两个平台开发应用分别要用Java和Objective-C语言。虽然Java和Objective-C就像是处在两个不同的世界,但这两种编程语言以及它们的平台库等等还是有许多相似的地方。本文为51CTO独家译文,讲述了外国开发者Genadiy Shteyman从Java开发转向Objective-C需要掌握技能。

以下为全部译文,(文章中的“我”指代"Genadiy Shteyman")

最近一段时间,我从编写企业Java应用转向使用Objective-C。经过长时间的困扰之后,我发现两者的相似之处很多,如果能够早些读到相关的文章,转换工作会容易得多。

所以我写下这篇文章,想要帮助Java程序员快速的掌握Objective-C开发的主要特点。我使用一个社交网络应用作为例子,演示怎样用这两种语言建立开发环境。例子中会包括创建基本对象与比较两种语言的MVC设计模式,还会演示两种语言中数据的存储和获取。

Objective-C开发:从哪里开始

开发iPhone应用,首先最好要使用Mac电脑。最新的Mac OS X 10.6版本通常包含了一份Xcode IDE,以及使用Objective-C的配套iPhone开发软件工具套装(图表一)。

Xcode IDE开发环境,项目视图
图表一:Xcode IDE开发环境,项目视图

2010年11月,苹果发布了期待已久的iOS SDK 4.2,其中包含了丰富的框架和功能,用来搭建互动iPhone应用。Xcode还包含了一个仿真器,可以让你在电脑中模拟程序运行在手机上的效果。

Objective-C是iPhone应用的主要开发语言。对Java开发者来说,幸运的是Objective-C是完全面向对象的,使用和其他OO语言相同的理念——继承、多态和封装等等。定义一个类(Objective-C中称为module或.m文件),首先要定义一个接口(一个header或.h文件),然后把它引入到类中。

我们来看这个社交网络应用的例子,这个应用需要建立一个联系册,让你和朋友们时常保持联系。朋友的档案存储在FriendProfile对象中,包含四个字段:朋友的名字、城市、国家和电话号码,如Listing One所示:

Listing One  //  FriendProfile.h  #import <Foundation/Foundation.h>  #import <UIKit/UIKit.h>   @interface FriendProfile : NSObject {  }  @property (nonatomic, retain) NSString * name;  @property (nonatomic, retain) NSString * country;  @property (nonatomic, retain) NSString * city;  @property (nonatomic, retain) NSString * phoneNbr;  @end  //FriendProfile.m  #import "FriendProfile.h"   @implementation FriendProfile  @synthesize name;   @synthesize country;   @synthesize city;   @synthesize phoneNbr;  @end

在这个例子中,接口FriendProfile:NSObject表示我们定义了一个叫做FriendProfile的接口,它从NSObject基类中继承各种功能。NSObject是Objective-C的根类,大多数Objective-C中用到的类都会从中继承,这和Java中的Object类相似。接下来,我们分配多个NSString类型变量(等同于Java中的String类型)用来存储朋友的数据。然后是建立FriendProfile类,使用@synthesize关键字自动创建各种get和set方法。建立一个FriendProfile对象可以使用如下的语句:

FriendProfile * profile = [[FriendProfile alloc] init];

这里的alloc和init就像Java里的new关键字,用来在内存中建立FriendProfile对象。接下来,就可以给对象的各种字段赋值了。

[profile setName:@"Albert"];  [profile setCountry:@"USA"];  [profile setCity:@"Houston"];  [profile setPhoneNbr:@"123-456-789"];

或者可以更简单一点:

profile.name = @"Albert";  profile.country = @"USA";  profile.city = @"Houston;  profile.phoneNbr = @"123-456-789";

想要充分了解Objective-C的语法和功能可以去苹果的开发者站点,那里的语言参考编写的非常好。

Java的构造

在Java中,如果我们想写一个FriendProfile类,所做的和Objective-C会非常相像,就像Listing Two所示:

Listing Two  package com.vo;   public class FriendProfile {     private String name;     private String country;     private String city;     private String phoneNbr;      public String getName() {        return name;     }      public void setName(String name) {       this.name = name;     }      public String getCountry() {        return country;     }      public void setCountry(String country) {        this.country = country;     }      public String getCity() {        return city;     }      public void setCity(String city) {        this.city = city;     }      public String getPhoneNbr() {        return phoneNbr;     }      public void setPhoneNbr(String phoneNbr) {        this.phoneNbr = phoneNbr;     }  }

Listing Two中提供了相似的字段,但是那些get和set必须清楚的写出来。现在我们看看怎样在通讯录里添加一个新朋友,参加Listing Three:

Listing Three   public class FriendlyServletController extends HttpServlet {       private static final long serialVersionUID = 1L;       /**       * @see HttpServlet#doGet(HttpServletRequest request,        * HttpServletResponse response)       */     protected void doGet(HttpServletRequest request,                            HttpServletResponse response)                 throws ServletException, IOException {          doPost(request, response);      }       /**       * @see HttpServlet#doPost(HttpServletRequest request,        * HttpServletResponse response)       */     protected void doPost(HttpServletRequest request,                             HttpServletResponse response)                 throws ServletException, IOException {          response.setContentType("text/html");          PrintWriter out = response.getWriter();                final String action =                   request.getParameter("requestedAction");            if (action==null || action.trim().length()==0){              out.println("invalid action requested");              return;          }          else           if (action.equalsIgnoreCase("addToContacts")){                                String name = request.getParameter("name");              String country = request.getParameter("country");              String city = request.getParameter("city");              String phoneNbr = request.getParameter("phoneNbr");                       //normally you have to validate browser-originated requests              boolean validParameters =                       validateParameters(name, country, city, phoneNbr);              if (validParameters==false){                  out.println(                     "please verify and submit correct information");                  return;              }                            FriendProfile newProfile = new FriendProfile();              newProfile.setName(name);              newProfile.setCountry(country);              newProfile.setCity(city);              newProfile.setPhoneNbr(phoneNbr);               ProfileManager.getInstance().addToContacts(newProfile);              out.println("Your friend is added to contacts");              return;          }          else{              out.println("invalid action requested");              return;          }             }  }

在这个例子里,FriendlyServletController类从HTTPServlet中获取行为,HTTPServlet是Java的客户端组件类,负责处理浏览器的请求。当用户登入网站并且决定添加一个朋友时,他会在HTML表单的字段中填入数据,表单提交时,Servlet收到并验证请求的参数,并创建一个FriendProfile对象,在内存中存储数据。而ProfileManager类会把你的FriendProfile对象存储到数据库中。

Objective-C的MVC模式

在Java Web应用中常采用Model-View-Controller(MVC)设计模式,iPhone开发中也是如此。如果你在iOS Reference Library中查找UIViewController类的定义,你会发现这样的话:“UIViewController类为iPhone应用提供最基本的视图管理模型……你可以使用UIViewController实例来管理视图结构。”UIViewController实际上是一个控制器组件,用来触发业务逻辑,更新客户端的视图。

Model-View-Controller(MVC)设计模式
图表2:Model-View-Controller(MVC)设计模式

如果你想在Xcode中创建一个UIViewController类型的对象,可以选择通过XIB文件来创建。这种特殊的Xcode文件定义了图形用户界面或者说视图,包含了各种不同的控件,比如按钮、图表和标签等等。

回到我们的例子中来,假设你已经在联系列表中添加了几个朋友,现在想按下某个朋友的链接来看查看他的详细信息,这个功能可以通过定义控制器类来完成。代码请见Listing Four:

Listing Four  //  FriendProfileViewController.h   #import <UIKit/UIKit.h>   @class FriendProfile;  @class DatabaseController;  @class MFriendProfile;   // define our custom controller to inherit from  // the UIViewController class  @interface FriendProfileViewController : UIViewController {      FriendProfile * profile;      MFriendProfile * mprofile;      DatabaseController *dbController;  }  @property(nonatomic, retain) IBOutlet UILabel *lname;  @property(nonatomic, retain) IBOutlet UILabel *lcountry;  @property(nonatomic, retain) IBOutlet UILabel *lcity;  @property(nonatomic, retain) IBOutlet UILabel *lphoneNbr;   -(IBAction)buttonPressed:(id)sender;  @end   #import "FriendProfileViewController.h"  #import "FriendProfile.h"  #import "DatabaseController.h"  #import "MFriendProfile.h"  @implementation FriendProfileViewController  ...   // Implement viewDidLoad to do additional setup after   // loading the view, typically from a nib.  - (void)viewDidLoad {      [super viewDidLoad];           //create sample profile      profile = [[FriendProfile alloc] init];      profile.name = @"Albert";       profile.country = @"USA";       profile.city = @"Houston";       profile.phoneNbr = @"123-456-789";       //show profile on a screen      lname.text = profile.name;      lcountry.text = profile.country;      lcity.text = profile.city;      lphoneNbr.text = profile.phoneNbr;     }   //call the model to bring friend information from database  -(IBAction)buttonPressed:(id)sender{      NSLog(@"fetching friend profile by name.");      // name is hardcoded for demo purposes.   // Usually entered by user.  mprofile = (MFriendProfile*)         [dbController getFriendProfileObjectbyName:@"Albert"];        lname.text = mprofile.name;      lcountry.text = mprofile.country;      lcity.text = mprofile.city;      lphoneNbr.text = mprofile.phoneNbr;  }

这段代码中,我们创建了一个FriendProfileViewController实例,在我们定义的View Bundle中进行初始化,显示出朋友的各种信息。

Alloc和initWithNibName都是控制器类创建实例时使用的方法,和Java的new关键字功能一样。

模型在装载视图时开始启动。每个控制器都有一些从父类UIViewController继承而来的生命周期方法。比如ViewdidLoad方法就是其中之一,它负责在视图装载之后的额外设置,从数据库中取出信息,更新视图。在最简单的情况下,我们的视图包含一系列标签,或者是UILabel类型的对象,可以在应用运行时设置各种文本,用户可以立即看见朋友信息被更新了。

Java的MVC模式

下面来看看如何使用Java后台在浏览器窗口中显示出朋友的详细信息。我们稍微修改一下FriendlyServletController即可,代码请见Listing Five:

Listing Five  import java.io.IOException;  import java.io.PrintWriter;   import javax.servlet.ServletException;  import javax.servlet.http.HttpServlet;  import javax.servlet.http.HttpServletRequest;  import javax.servlet.http.HttpServletResponse;   import com.model.ProfileManager;  import com.vo.FriendProfile;   /**   * Servlet implementation class FriendlyServletController   */ public class FriendlyServletController extends HttpServlet {      private static final long serialVersionUID = 1L;      /**      * @see HttpServlet#doGet(HttpServletRequest request,       * HttpServletResponse response)      */    protected void doGet(HttpServletRequest request,                           HttpServletResponse response)                throws ServletException, IOException {        doPost(request, response);     }      /**      * @see HttpServlet#doPost(HttpServletRequest request,       *      HttpServletResponse response)      */    protected void doPost(HttpServletRequest request,                            HttpServletResponse response)                throws ServletException, IOException {        response.setContentType("text/html");        PrintWriter out = response.getWriter();         final String action = request.getParameter("requestedAction");          if (action==null || action.trim().length()==0){           out.println("invalid action requested");           return;        }        else if(action.equalsIgnoreCase("showFriendProfile")){           String name = request.getParameter("name");           FriendProfile existProfile = new FriendProfile();           existProfile.setName(name);           existProfile =               ProfileManager.getInstance().lookupContact(existProfile);           if (existProfile==null){               out.println("profile was not found");           }else{               out.println("here is your contact information:" +                existProfile.getName() + " from " +                existProfile.getCity() + " in " +                existProfile.getCountry() + " at " +                existProfile.getPhoneNbr());           }           return;                    }        else if (action.equalsIgnoreCase("addToContacts")){           String name = request.getParameter("name");           String country = request.getParameter("country");           String city = request.getParameter("city");           String phoneNbr = request.getParameter("phoneNbr");            //normally you have to validate browser-originated requests           boolean validParameters =               validateParameters(name, country, city, phoneNbr);           if (validParameters==false){              out.println("please verify and submit correct information");              return;           }            FriendProfile newProfile = new FriendProfile();           newProfile.setName(name);           newProfile.setCountry(country);           newProfile.setCity(city);           newProfile.setPhoneNbr(phoneNbr);            ProfileManager.getInstance().addToContacts(newProfile);           out.println("Your friend is added to contacts");           return;        }        else{           out.println("invalid action requested");           return;        }     }      //basic parameter validation routine     private boolean validateParameters(String name, String country,                                         String city, String phoneNbr){        /basic validation to check if all parameters are sent        if (name==null || name.trim().length()==0 ||            country==null || country.trim().length()==0 ||            city ==null || city.trim().length()==0 ||            phoneNbr == null || phoneNbr.trim().length()==0){           return false;        }        return true;     }  }

在这个例子中,FriendlyServletController接收表单产生的HTTP请求,我们特别编写了一个事件叫做showFriendProfile。这里我们的模型是一个ProfileManager对象,负责通过朋友姓名在数据库中查找记录。然后查找到的数据库记录会以FriendProfile对象的形式返回到控制器,其中包含了各种详细信息,组成视图显示在浏览器窗口中。

Objective-C的数据库访问

较复杂的应用都会用到某类数据存储方式,通常是一个数据库。苹果推荐开发者使用称为Core Data的Cocoa API框架进行数据库存取操作。Core Data框架能够直接与SQLite数据库相结合(我们例子中的数据库运行在移动设备上)。Core Data隐藏了复杂的SQL操作,取而代之的是非常方便的NSManagedObject界面,你可以直接操作整个对象实例的各种字段,这些字段可以自动存入数据库。Core Data框架的另一个方便之处是在数据库中创建表(以及向表中添加关联与限制),这些都可以在Core Data的用户界面中完成。

Core Data stack结构
图表3:Core Data stack结构

现在回到我们的社交网络应用例子,看看怎么从数据库中取出朋友的信息。我们使用SQLite 和Core Data API,但首先我们要稍微修改一下FriendProfile类,代码请见Listing Six:

Listing Six  //FriendProfile.h interface file// MFriendProfile.h  #import <Foundation/Foundation.h>  #import <CoreData/CoreData.h>   @interface MFriendProfile : NSManagedObject {   }  @property (nonatomic, retain) NSString * name;  @property (nonatomic, retain) NSString * country;  @property (nonatomic, retain) NSString * city;  @property (nonatomic, retain) NSString * phoneNbr;   @end  // MFriendProfile.m  #import "MFriendProfile.h"  @implementation MFriendProfile  @dynamic name;  @dynamic country;  @dynamic city;  @dynamic phoneNbr;   @end

这里的FriendProfile类与Listing One中的不同之处在于在这里我加入了Core Data框架的头文件。而且在这里我们的类是从NSManagedObject中扩展出来,带有了Core Data对象需要的全部基本行为。Core Data的NSManagedObject类中使用到的Accessor则在应用运行时动态创建。如果你想在FriendProfile类中声明或使用属性,但不想在编译时出现缺少方法的警告,可以使用@dynamic指令,而不是@synthesize指令。

使用NSManagedObject API有些复杂,但你理解之后就会变得很好用。Listing Seven是一个示例方法,从数据库的FRIENDPROFILE表中取得朋友的信息。表包含四列:NAME、COUNTRY、CITY和PHONE­NBR。

Listing Seven  // DatabaseController.m  #import "DatabaseController.h" #import <sqlite3.h>  #define kDatabaseName @"SocialNetworking.sqlite"  ...   - (NSManagedObject *)getFriendProfileObjectbyName:(NSString *)name {      managedObjectContext = [self managedObjectContext];        //create sort descriptors to specify preferred sort order    NSSortDescriptor *sortDescriptor =         [[NSSortDescriptor alloc] initWithKey:@"name" ascending:YES];      NSArray *sortDescriptors =         [[NSArray alloc]  initWithObjects:sortDescriptor,nil];      //specify where clause      NSPredicate *predicate =         [NSPredicate predicateWithFormat:@"name == %d", name];           //fetch our friendís profile from database table       NSEntityDescription *entity =         [NSEntityDescription entityForName:@"MFRIENDPROFILE"           inManagedObjectContext:managedObjectContext];          NSFetchRequest *request = [[NSFetchRequest alloc] init];      [request setEntity:entity];       // Set the predicate for the current view     if (predicate)       {          [request setPredicate:predicate];      }       if (sortDescriptors)       {          [request setSortDescriptors:sortDescriptors];      }      NSError *error = nil;     NSMutableArray *results = [[managedObjectContext         executeFetchRequest:request error:&error] mutableCopy];         if (error)     {        NSLog(@"error in getFriendProfileObjectbyName:%@",         [error userInfo]);     }      [sortDescriptor release];      [sortDescriptors release];      [predicate release];      if ([results count] > 0) {         return [results objectAtIndex:0];     }     return nil;  }

getFriendProfileObjectbyName方法把朋友的姓名作为一个参数接收过来。通过使用Core Data API,我们可以指定在哪一个表中进行查询和排序,并且在后台执行SQL语句。

SQL>select * from FriendProfile where name = "Albert";

Core Data API有许多种没有封装的“半成品”代码,可以访问NSManagedObjectContext、NSPersistentStoreCoordinator和NSManaged­ObjectModel对象。你可以复制这些代码,只要你取得了FriendProfile对象,就能以下面的形式取得它的属性:

NSString* name =        FriendProfile.name;  NSString* country = FriendProfile.country;  NSString* city =        FriendProfile.city;  NSString* phoneNbr =    FriendProfile.phoneNbr;

总的来说,Core Data是一个非常有用的功能,可以让你通过图表来定义数据表和管理,可以动态生成相应的对象,而且无需使用复杂的SQL语句。但不好的方面是这里有大量的没有经过封装的代码,这样你在使用它们与测试时需要非常小心。

Java:数据库存取

Java有许多数据库框架。在我看来,Hibernate是和Core Data API最相像的Java框架。Hibernate使用的是对象关系映射(Object-Relational Mapping,ORM)机制,这样你可以通过简单的在对象中设置字段并且直接映射成数据库中的表来把对象数据放入关系型数据库中。映射可以通过XML文件,也可以通过Java 5中的metadata annotation方法获得。Listing Eight是使用XML进行映射的一个例子。

Listing Eight

此例中,Listing Two中的FriendProfile对象被映射到数据库中的一个同名表,这是一种传统的数据映射做法。对象的四个字段被直接映射到表中的四列,通过映射,Hibernate可以使用SQL语句来完成各种操作。

另一个配置文件叫做hibernate.cfg.xml,包含了数据库连接设置的详细信息,包括数据库URL、数据库驱动以及用户名和密码等,代码请见Listing Nine:

Listing Eight  <hibernate-mapping>   <class name="com.vo.FriendProfile" table=" FRIENPROFILE ">   <property name="name">      <column name="NAME" />   </property>    <property name="country">     <column name="COUNTRY"/>   </property>  <property name="city">     <column name="CITY"/>   </property>   <property name="phoneNbr">     <column name="PHONENBR"/>   </property>  </class> </hibernate-mapping>

Listing Nine中我们导入了所有需要的Hibernate库,创建了一个Hibernate Session并且开始事务,接下来我们仅简单使用了Session对象的get方法就轻松检索到了FriendProfile对象,传递回所需要的对象类型并过滤出查询的字段——朋友的姓名。

结论

除去语法结构与运行平台的不同,使用Objective-C进行iPhone开发与使用Java进行网络应用开发在下面几个方面是相同的:

◆两种语言都是面向对象的

◆两种语言使用同样的设计模式,例如MVC

◆两种语言使用相似的数据库存储技术,例如ORM

然而,对于Java开发者,使用Objective-C时在有些地方要格外小心:

◆创建对象:Java对象是在运行时通过new关键字创建的。因此Java程序员无需担心内存分配问题。而在Objective-C中,一个对象可以由三个关键字创建,alloc、new或者copy,这三个关键字在创建对象时都会增加对象的持有计数(retain count),持有计数是Objective-C特有的内存管理方法,显示有多少个指针指向对象,是否可以被内存管理器回收。

◆销毁对象:由于强大的垃圾回收机制,Java的内存管理工作极度简单。Java的引用对象都存储在JVM的堆内存中,一旦不再被引用,就可以作为垃圾回收。Objective-C使用的是内存管理器,而不是垃圾回收器。如果你使用上面说的三种方法在内存中创建了一个对象,那么必须使用release方法来释放对象。release方法会减少持有计数,当计数降到0时,被引用的对象会接受一个来自高级类的dealloc方法,释放它占用的内存并重新分配。如果忘记了释放内存或释放失败,那么会造成内存泄露和不可预见的错误。

◆过多释放和过早重新分配内存:由于垃圾回收机制,Java程序员可以完全不考虑这些问题。但Objective-C程序员需要小心,不能释放出比分配的更多的内存。如果在已经重新分配的对象上过多释放内存,就会造成应用的崩溃。

上面这些例子说明了Objective-C和Java在语法和语言元素上有很多相同之处。更重要的是,它们解决问题的思路和用到的组件也是非常相似的。如果你是Java程序员,相信你在看完这篇文章后,转向Objective-C的道路会更加通顺


本文转载自:http://mobile.51cto.com/iphone-262962_all.htm

Ryon_king
粉丝 1
博文 4
码字总数 2767
作品 0
郑州
程序员
私信 提问
2014年值得学习的编程语言书

经过数据分析和研究 Jobs Tractor 的 45000 个开发人员招聘职位数据,我们得到了上图的结果: 自上一年,主要的变化如下: PHP和Java换了位置,但是仍旧是高居不下 Java的Android已经取代了S...

modernizr
2014/05/22
15.7K
16
如果有人让你推荐编程技术书,请叫他看这个列表

计算机系统与网络 《图灵的秘密:他的生平、思想及论文解读》 《计算机系统概论》 《深入理解Linux内核》 《深入Linux内核架构》 《TCP/IP详解 卷1:协议》 《Linux系统编程(第2版)》 《Lin...

retref
2017/02/28
8.7K
47
IOS学习笔记——Objective-c基础(一)

最近自学ios,ios的支持语言是Objective-c,所以要学习ios就需要先学习Objective-c语言。当掌握了Objective-c的基础知识之后,我们就可以入手学习iOS开发做出一些自己的应用。 我现在还是...

丛林迷雾
2012/12/30
1K
1
Java反转字符串的10种方法(代码段)

本文由ImportNew -唐尤华 翻译自dzone。欢迎加入翻译小组。转载请见文末要求。 在这篇文章中,我们会讨论10种用Java反转字符串的方法,通过10个Java程序反转字符串。例如,把字符串“javagui...

ImportNew
2018/11/21
0
0
Objective-C类和Java类

1.类的定义 Objective-C类 @interface Circle:NSObject{int fillColor;int bounds;} (void)setFillColor:(int)fillColor; (void)setBounds:(int)bounds; (void)draw; @end// Circle 类接口部......

malawo
2012/11/27
306
0

没有更多内容

加载失败,请刷新页面

加载更多

重新开始学Java——反射

概念 reflection:自省 反射:镜子可以反射阳光一个java类 或 对象 通过照"镜子"来认知自己 Java语言中是怎么实现照镜子? java.lang.reflect 包 提供了"照镜子"API(应用程序接口) 如果要...

大家都是低调来的
13分钟前
5
0
爬取720万条城市历史天气数据

内容爬虫完毕,校验完毕,缺失信息暂未统计。总数据720万,地区3200个,年份从2011-2019,大小950Mb,原始数据已丢失,需要的朋友可以自己运行脚本挂一晚上。中间遇到了很多坑,有机会我再写...

八音弦
16分钟前
10
0
python的字典类型

1、新建字典 通过键值对 dict_1 = {'a':1,'b':2,'c':3} 通过dict()函数 list_1 = ['adam', 'bob', 'cathy', 'david', 'emma'] list_2 = [1,2,3,4,5] dict_2 = dict(zip(list_1,list_2)) 2、字......

davidwbnu
18分钟前
2
0
springcloud vue.js 前后分离 activiti工作流

本商品为 :springcloud + Springboot 微服务\分布式 工作流 前后分离 + 跨域 版本 (权限控制到菜单和按钮) 后台框架 :springcloud Greenwich.SR1 + springboot 2.1.4 + activiti6.0.0 + ...

java框架开发者
24分钟前
8
0
【jQuery基础学习】07 jQuery表单插件-Form

本文转载于:专业的前端网站➦【jQuery基础学习】07 jQuery表单插件-Form 作用:jQuery Form插件的作用是为了让我们可以很方便地用ajax的方式提交表单,从而使我们提交表单的时候页面不用进行...

前端老手
33分钟前
5
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部