编写高效友好的多线程J2ME联网应用

原创 2004年09月16日 08:35:00
初次编写J2ME的联网应用程序的时候往往会写出这样的代码:
public void commandAction(Command c, Displayable s) {
    if (c == mExitCommand)
      notifyDestroyed();
    else if (c == mConnectCommand)
      connect();
}
 
  private void connect() {
    String url = getAppProperty("NetworkThreading.URL");
   
    try {
     
      HttpConnection hc = (HttpConnection)Connector.open(url);
      InputStream in = hc.openInputStream();
     
      int contentLength = (int)hc.getLength();
      if (contentLength == -1) contentLength = 255;
      byte[] raw = new byte[contentLength];
........
.......
}
当你运行这样的程序的时候会出现什么问题呢?wtk会提示你联网工作可能会堵塞用户输入,让你到另外一个线程去进行联网操作。OK,那么我们按照他的要求新建一个线程来进行联网,比如这样写:
public void commandAction(Command c, Displayable s) {
  if (c == mExitCommand)
    notifyDestroyed();
  else if (c == mConnectCommand) {
    Thread t = new Thread() {
      public void run() {
        connect();
      }
    };
    t.start();
  }
}
这样联网就可以通过了,那么为什么会这样呢?明白事物背后的原理显得非常的重要,下面我做一下介绍,当程序运行的时候,Application Management Software(应用管理软件)首先初始化一个MIDlet,然后调用他的startApp()方法使得MIDlet进入active状态,这条程序分支就是主线程,他执行其他的方法后都会返回到这个分支上来继续执行。然后联网是个可能堵塞的操作,意味着他可能长时间都不返回。我曾经写的联网程序在手机上运行,的确比较耗费时间。如果他不返回,那么就不能进行后面的操作,用户也不能输入。这样看上去手机就像死了一样,这显然不够友好。看看下面的原理图吧:
Hijacking the system event thread
当我们在应用程序中新建一个线程来处理联网或者浏览大量RMS数据的时候就不一样了。这时候启动线程后主线程会立刻返回,不会堵塞。
Using a new thread for network activity
仔细想想这样同样有不好的地方,尽管他可以正常工作,但是每次用户按下按钮的时候都会有新的线程产生,这样显然不够高效。幸好java中提供了wait()和notify()/notifyAll()来协调这样的问题,我们启动线程后让他进入等待的状态,当用户按下按钮的时候才让他继续运行。代码类似这样:
public synchronized void run() {
  while (mTrucking) {
    try { wait(); }
    catch (InterruptedException ie) {}
    if (mTrucking) connect();
  }
}

public synchronized void go() {
  notify();
}
Using a single worker thread
这样效率比较高了!当用户进行联网操作的时候我们应该做一个提示界面,比如一个动画告诉用户正在进行联网操作。这样比较友好。那么当用户选择联网动作的时候,我们让我提前做好的欢迎界面显示在屏幕上,联网结束后再把返回的结果显示出来。这样就是一个出色的联网应用程序了。下面的这个代码可以在屏幕上描绘一个动画的效果,当然你也可以修改一下做成自己喜欢的样子。
import java.util.*;

import javax.microedition.lcdui.*;

public class WaitCanvas
    extends Canvas {
  private int mCount, mMaximum;
  private int mInterval;
 
  private int mWidth, mHeight, mX, mY, mRadius;
  private String mMessage;
 
  public WaitCanvas() {
    mCount = 0;
    mMaximum = 36;
    mInterval = 100;
   
    mWidth = getWidth();
    mHeight = getHeight();
   
    // Calculate the radius.
    int halfWidth = (mWidth - mRadius) / 2;
    int halfHeight = (mHeight - mRadius) / 2;
    mRadius = Math.min(halfWidth, halfHeight);
   
    // Calculate the location.
    mX = halfWidth - mRadius / 2;
    mY = halfHeight - mRadius / 2;

    // Create a Timer to update the display.
    TimerTask task = new TimerTask() {
      public void run() {
        mCount = (mCount + 1) % mMaximum;
        repaint();
      }
    };
    Timer timer = new Timer();
    timer.schedule(task, 0, mInterval);
  }
 
  public void setMessage(String s) {
    mMessage = s;
    repaint();
  }
 
  public void paint(Graphics g) {
    int theta = -(mCount * 360 / mMaximum);
   
    // Clear the whole screen.
    g.setColor(255, 255, 255);
    g.fillRect(0, 0, mWidth, mHeight);
   
    // Now draw the pinwheel.
    g.setColor(0, 0, 0);
    g.drawArc(mX, mY, mRadius, mRadius, 0, 360);
    g.fillArc(mX, mY, mRadius, mRadius, theta +  90, 90);
    g.fillArc(mX, mY, mRadius, mRadius, theta + 270, 90);
   
    // Draw the message, if there is a message.
    if (mMessage != null)
      g.drawString(mMessage, mWidth / 2, mHeight,
          Graphics.BOTTOM | Graphics.HCENTER);
  }
}

你可以从这里下载SUN提供的实例代码,仔细研究一下一定受益匪浅。

基于c++11新标准开发一个支持多线程高并发的网络库

背景         新的c++11标准出后,c++语法得到了很多的扩展,比起以往任何时候都要灵活和高效,提高了程序编码的效率,为软件开发人员节省了不少的时间。 之前我也写过基于ACE的网络服务器框架...
  • lcabcdefg
  • lcabcdefg
  • 2015年07月09日 17:39
  • 1775

网页中嵌套QQ代码设置在线状态无需加好友聊天

为了更方便与客户的交流,我们经常会在网页中引用QQ,实现随时都与客户联系的效果。 有多种代码: 一、文字类型的: 1、http://wpa.qq.com/msgrd?V=1&Uin=你的...
  • q383965374
  • q383965374
  • 2015年03月19日 15:29
  • 4785

仿QQ聊天(3)—好友列表的实现

消息和动态这两个Fragment都没有都没有写。这里只写了联系人Fragment。 Fragment顶部是一个是一个搜索框,然后是新朋友,特别关心什么的。下面就是好友列表。 先模拟一些QQ好友的数...
  • u013278261
  • u013278261
  • 2015年11月05日 15:10
  • 2067

j2me 使用Timer和TimerTask来实现多线程

在Java中可以使用两种方式来实现多线程操作,这两种方式依次是: Timer类—定时器类派生自Object。它是个定时器类,作为后台线程,执行未来的任务。任务可安排一次性执行,或定期重复执行。于每个 ...
  • yaohui_zhang
  • yaohui_zhang
  • 2011年06月03日 16:16
  • 589

J2ME的多线程教程和测试

  • 2011年07月09日 14:45
  • 7KB
  • 下载

通用联网框架在J2ME平台

通用联网框架在J2ME平台中扮演着十分重要的角色,如果没有它,我们写的应用程序对外人来说只能说“神秘”,而应用了通用联网框架进行联网的程序就显得“神奇”了。本文将对通用联网框架进行整体的分析,具体的应...
  • wetgsg
  • wetgsg
  • 2014年08月08日 17:11
  • 270

j2me联网方案

1) .由于无线设备所能支持的网络协议非常有限,仅限于HTTP,Socket,UDP等几种协议,不同的厂家可能还支持其他网络协议,但是,MIDP 1.0规范规定,HTTP协议是必须实现的协议,而其他协...
  • happyq
  • happyq
  • 2011年03月17日 16:44
  • 370

J2ME最佳实践之联网开发

核心提示:编写反应灵敏的联网提示界面 由于无线设备所能支持的网络协议非常有限,仅限于HTTP,Socket,UDP等几种协议,不同的厂家可能还支持其他网络协议,但是,MIDP 1.0规范规定,HTTP...
  • leilu2008
  • leilu2008
  • 2011年08月23日 14:09
  • 517

j2me,Http,l联网

1) .由于无线设备所能支持的网络协议非常有限,仅限于HTTP,Socket,UDP等几种协议,不同的厂家可能还支持其他网络协议,但是,MIDP 1.0规范规定,HTTP协议是必须实现的协议,而其他协...
  • happyq
  • happyq
  • 2011年03月24日 11:23
  • 855

J2ME的通用联网框架

用J2ME的通用联网框架开发联网的应用程序尽管目前的无线网络不够理想,手机联网还是给我们开发人员不小的震撼的。毕竟这真的是件神奇的事情,不是吗?本文将讲述如何应用J2ME平台中的通用联网框架开发联网的...
  • zengzhongxiang
  • zengzhongxiang
  • 2011年07月22日 10:40
  • 224
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:编写高效友好的多线程J2ME联网应用
举报原因:
原因补充:

(最多只允许输入30个字)