編寫靈活的RMS應(yīng)用
MIDP應(yīng)用程序的標(biāo)準(zhǔn)持久化方案就是使用RMS。RMS類似于一個小型數(shù)據(jù)庫,RecordStore相當(dāng)于數(shù)據(jù)庫的表,每個“表”由若干記錄(Record)構(gòu)成,一條記錄就是一個用int表示的記錄號RecordID和用byte[]表示的內(nèi)容。記錄號可以看作是“主鍵”,byte[]數(shù)組存儲內(nèi)容。
RMS提供的記錄操作可以實現(xiàn)根據(jù)ID直接獲得記錄,或者枚舉出一個表中的所有記錄。
枚舉記錄是非常低效的,因為只能比較byte[]數(shù)據(jù)來確定該記錄是否是所需的記錄。通過ID獲得記錄是高效而方便的,類似于SQL語句“SELECT byteArrayData FROM recordStoreName WHERE RecordID=?”。然而,通常應(yīng)用程序很難知道某條記錄的ID號,而RMS記錄的“主鍵”又僅限于int類型,無法使用其他類型如String作為“主鍵”來查找。因此,對于需要存取不同類型對象的應(yīng)用程序而言,就需要一個靈活的RMS操作框架。
我們的基本設(shè)想是,如果能使用String作為“主鍵”來查找記錄,就能非常方便地獲得所需的內(nèi)容。例如,應(yīng)用程序設(shè)置可以通過"sys.settings"獲得byte[]數(shù)組,并依次讀取出設(shè)置,用戶登錄信息可以通過"user.info"獲得byte[]數(shù)組,再分解出用戶名和口令。
因此,我們實現(xiàn)一個StorageHandler類,提供唯一的RMS訪問接口,使得其他類完全不必考慮底層的RMS操作,只需提供能標(biāo)識自身的一個String即可。
如果我們能實現(xiàn)一種類似于數(shù)據(jù)庫索引的查找表,就能根據(jù)String關(guān)鍵字查找某條記錄。因此,我們使用一個名為"index"的RecordStore來存儲所有的索引,每一條索引都指向某一條具體記錄的ID,設(shè)計一個IndexEntry表示一條索引:
class IndexEntry {
private int selfId; // IndexEntry的ID
private int recordId; // 對應(yīng)記錄的ID
private String key; // 訪問記錄的Key
}
根據(jù)索引查找,分3步進行:
1.在名為"index"的RecordStore中根據(jù)String查找對應(yīng)的IndexEntry。
2.取出IndexEntry,獲得記錄ID號。
3.根據(jù)ID號獲得另一個RecordStore的記錄,然后就可以讀取、更新和刪除該記錄。
如下圖所示:

由于IndexEntry保存的數(shù)據(jù)很少,為了加快查找速度,可以在應(yīng)用程序啟動時,把所有的IndexEntry讀入一個Vector,在后面的操作中更新這個Vector并與RecordStore保持同步。
為了處理不同類型的數(shù)據(jù),所有可通過StorageHandler存取的類都必須實現(xiàn)一個Storable接口:
public interface Storable {
String getKey();
void getData(DataOutputStream output) throws IOException;
void setData(DataInputStream input) throws IOException;
}
前面已經(jīng)提到,在MIDP應(yīng)用程序中,序列化一個類的最佳方法是使用DataInputStream和DataOutputStream。因此,需要持久化的類可以通過getData()和setData()方法非常方便地存取。假定應(yīng)用程序的類UserInfo保存了用戶的登錄名、口令和是否自動登錄的信息:
public class UserInfo {
String username;
String password;
boolean autoLogin;
}
為了能將UserInfo存入RMS,需要實現(xiàn)Storable接口:
class UserInfo implements Storable {
String username;
String password;
boolean autoLogin;
public String getKey() { return "user.info"; } // 提供一個唯一標(biāo)識符即可
public void getData(DataOutputStream output) throws IOException {
output.writeUTF(username);
output.writeUTF(password);
output.writeBoolean(autoLogin);
}
public void setData(DataInputStream input) throws IOException {
username = input.readUTF();
password = input.readUTF();
autoLogin = input.readBoolean();
}
// getters here...
}
要保存UserInfo,只需調(diào)用StorageHandler的保存方法:
StorageHandler.storeOrUpdate(userinfo);
要讀取UserInfo,調(diào)用StorageHandler的讀取方法:
UserInfo userinfo = new UserInfo();
StorageHandler.load(userinfo);
這樣,需要讀取或保存數(shù)據(jù)的類完全不必涉及底層的RMS操作,大大簡化了應(yīng)用程序的設(shè)計,增強了源代碼的可復(fù)用性與可維護性。