程序員轉(zhuǎn)型指南 當(dāng)Java遇見(jiàn)了Objective-C
原創(chuàng)【51CTO譯文】目前在移動(dòng)開(kāi)發(fā)領(lǐng)域最重要的兩個(gè)平臺(tái)分別為Android平臺(tái)和iOS,在兩個(gè)平臺(tái)開(kāi)發(fā)應(yīng)用分別要用Java和Objective-C語(yǔ)言。雖然Java和Objective-C就像是處在兩個(gè)不同的世界,但這兩種編程語(yǔ)言以及它們的平臺(tái)庫(kù)等等還是有許多相似的地方。本文為51CTO獨(dú)家譯文,講述了外國(guó)開(kāi)發(fā)者Genadiy Shteyman從Java開(kāi)發(fā)轉(zhuǎn)向Objective-C需要掌握技能。
以下為全部譯文,(文章中的“我”指代"Genadiy Shteyman"):
最近一段時(shí)間,我從編寫(xiě)企業(yè)Java應(yīng)用轉(zhuǎn)向使用Objective-C。經(jīng)過(guò)長(zhǎng)時(shí)間的困擾之后,我發(fā)現(xiàn)兩者的相似之處很多,如果能夠早些讀到相關(guān)的文章,轉(zhuǎn)換工作會(huì)容易得多。
所以我寫(xiě)下這篇文章,想要幫助Java程序員快速的掌握Objective-C開(kāi)發(fā)的主要特點(diǎn)。我使用一個(gè)社交網(wǎng)絡(luò)應(yīng)用作為例子,演示怎樣用這兩種語(yǔ)言建立開(kāi)發(fā)環(huán)境。例子中會(huì)包括創(chuàng)建基本對(duì)象與比較兩種語(yǔ)言的MVC設(shè)計(jì)模式,還會(huì)演示兩種語(yǔ)言中數(shù)據(jù)的存儲(chǔ)和獲取。
Objective-C開(kāi)發(fā):從哪里開(kāi)始
開(kāi)發(fā)iPhone應(yīng)用,首先最好要使用Mac電腦。最新的Mac OS X 10.6版本通常包含了一份Xcode IDE,以及使用Objective-C的配套iPhone開(kāi)發(fā)軟件工具套裝(圖表一)。
圖表一:Xcode IDE開(kāi)發(fā)環(huán)境,項(xiàng)目視圖
2010年11月,蘋(píng)果發(fā)布了期待已久的iOS SDK 4.2,其中包含了豐富的框架和功能,用來(lái)搭建互動(dòng)iPhone應(yīng)用。Xcode還包含了一個(gè)仿真器,可以讓你在電腦中模擬程序運(yùn)行在手機(jī)上的效果。
Objective-C是iPhone應(yīng)用的主要開(kāi)發(fā)語(yǔ)言。對(duì)Java開(kāi)發(fā)者來(lái)說(shuō),幸運(yùn)的是Objective-C是完全面向?qū)ο蟮?,使用和其他OO語(yǔ)言相同的理念——繼承、多態(tài)和封裝等等。定義一個(gè)類(lèi)(Objective-C中稱為module或.m文件),首先要定義一個(gè)接口(一個(gè)header或.h文件),然后把它引入到類(lèi)中。
我們來(lái)看這個(gè)社交網(wǎng)絡(luò)應(yīng)用的例子,這個(gè)應(yīng)用需要建立一個(gè)聯(lián)系冊(cè),讓你和朋友們時(shí)常保持聯(lián)系。朋友的檔案存儲(chǔ)在FriendProfile對(duì)象中,包含四個(gè)字段:朋友的名字、城市、國(guó)家和電話號(hào)碼,如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
在這個(gè)例子中,接口FriendProfile:NSObject表示我們定義了一個(gè)叫做FriendProfile的接口,它從NSObject基類(lèi)中繼承各種功能。NSObject是Objective-C的根類(lèi),大多數(shù)Objective-C中用到的類(lèi)都會(huì)從中繼承,這和Java中的Object類(lèi)相似。接下來(lái),我們分配多個(gè)NSString類(lèi)型變量(等同于Java中的String類(lèi)型)用來(lái)存儲(chǔ)朋友的數(shù)據(jù)。然后是建立FriendProfile類(lèi),使用@synthesize關(guān)鍵字自動(dòng)創(chuàng)建各種get和set方法。建立一個(gè)FriendProfile對(duì)象可以使用如下的語(yǔ)句:
- FriendProfile * profile = [[FriendProfile alloc] init];
這里的alloc和init就像Java里的new關(guān)鍵字,用來(lái)在內(nèi)存中建立FriendProfile對(duì)象。接下來(lái),就可以給對(duì)象的各種字段賦值了。
- [profile setName:@"Albert"];
- [profile setCountry:@"USA"];
- [profile setCity:@"Houston"];
- [profile setPhoneNbr:@"123-456-789"];
或者可以更簡(jiǎn)單一點(diǎn):
- profile.name = @"Albert";
- profile.country = @"USA";
- profile.city = @"Houston;
- profile.phoneNbr = @"123-456-789";
想要充分了解Objective-C的語(yǔ)法和功能可以去蘋(píng)果的開(kāi)發(fā)者站點(diǎn),那里的語(yǔ)言參考編寫(xiě)的非常好。
Java的構(gòu)造
在Java中,如果我們想寫(xiě)一個(gè)FriendProfile類(lèi),所做的和Objective-C會(huì)非常相像,就像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必須清楚的寫(xiě)出來(lái)。現(xiàn)在我們看看怎樣在通訊錄里添加一個(gè)新朋友,參加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;
- }
- }
- }
在這個(gè)例子里,F(xiàn)riendlyServletController類(lèi)從HTTPServlet中獲取行為,HTTPServlet是Java的客戶端組件類(lèi),負(fù)責(zé)處理瀏覽器的請(qǐng)求。當(dāng)用戶登入網(wǎng)站并且決定添加一個(gè)朋友時(shí),他會(huì)在HTML表單的字段中填入數(shù)據(jù),表單提交時(shí),Servlet收到并驗(yàn)證請(qǐng)求的參數(shù),并創(chuàng)建一個(gè)FriendProfile對(duì)象,在內(nèi)存中存儲(chǔ)數(shù)據(jù)。而ProfileManager類(lèi)會(huì)把你的FriendProfile對(duì)象存儲(chǔ)到數(shù)據(jù)庫(kù)中。#p#
Objective-C的MVC模式
在Java Web應(yīng)用中常采用Model-View-Controller(MVC)設(shè)計(jì)模式,iPhone開(kāi)發(fā)中也是如此。如果你在iOS Reference Library中查找UIViewController類(lèi)的定義,你會(huì)發(fā)現(xiàn)這樣的話:“UIViewController類(lèi)為iPhone應(yīng)用提供最基本的視圖管理模型……你可以使用UIViewController實(shí)例來(lái)管理視圖結(jié)構(gòu)。”UIViewController實(shí)際上是一個(gè)控制器組件,用來(lái)觸發(fā)業(yè)務(wù)邏輯,更新客戶端的視圖。
圖表2:Model-View-Controller(MVC)設(shè)計(jì)模式
如果你想在Xcode中創(chuàng)建一個(gè)UIViewController類(lèi)型的對(duì)象,可以選擇通過(guò)XIB文件來(lái)創(chuàng)建。這種特殊的Xcode文件定義了圖形用戶界面或者說(shuō)視圖,包含了各種不同的控件,比如按鈕、圖表和標(biāo)簽等等。
回到我們的例子中來(lái),假設(shè)你已經(jīng)在聯(lián)系列表中添加了幾個(gè)朋友,現(xiàn)在想按下某個(gè)朋友的鏈接來(lái)看查看他的詳細(xì)信息,這個(gè)功能可以通過(guò)定義控制器類(lèi)來(lái)完成。代碼請(qǐng)見(jiàn)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;
- }
這段代碼中,我們創(chuàng)建了一個(gè)FriendProfileViewController實(shí)例,在我們定義的View Bundle中進(jìn)行初始化,顯示出朋友的各種信息。
Alloc和initWithNibName都是控制器類(lèi)創(chuàng)建實(shí)例時(shí)使用的方法,和Java的new關(guān)鍵字功能一樣。
模型在裝載視圖時(shí)開(kāi)始啟動(dòng)。每個(gè)控制器都有一些從父類(lèi)UIViewController繼承而來(lái)的生命周期方法。比如ViewdidLoad方法就是其中之一,它負(fù)責(zé)在視圖裝載之后的額外設(shè)置,從數(shù)據(jù)庫(kù)中取出信息,更新視圖。在最簡(jiǎn)單的情況下,我們的視圖包含一系列標(biāo)簽,或者是UILabel類(lèi)型的對(duì)象,可以在應(yīng)用運(yùn)行時(shí)設(shè)置各種文本,用戶可以立即看見(jiàn)朋友信息被更新了。
Java的MVC模式
下面來(lái)看看如何使用Java后臺(tái)在瀏覽器窗口中顯示出朋友的詳細(xì)信息。我們稍微修改一下FriendlyServletController即可,代碼請(qǐng)見(jiàn)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;
- }
- }
在這個(gè)例子中,F(xiàn)riendlyServletController接收表單產(chǎn)生的HTTP請(qǐng)求,我們特別編寫(xiě)了一個(gè)事件叫做showFriendProfile。這里我們的模型是一個(gè)ProfileManager對(duì)象,負(fù)責(zé)通過(guò)朋友姓名在數(shù)據(jù)庫(kù)中查找記錄。然后查找到的數(shù)據(jù)庫(kù)記錄會(huì)以FriendProfile對(duì)象的形式返回到控制器,其中包含了各種詳細(xì)信息,組成視圖顯示在瀏覽器窗口中。#p#
Objective-C的數(shù)據(jù)庫(kù)訪問(wèn)
較復(fù)雜的應(yīng)用都會(huì)用到某類(lèi)數(shù)據(jù)存儲(chǔ)方式,通常是一個(gè)數(shù)據(jù)庫(kù)。蘋(píng)果推薦開(kāi)發(fā)者使用稱為Core Data的Cocoa API框架進(jìn)行數(shù)據(jù)庫(kù)存取操作。Core Data框架能夠直接與SQLite數(shù)據(jù)庫(kù)相結(jié)合(我們例子中的數(shù)據(jù)庫(kù)運(yùn)行在移動(dòng)設(shè)備上)。Core Data隱藏了復(fù)雜的SQL操作,取而代之的是非常方便的NSManagedObject界面,你可以直接操作整個(gè)對(duì)象實(shí)例的各種字段,這些字段可以自動(dòng)存入數(shù)據(jù)庫(kù)。Core Data框架的另一個(gè)方便之處是在數(shù)據(jù)庫(kù)中創(chuàng)建表(以及向表中添加關(guān)聯(lián)與限制),這些都可以在Core Data的用戶界面中完成。
圖表3:Core Data stack結(jié)構(gòu)
現(xiàn)在回到我們的社交網(wǎng)絡(luò)應(yīng)用例子,看看怎么從數(shù)據(jù)庫(kù)中取出朋友的信息。我們使用SQLite 和Core Data API,但首先我們要稍微修改一下FriendProfile類(lèi),代碼請(qǐng)見(jiàn)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類(lèi)與Listing One中的不同之處在于在這里我加入了Core Data框架的頭文件。而且在這里我們的類(lèi)是從NSManagedObject中擴(kuò)展出來(lái),帶有了Core Data對(duì)象需要的全部基本行為。Core Data的NSManagedObject類(lèi)中使用到的Accessor則在應(yīng)用運(yùn)行時(shí)動(dòng)態(tài)創(chuàng)建。如果你想在FriendProfile類(lèi)中聲明或使用屬性,但不想在編譯時(shí)出現(xiàn)缺少方法的警告,可以使用@dynamic指令,而不是@synthesize指令。
使用NSManagedObject API有些復(fù)雜,但你理解之后就會(huì)變得很好用。Listing Seven是一個(gè)示例方法,從數(shù)據(jù)庫(kù)的FRIENDPROFILE表中取得朋友的信息。表包含四列:NAME、COUNTRY、CITY和PHONENBR。
- 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方法把朋友的姓名作為一個(gè)參數(shù)接收過(guò)來(lái)。通過(guò)使用Core Data API,我們可以指定在哪一個(gè)表中進(jìn)行查詢和排序,并且在后臺(tái)執(zhí)行SQL語(yǔ)句。
- SQL>select * from FriendProfile where name = "Albert";
Core Data API有許多種沒(méi)有封裝的“半成品”代碼,可以訪問(wèn)NSManagedObjectContext、NSPersistentStoreCoordinator和NSManagedObjectModel對(duì)象。你可以復(fù)制這些代碼,只要你取得了FriendProfile對(duì)象,就能以下面的形式取得它的屬性:
- NSString* name = FriendProfile.name;
- NSString* country = FriendProfile.country;
- NSString* city = FriendProfile.city;
- NSString* phoneNbr = FriendProfile.phoneNbr;
總的來(lái)說(shuō),Core Data是一個(gè)非常有用的功能,可以讓你通過(guò)圖表來(lái)定義數(shù)據(jù)表和管理,可以動(dòng)態(tài)生成相應(yīng)的對(duì)象,而且無(wú)需使用復(fù)雜的SQL語(yǔ)句。但不好的方面是這里有大量的沒(méi)有經(jīng)過(guò)封裝的代碼,這樣你在使用它們與測(cè)試時(shí)需要非常小心。
Java:數(shù)據(jù)庫(kù)存取
Java有許多數(shù)據(jù)庫(kù)框架。在我看來(lái),Hibernate是和Core Data API最相像的Java框架。Hibernate使用的是對(duì)象關(guān)系映射(Object-Relational Mapping,ORM)機(jī)制,這樣你可以通過(guò)簡(jiǎn)單的在對(duì)象中設(shè)置字段并且直接映射成數(shù)據(jù)庫(kù)中的表來(lái)把對(duì)象數(shù)據(jù)放入關(guān)系型數(shù)據(jù)庫(kù)中。映射可以通過(guò)XML文件,也可以通過(guò)Java 5中的metadata annotation方法獲得。Listing Eight是使用XML進(jìn)行映射的一個(gè)例子。
Listing Eight
此例中,Listing Two中的FriendProfile對(duì)象被映射到數(shù)據(jù)庫(kù)中的一個(gè)同名表,這是一種傳統(tǒng)的數(shù)據(jù)映射做法。對(duì)象的四個(gè)字段被直接映射到表中的四列,通過(guò)映射,Hibernate可以使用SQL語(yǔ)句來(lái)完成各種操作。
另一個(gè)配置文件叫做hibernate.cfg.xml,包含了數(shù)據(jù)庫(kù)連接設(shè)置的詳細(xì)信息,包括數(shù)據(jù)庫(kù)URL、數(shù)據(jù)庫(kù)驅(qū)動(dòng)以及用戶名和密碼等,代碼請(qǐng)見(jiàn)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中我們導(dǎo)入了所有需要的Hibernate庫(kù),創(chuàng)建了一個(gè)Hibernate Session并且開(kāi)始事務(wù),接下來(lái)我們僅簡(jiǎn)單使用了Session對(duì)象的get方法就輕松檢索到了FriendProfile對(duì)象,傳遞回所需要的對(duì)象類(lèi)型并過(guò)濾出查詢的字段——朋友的姓名。#p#
結(jié)論
除去語(yǔ)法結(jié)構(gòu)與運(yùn)行平臺(tái)的不同,使用Objective-C進(jìn)行iPhone開(kāi)發(fā)與使用Java進(jìn)行網(wǎng)絡(luò)應(yīng)用開(kāi)發(fā)在下面幾個(gè)方面是相同的:
◆兩種語(yǔ)言都是面向?qū)ο蟮?/p>
◆兩種語(yǔ)言使用同樣的設(shè)計(jì)模式,例如MVC
◆兩種語(yǔ)言使用相似的數(shù)據(jù)庫(kù)存儲(chǔ)技術(shù),例如ORM
然而,對(duì)于Java開(kāi)發(fā)者,使用Objective-C時(shí)在有些地方要格外小心:
◆創(chuàng)建對(duì)象:Java對(duì)象是在運(yùn)行時(shí)通過(guò)new關(guān)鍵字創(chuàng)建的。因此Java程序員無(wú)需擔(dān)心內(nèi)存分配問(wèn)題。而在Objective-C中,一個(gè)對(duì)象可以由三個(gè)關(guān)鍵字創(chuàng)建,alloc、new或者copy,這三個(gè)關(guān)鍵字在創(chuàng)建對(duì)象時(shí)都會(huì)增加對(duì)象的持有計(jì)數(shù)(retain count),持有計(jì)數(shù)是Objective-C特有的內(nèi)存管理方法,顯示有多少個(gè)指針指向?qū)ο螅欠窨梢员粌?nèi)存管理器回收。
◆銷(xiāo)毀對(duì)象:由于強(qiáng)大的垃圾回收機(jī)制,Java的內(nèi)存管理工作極度簡(jiǎn)單。Java的引用對(duì)象都存儲(chǔ)在JVM的堆內(nèi)存中,一旦不再被引用,就可以作為垃圾回收。Objective-C使用的是內(nèi)存管理器,而不是垃圾回收器。如果你使用上面說(shuō)的三種方法在內(nèi)存中創(chuàng)建了一個(gè)對(duì)象,那么必須使用release方法來(lái)釋放對(duì)象。release方法會(huì)減少持有計(jì)數(shù),當(dāng)計(jì)數(shù)降到0時(shí),被引用的對(duì)象會(huì)接受一個(gè)來(lái)自高級(jí)類(lèi)的dealloc方法,釋放它占用的內(nèi)存并重新分配。如果忘記了釋放內(nèi)存或釋放失敗,那么會(huì)造成內(nèi)存泄露和不可預(yù)見(jiàn)的錯(cuò)誤。
◆過(guò)多釋放和過(guò)早重新分配內(nèi)存:由于垃圾回收機(jī)制,Java程序員可以完全不考慮這些問(wèn)題。但Objective-C程序員需要小心,不能釋放出比分配的更多的內(nèi)存。如果在已經(jīng)重新分配的對(duì)象上過(guò)多釋放內(nèi)存,就會(huì)造成應(yīng)用的崩潰。
上面這些例子說(shuō)明了Objective-C和Java在語(yǔ)法和語(yǔ)言元素上有很多相同之處。更重要的是,它們解決問(wèn)題的思路和用到的組件也是非常相似的。如果你是Java程序員,相信你在看完這篇文章后,轉(zhuǎn)向Objective-C的道路會(huì)更加通順。
【51CTO譯稿,非經(jīng)授權(quán)謝絕轉(zhuǎn)載,合作媒體轉(zhuǎn)載請(qǐng)注明原文出處、作者及51CTO譯者!】