移动信息设备简表MIDP(Mobile Information Device Profile)应用程序,即MIDlets,它的命名延续了applets和servlets的风格。对于一个有经验的java程序员来说写MIDlets程序相对容易的多。毕竟编程语言仍是java,而且,MIDP中很多来自java.lang和java.io的基本API和J2SE中的一样,学习新的API(主要来自javax.microedition的)也不是很难。
然而实际运行起来,比起J2SE来,MIDlets显得稍微复杂一点。除了基本编译环境,MIDlets还需要一些额外的开发包。完整的编译过程包含:编辑,源码,编译,类包,测试。
为了说明MIDlets的开发过程,这篇文章将建立和运行一个简单的MIDlet,读完这篇文章你应该对MIDlet开发有个全局上的了解。
工具准备 MIDlets可以在一般的桌面系统上开发,尽管它本身是专门为小设备设计的。首先你需要从Sun或者其他厂商那里获得一些工具。记住MIDP只是一种规范,所有厂商都可以自由选择自己的实现方式。现在有很多MIDlet的开发工具并且一般都是免费的。
最主要的工具应当是sun的MIDP标准实现,它包含一个预审工具(preverify tool),一个MIDP设备模拟器,部分源码和说明文档。可以从sun的官方下载。但通常我们都不用这些标准实现,除非你对MIDlets编译和打包特别感兴趣。(当然如果你想要把MIDP运行时环境移植到一个新的设备或者平台就去深入钻研吧。)
对于初学者另外一个较好用的工具是J2ME 的无线工具包WTK(WirelessToolKit)。WTK中的GUI接口隐藏了多数的MIDlets实现和打包细节,提供了从源码到实现MIDlets的简单方法。而且WTK占用资源很小,可以说是一个微型的IDE,几乎不会影响到你的机器性能。
其他还有很多来自设备制造商,无线运营商,IDE开发商和一些开源组织的大一点的IDE工具,典型的几个列在下面(包含可用链接):
Borland JBuilder X Mobile Edition
IBM WebSphere Studio Device Developer
Research In Motion BlackBerry Java Development Environment
Sun Java Studio Mobility
NetBeans IDE 4.x
Eclipse J2ME plug-in
Nokia Developer's Suite for J2ME
实际上你可以用任何你认为合适的开发工具,不过我们这里建议使用WTK,我们一下的内容将以WTK2.2进行讲解。毕竟其他的IDE都是用WTK作为Plug-in的,这样一来你的开发经验并不取决于你用的什么工具。在这篇文章里我们将详细介绍开发环境,开发工具以及模拟器的详细细节。
调试你的MIDlets 上文中提到的任何一款IDE都集成了完整的调试工具。如果单独使用WTK外加一个文本编辑器,你可以使用System.out.println()方法将调试结果在控制台中输出,WTK中的控制台可以显示给你所有的调试信息。
编写代码 我们仍然像在其他环境一样编写代码:用你最喜爱的文本编辑器编写扩展名为.java的文件。后面我们会给出一个可以加深你对黑客词典理解的的MIDlet程序实例Jargoneer,主要是在Jargon文件中查找特定的单词。当你在Jargoneer中输入一个词,它会连接到服务器去查找定义。这个MIDlet会让你公司里的黑客朋友觉得你很“cool”哦,呵呵。当有人使用一些例如“cruft”或者“grok”等不常见的词语时,你就可以通过在你的手机里面输入它并且在很短的时间里找到它的解释。下面就是Jargoneer的完整代码(你可以到Apress网站下载这个程序)。
源码如下:
import java.io.*;import javax.microedition.io.*;import javax.microedition.midlet.*;import javax.microedition.lcdui.*;public class Jargoneer extends MIDlet implements CommandListener, Runnable { private Display mDisplay; private Command mExitCommand, mFindCommand, mCancelCommand; private TextBox mSubmitBox; private Form mProgressForm; private StringItem mProgressString; public Jargoneer() { mExitCommand = new Command("Exit", Command.EXIT, 0); mFindCommand = new Command("Find", Command.SCREEN, 0); mCancelCommand = new Command("Cancel", Command.CANCEL, 0); mSubmitBox = new TextBox("Jargoneer", "", 32, 0); mSubmitBox.addCommand(mExitCommand); mSubmitBox.addCommand(mFindCommand); mSubmitBox.setCommandListener(this); mProgressForm = new Form("Lookup progress"); mProgressString = new StringItem(null, null); mProgressForm.append(mProgressString); } public void startApp() { mDisplay = Display.getDisplay(this); mDisplay.setCurrent(mSubmitBox); } public void pauseApp() {} public void destroyApp(boolean unconditional) {} public void commandAction(Command c, Displayable s) { if (c == mExitCommand) { destroyApp(false); notifyDestroyed(); } else if (c == mFindCommand) { // Show the progress form. mDisplay.setCurrent(mProgressForm); // Kick off the thread to do the query. Thread t = new Thread(this); t.start(); } } public void run() { String word = mSubmitBox.getString(); String definition; try { definition = lookUp(word); } catch (IOException ioe) { Alert report = new Alert( "Sorry", "Something went wrong and that " + "definition could not be retrieved.", null, null); report.setTimeout(Alert.FOREVER); mDisplay.setCurrent(report, mSubmitBox); return; } Alert results = new Alert("Definition", definition, null, null); results.setTimeout(Alert.FOREVER); mDisplay.setCurrent(results, mSubmitBox); } private String lookUp(String word) throws IOException { HttpConnection hc = null; InputStream in = null; String definition = null; try { String baseURL = "http://65.215.221.148:8080/wj2/jargoneer?word="; String url = baseURL + word; mProgressString.setText("Connecting..."); hc = (HttpConnection)Connector.open(url); hc.setRequestProperty("Connection", "close"); in = hc.openInputStream(); mProgressString.setText("Reading..."); int contentLength = (int)hc.getLength(); if (contentLength == -1) contentLength = 255; byte[] raw = new byte[contentLength]; int length = in.read(raw); // Clean up. in.close(); hc.close(); definition = new String(raw, 0, length); } finally { try { if (in != null) in.close(); if (hc != null) hc.close(); } catch (IOException ignored) {} } return definition; }}
编译MIDlet
写MIDlets应用程序是一种交叉编译,意味着你要在一个平台编译而在另外一个平台运行。具体说就是要在我们自己的电脑平台上使用J2SE编译MIDlets,编译好的MIDlet可以运行于移动电话,寻呼机,或者其他支持MIDP的移动信息设备。只要你将源程序放在正确的目录下,WTK可以完成所以的细节工作。
1 启动Ktoolbar
2 从工具栏选择新建工程命令建立一个新项目。
3 看到WTK提示,输入Jargoneer作为工程名MIDlet类名。
4 点击建立工程按钮,然后“OK”来关闭工程设定窗口。
图示如下:
图1 新建工程对话框 WTK将工程保存在默认目录\apps下。以下是目录说明:
<J2ME Wireless Toolkit directory> apps Jargoneer bin lib res src
将上文源码命名Jargoneer.java保存到src目录下,你可以简单的单击工具栏上的Build按钮编译已经打开的MIDlet工程。接着,WTK使用J2SE的编译器进行编译。一般说来,如果你编译J2SE程序,classpath环境变量会指向所有你程序需要联系的类。当你使用javac命令进行编译文件的时候,一些例如java.lang的API就会被导入,意思即是说在你的MIDlet中使用了类java.lang.System。那么怎样才能让编译器知道你要使用的是这个类的MIDP版本而不是J2SE版本呢?其实我们有一个命令行选项-bootclasspath,它允许你指出描述你将要使用的基础API的classpath。这个选项这个时候就用来改变MIDP安装默认的classes目录。例如如下的命令所示:
javac -bootclasspath\midp\classes Jargoneer.java
如果你的MIDP安装在不同的路径,你需要把它调整到classes路径下。
预审类文件 下一个全新的编译步骤就是预审了。由于小设备的存储限制,MIDP(确切的说CLDC,即Connected Limited Device Configuration,受限设备配置简表)详细说明了比特码的验证被分成两个部分,preverify将在设备以外的那部分起作用。而设备本身仅仅用来在加载class以前做一些轻松的二次验证工作。如果使用WTK,你完全不用担心预审,你单击Build的时候它会帮你自动完成,你甚至都无法意识到。如果你想要了解更多关于预审器的东西,就继续看下去,否则可以直接跳过这一节。比特码验证在java运