在 ListField Blackberry 上异步加载图像



如何在列表字段上异步显示从 url 获取的图像?运行代码时出现异常。这是我的代码。我正在收到未捕获的NoClassFoundError

private Bitmap getBitmap(final String strEventCode)
{
    if(hst.containsKey(strEventCode))
        return (Bitmap) hst.get(strEventCode);
    else
    {           
        Thread t = new Thread(new Runnable() 
        {
            public void run() 
            {                       
                Bitmap bmp = HttpUtils.getBitmap(strHalfUrl+strEventCode+".jpg");
                hst.put(strEventCode, bmp);
            }
        });
        t.start();
    }
    return null; 
}

我使用以下代码使用ListFieldCallBack绘制图像:

class ListCallBack implements ListFieldCallback
{       
   public  void drawListRow(final ListField list, final net.rim.device.api.ui.Graphics g, final int index, final int y, final int w)
   {                  
       Event objEvent = (Event) eventData.elementAt(index);
       if(list.getSelectedIndex() == index)
       {              
           g.setColor(Color.LIGHTGRAY);
           g.fillRect(0, y, getWidth(), getHeight());            
       }
       Bitmap bmp = getBitmap(objEvent.getStrName());
       if(bmp==null)
           g.drawBitmap(0, y+5, loadingImage.getWidth(),loadingImage.getHeight(),loadingImage, 0, 0);
       else
           g.drawBitmap(0, y+5, bmp.getWidth(),bmp.getHeight(),bmp, 0, 0);
       g.setColor(Color.BLACK);
       int yPos = y + list.getRowHeight() - 1;
       g.drawLine(0, yPos, w, yPos);
       //final Bitmap b=(Bitmap)myImages.elementAt(index);
       //g.drawBitmap(0, y+5, b.getWidth(),b.getHeight(),b, 0, 0);         
   } 
   public Object get(ListField list, int index)
   { 
       return eventData.elementAt(index); 
   } 
   public int getPreferredWidth(ListField list)
   {
       return Display.getWidth();
   }
   public int indexOfList(ListField listField, String prefix, int start) 
   {
      return eventData.indexOf(prefix,start);
   }
}

我会改变一些事情:

  • 在BB中,你可以生成的最大线程数,我认为是<20。您正在为每个图像生成一个线程,因此您迟早会遇到TooManyThreadsException。相反,您应该在屏幕中有一个下载图像的工作线程。您可以使用使用者-生产者模式:用户向下滚动时生成图像下载请求(生产者(,工作线程(使用者(将它们排队等待处理。您可以使用由 Vector 自定义的 LIFO 结构,并在没有请求时对其进行wait
  • 您应该提供一个默认图像,以便在尚未下载真实图像时显示。
  • 我几乎可以肯定您当前的代码对于hst哈希表不是线程安全的。您应该确保您的使用者是线程安全的。网上有很多关于如何做到这一点的例子。
  • 当然,工作线程应该在用户更改屏幕时终止。在当前代码中,线程保持活动状态,即使用户已更改为另一个屏幕,线程也会尝试更新哈希表。

作为优化,如果您事先知道您的应用程序通常有多少线程(不计算图像(,并且您知道不会超过最大线程限制,则可以拥有一个线程池,例如,5 个线程来下载图像而不是单个工作线程。当所有 5 个线程都繁忙时,您将开始对请求进行排队。您还可以添加每个请求的最大时间机制,以防止线程忙于下载失败,因此不是在 2 分钟超时,而是在 30 秒超时并发出第二个请求。

什么是 HttpUtils.getBitmap((? 如果它是为Java-SE编写的Java代码,那么它在BlackBerry上将无法正常工作,因为BlackBerry设备仅支持Java-ME,其功能远低于现代Java-SE运行时。

至于异步加载,您需要在获取完成后将事件传递回 UI。 如果要一次获取多张照片,则还需要向该事件添加某种批处理,因为为每张照片发送事件可能会使 UI 应用程序上的事件队列溢出。

最新更新