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://m.survivalescaperooms.com)
新聞熱點(diǎn)
疑難解答
圖片精選