Record Management System是J2ME的一個重要的子系統(tǒng),目的是實(shí)現(xiàn)應(yīng)用程序本地?cái)?shù)據(jù)的持久性存儲。目前支持文件系統(tǒng)的移動信息設(shè)備還有限,因此Record Management System是J2ME開發(fā)人員實(shí)現(xiàn)本地?cái)?shù)據(jù)存儲的首選途徑。本文的目的就是全面的介紹Record Management System的知識。
顧名思義Record Management System是管理數(shù)據(jù)的系統(tǒng),Record是系統(tǒng)中最重要的實(shí)體。在移動設(shè)備存儲空間存儲的并不是字段,而是字節(jié)數(shù)組。Mobile Infomation Device PRofile(MIDP)規(guī)范中并沒有規(guī)定什么樣的數(shù)據(jù)才能存儲為記錄,事實(shí)上記錄是任何可以被字節(jié)數(shù)組表示的數(shù)據(jù),例如圖片、文本等。Record Management System的職責(zé)是存儲和唯一標(biāo)識記錄,而表示數(shù)據(jù)的任務(wù)是由應(yīng)用程序來完成的,因此J2ME的開發(fā)人員往往要花費(fèi)更多的精力來處理存儲空間中的數(shù)據(jù)。這樣做的目的是簡化MIDP的實(shí)現(xiàn),使得J2ME的子系統(tǒng)盡量的小巧、靈活。畢竟移動信息設(shè)備的存儲空間和處理器的能力都有限。
Record Store是一系列記錄的有序集合,記錄是不能單獨(dú)存在的,必須屬于Record Store。Record Store保證記錄的讀寫操作都是原子的,數(shù)據(jù)不會被破壞。在API中Record Store是由javax.microedition.rms.RecordStore實(shí)現(xiàn)的,關(guān)于RecordStore的具體操作在接下來的文章中會有詳細(xì)的介紹。
MIDP規(guī)范中說明移動信息設(shè)備要提供至少8K的非易失性存儲空間給應(yīng)用程序來實(shí)現(xiàn)數(shù)據(jù)的持久性存儲。但是不同的設(shè)備提供的空間并不相同。如果MIDlet suite使用了Record Management System,那么它必須在MANIFEST文件和JAD文件中通過設(shè)置MIDlet-Data-Size來說明它所需要的最小的數(shù)據(jù)存儲空間,單位是字節(jié),例如MIDlet-Data-Size:8192。如果你的值超過了移動設(shè)備規(guī)定的最大值那么你的應(yīng)用程序?qū)⒉荒苷_安裝。這個值并不是移動設(shè)備真正提供給應(yīng)用程序的最大Record Management System的存儲空間,往往要大一些,因此開發(fā)人員應(yīng)該避免把應(yīng)用程序需要的最小存儲空間設(shè)置的過大,必要的時候應(yīng)該參考相關(guān)設(shè)備的說明手冊。在非易失性存儲空間內(nèi)讀寫數(shù)據(jù)往往速度會比較慢,因此針對頻繁訪問的數(shù)據(jù)最好提供緩存的機(jī)制來提供性能。Record Management System的讀寫操作是線程安全的,但是由于Record Store是被整個MIDlet suite共享的,所以如果不同MIDlet上運(yùn)行的線程操作Record Store的時候,我們應(yīng)該進(jìn)行必要的線程同步,避免數(shù)據(jù)被破壞。
MIDP1.0和MIDP2.0中關(guān)于Record Management System的實(shí)現(xiàn)有些不同,在同一個MIDlet suite里面的MIDlets可以相互訪問彼此的Record Store。但是在MIDP1.0的實(shí)現(xiàn)中,并沒有提供在不同MIDlet suite之間共享Record Store的機(jī)制。在MIDP2.0中提供的了新的API來解決不同MIDlet suite之間共享Record Store的問題,在創(chuàng)建Record Store的時候通過授權(quán)模式和讀寫控制參數(shù)來進(jìn)行共享機(jī)制的管理,我將在下篇文章中進(jìn)行詳細(xì)的介紹。
加強(qiáng)對Record Management System的理解的最好的辦法就是進(jìn)行實(shí)際的開發(fā),在進(jìn)行開發(fā)中我發(fā)現(xiàn)并不是所有移動設(shè)備的MIDP實(shí)現(xiàn)都準(zhǔn)確無誤。當(dāng)我用getSizeAvaliable()方法查詢Nokia6108的可用Record Store空間的時候得到的數(shù)值是超過1M字節(jié),但是當(dāng)我寫入40K的數(shù)據(jù)的時候就出現(xiàn)了RecordStoreFullException異常,因此我編寫了一個自動測試手機(jī)Record Store最大存儲空間的軟件。原理是每隔一定時間例如100-500毫秒向Record Store內(nèi)寫入1K字節(jié)的數(shù)據(jù),當(dāng)拋出存儲空間已滿的異常的時候就可以得到最大值了,精確單位為K字節(jié)。下面是程序的源代碼和JAD文件的內(nèi)容,開發(fā)平臺為Eclipse3.0RC2+EclipseME0.4.1+Wtk2.1+J2SDK1.4.2._03,在真機(jī)Nokia 6108上測試通過并顯示最大值為31K。(請不要在模擬器上進(jìn)行測試,那樣結(jié)果沒有意義)
總結(jié):本文只是帶領(lǐng)讀者對Record Management System進(jìn)行了大概的了解,雖然在文章最后提供了一個應(yīng)用程序。但是并沒有深入分析如何使用Record Management System。在接下來的文章中我們會深入分析javax.microedition.rms包中的類,重點(diǎn)是如何使用RecordStore類。
import javax.microedition.lcdui.Alert;
import javax.microedition.lcdui.AlertType;
import javax.microedition.lcdui.Display;
import javax.microedition.midlet.MIDlet;
import javax.microedition.midlet.MIDletStateChangeException;
import javax.microedition.rms.RecordStoreException;
public class RMSAnalyzer extends MIDlet
{
private Display display;
private CounterCanvas counterCanvas;
private Alert alert;
protected void startApp() throws MIDletStateChangeException
{
display = Display.getDisplay(RMSAnalyzer.this);
alert = new Alert("錯誤提示");
try
{
String interval = this.getAppProperty("INTER");
int t = Integer.parseInt(interval);
counterCanvas = new CounterCanvas(t, 1, this);
}
catch (RecordStoreException e)
{
this.showAlertError(e.getMessage());
}
display.setCurrent(counterCanvas);
}
public Display getDisplay()
{
return display;
}
protected void pauseApp()
{
}
protected void destroyApp(boolean arg0) throws MIDletStateChangeException
{
}
public void showAlertError(String message)
{
alert.setString(message);
alert.setType(AlertType.ERROR);
alert.setTimeout(3000);
display.setCurrent(alert);
}
}
import java.util.Timer;
import java.util.TimerTask;
import javax.microedition.lcdui.Canvas;
import javax.microedition.lcdui.Command;
import javax.microedition.lcdui.CommandListener;
import javax.microedition.lcdui.Displayable;
import javax.microedition.lcdui.Graphics;
import javax.microedition.midlet.MIDletStateChangeException;
import javax.microedition.rms.*;
public class CounterCanvas extends Canvas implements CommandListener
{
private RMSModel model;
private RMSAnalyzer RMSanalyzer;
private int interTime;
private int counter;
private boolean go = true;
public static Command backCommand = new Command("退出", Command.EXIT, 3);
public static final int INC = 1;
public final Timer timer = new Timer();
public CounterCanvas(int interTime, int base, RMSAnalyzer rmsa)
throws RecordStoreException
{
this.interTime = interTime;
this.counter = base;
this.RMSanalyzer = rmsa;
model = new RMSModel(base, RMSanalyzer);
this.addCommand(backCommand);
this.setCommandListener(this);
TimerTask timerTask = new TimerTask()
{
public void run()
{
try
{
model.writeRecord(INC);
counter++;
} catch (RecordStoreFullException e)
{
go = false;
model.deleteRMS();
timer.cancel();
} catch (RecordStoreException e)
{
model.deleteRMS();
RMSanalyzer.showAlertError(e.getMessage());
timer.cancel();
}
repaint();
}
};
timer.schedule(timerTask, 1000, interTime);
}
public void setCounter(int counter)
{
this.counter = counter;
}
public void setInterTime(int interTime)
{
this.interTime = interTime;
}
protected void paint(Graphics arg0)
{
int SCREEN_WIDTH = this.getWidth();
int SCREEN_HEIGHT = this.getHeight();
arg0.drawRect(SCREEN_WIDTH / 10, SCREEN_HEIGHT / 2,
SCREEN_WIDTH * 4 / 5, 10);
if(RMSanalyzer.getDisplay().isColor())
{
arg0.setColor(128, 128, 255);
}
arg0.fillRect(SCREEN_WIDTH / 10, SCREEN_HEIGHT / 2, counter, 10);
if (!go)
arg0.drawString("最大值:" + counter+"K字節(jié)", 0, 0, Graphics.TOP
Graphics.LEFT);
}
public void commandAction(Command arg0, Displayable arg1)
{
if (arg0 == backCommand)
{
try
{
model.deleteRMS();
RMSanalyzer.destroyApp(false);
RMSanalyzer.notifyDestroyed();
} catch (MIDletStateChangeException e)
{
}
}
}
}
import javax.microedition.rms.*;
public class RMSModel
{
public static final int K = 1024;
private RecordStore rs;
private int baseCount;
private RMSAnalyzer RMSanalyzer;
public static final String name = "test";
public RMSModel(int baseCount, RMSAnalyzer rmsa)
throws RecordStoreException
{
this.baseCount = baseCount;
this.RMSanalyzer = rmsa;
if (rs == null)
{
rs = RecordStore.openRecordStore(name, true);
writeRecord(baseCount);
}
}
public void writeRecord(int count) throws RecordStoreException
{
byte[] data = new byte[count * K];
for (int i = 0; i < count; i++)
{
data[i] = 1;
}
rs.addRecord(data, 0, count * K);
}
public void deleteRMS()
{
try
{
rs.closeRecordStore();
RecordStore.deleteRecordStore(name);
} catch (RecordStoreException e)
{
RMSanalyzer.showAlertError(e.getMessage());
}
}
}
RMSAnalyzer.jad
MIDlet-Jar-Size: 5293
MIDlet-1: RMSAnalyzer,,RMSAnalyzer
MIDlet-Jar-URL: RMSAnalyzer.jar
MicroEdition-Configuration: CLDC-1.0
MIDlet-Version: 1.0.0
MIDlet-Name: RMSAnalyzer
MIDlet-Data-Size: 8192
MIDlet-Vendor: Midlet Suite Vendor
MicroEdition-Profile: MIDP-1.0
INTER: 100
(出處:http://www.survivalescaperooms.com)
新聞熱點(diǎn)
疑難解答
圖片精選