2

概述:

我们的应用程序扩展了一个 UIApplication 并有一个在启动时注册的 SMS Listener 类。当收到符合我们标准的消息时,我们会处理该消息,然后我们希望将其保存到本地 SQLite 数据库并将其上传到 Web 服务器。重要的是,在收到 SMS 后尽快发生这种情况,即使 UI 应用程序在那个阶段没有打开。

问题:

当 SMSListener 实例在后台运行,没有 UIApplication 实例处于活动状态,并且想要访问 SQLite 数据库或尝试创建 HTTP 连接时,会抛出“No Application Instance”异常。

期望的结果:

即使 UIApplication 未处于活动状态,我们也希望处理、保存和上传来自 SMSListener 后台线程的所有消息。目前 SMSListener 后台线程会将消息存储在 RuntimeStore 中;当 UI 应用程序启动时,它会从 RuntimeStore 读取消息并将其保存到数据库中。虽然这不是最佳解决方案,因为与 Web 服务器的同步也只会在下次打开 UI 应用程序时发生。重要的是它在收到消息时进行同步。

应用程序伪代码:

Main Class,检查启动并创建 SMSListener 实例或从 RuntimeStore 获取实例。

public class OurAppUi extends UiApplication {
public static void main(String[] args) {
    if (args != null && args.length > 0 && args[0].endsWith("gui")) {
        // Create a new instance of the application and make the currently
        // running thread the application's event dispatch thread.
        OurAppUi theApp = new OurAppUi();
        theApp.enterEventDispatcher();
    } else {
        // Entered through the alternate application entry point
        SmsListener.waitForSingleton();
    }
}
}

SMSListener 类使用 RuntimeStore 单例模型侦听任何传入消息。这按预期工作。

public class SmsListener implements javax.wireless.messaging.MessageListener {
public static SmsListener waitForSingleton() {
    //Ensure this is a singleton instance.
    //Open RuntimeStore and obtain the reference of BackgroundListener
    RuntimeStore store = RuntimeStore.getRuntimeStore();
    Object obj = store.get(ID_BACKGROUND_LISTENER);

    //If obj is null, there is no current reference to BackgroundListener
    //Start a new instance of BackgroundLIstener if one is not running
    if(obj == null) {
        store.put(ID_BACKGROUND_LISTENER, new SmsListener());
        return (SmsListener)store.get(ID_BACKGROUND_LISTENER);
    } else {
        return(SmsListener)obj;
    }
}

public void notifyIncomingMessage(MessageConnection conn) {
    new Thread() {
        MessageConnection connection;
        Thread set (MessageConnection con) {
            this.connection = con;
            return (this);
        }

        public void run() {
            try {
                Message m = connection.receive();
                String msg = null;

                if (m instanceof TextMessage) {
                    TextMessage tm = (TextMessage)m;
                    msg = tm.getPayloadText();
                }
                // Process the SMS
                SMSObject sms = processSMS(msg);
                // Save to DataBase { Exception is Thrown Here }
                SQLManager.getInstance().save(sms);
                // Upload to Web Server { Exception is Thrown Here }
                WebServer.upload(sms);

            } catch(Exception error) {
            }
        }
    }.set(conn).start();

}
}

当 SmsListener 实例想要访问 SQLite 数据库或尝试创建 HTTP 连接时,会抛出“No Application Instance”异常。

public final class SQLManager {
private SQLManager() {
    try {
        db = OpenOrCreateDatabase();
    } catch (MalformedURIException e) {
        Debug.log(TAG, "Get connection: URI: " + e.getMessage());   
    } catch (ControlledAccessException e) {
        Debug.log(TAG, "Get connection: Controlled Access: " + e.getMessage()); 
    } catch (DatabasePathException e) {
        Debug.log(TAG, "Get connection: Database Path: " + e.getMessage()); 
    } catch (DatabaseIOException e) {
        Debug.log(TAG, "Get connection: Database IO: " + e.getMessage());   
    } catch (Exception e) {
        Debug.log(TAG, e);  
    }
}

public static synchronized SQLManager getInstance() {
    if (instance == null) {
        instance = new SQLManager();
    }
    return instance;
}
}

我们尝试将 SQLite 实例存储在 RuntimeStore 中,使用与 SMSListener 相同的单例模型,但当 UI 应用程序尝试访问存储的数据库实例时收到错误。

4

1 回答 1

2

一般来说,处理此类活动的方法是将应用程序分为两部分:

  1. 需要 UI 且仅在用户想要与应用程序交互时才运行的用户交互部分;
  2. 将存储数据并与远程服务器通信的后台处理部分。

后台处理应该在 net.rim.device.api.system.Application 扩展的上下文中进行,这可能应该是基于 RuntimeStore 的单例。这部分应该从您的自动运行代码开始,注册侦听器并保持活动状态。确保代码在正确的上下文中执行涉及一些复杂性。我有一篇博文可能会有所帮助。

于 2012-09-06T13:39:43.993 回答