代号为One的BackgroundFetch

时间:2019-06-10 12:52:30

标签: ios codenameone

我正在为iOS开发一个Codename One应用,并且试图使用BackgroundFetch界面。

我复制了用Javadoc(https://www.codenameone.com/javadoc/com/codename1/background/BackgroundFetch.html)编写的示例代码,并添加了 ios.background_modes = fetch 构建提示。

在模拟器上启动应用程序,可以正确执行后台操作。

在实际设备(iPhone 7s,iOs 12.1.4)上启动它时,行为是无法预测的。尽管有setPreferredBackgroundFetchInterval(10),但我几乎每次启动应用程序时都会注意到,后台操作未执行。很少会执行后台操作,但是应用必须在后台运行几分钟才能恢复它,而不是通过setPreferredBackgroundFetchInterval(10)方法设置的10秒。

Display.isBackgroundFetchSupported()方法返回true。

如果负担得起和可预测的话,我不知道该怎么做。


编辑

我仅在performBackgroundFetch()实现中修改了示例代码(Display.setPreferredBackgroundFetchInterval(10)不变)。我只是在标签上放了一些文字:

@Override
public void performBackgroundFetch(long deadline, Callback<Boolean> onComplete) {
    supported.setText("deadline: " + deadline + "; timeMillis: " + System.currentTimeMillis());
    onComplete.onSucess(Boolean.TRUE);
}

我观察到模拟器和真实设备的两种不同行为。 在模拟器中,进入暂停状态后恰好10秒钟执行该方法。在实际设备中,该方法在进入暂停状态后10秒钟不会执行:在某些情况下,它会在20分钟后执行(在其他情况下,则根本不会执行)。 但是,在这两种情况下,我都可以计算出截止日期与方法执行时间之间的差:始终为25分钟。

作为示例,您可以看到该应用程序的以下屏幕截图(在iPhone上运行):

最后期限= 1560246881647 时间戳= 1560245381647

截止日期-时间戳= 1500000毫秒= 1500秒= 25分钟。

据我了解,在iOS上,执行后台抓取操作的时间限制为30秒,否则操作系统将终止该应用程序。此外,Display.setPreferredBackgroundFetchInterval()用于设置后台获取之间的首选时间间隔,但不能保证,因为iOS可以控制后台获取的执行。

使用后台获取的正确方法是什么?

这是完整的代码:

public class MyApplication implements BackgroundFetch{

   private Form current;
   private Resources theme;
   List<Map> records;
   Label supported;
   // Container to hold the list of records.
   Container recordsContainer;

   public void init(Object context) {
      theme = UIManager.initFirstTheme("/theme");

      // Enable Toolbar on all Forms by default
      Toolbar.setGlobalToolbar(true);

      // Pro only feature, uncomment if you have a pro subscription
      // Log.bindCrashProtection(true);
   }

   public void start() {
      if(current != null){
         // Make sure we update the records as we are coming in from the 
         // background.
         updateRecords();
         current.show();
         return;
      }
      Display d = Display.getInstance();

      // This call is necessary to initialize background fetch
      d.setPreferredBackgroundFetchInterval(10);

      Form hi = new Form("Background Fetch Demo");
      hi.setLayout(new BoxLayout(BoxLayout.Y_AXIS));

      supported = new Label();
      if (d.isBackgroundFetchSupported()){
         supported.setText("Background Fetch IS Supported");
      } else {
         supported.setText("Background Fetch is NOT Supported");
      }

      hi.addComponent(new Label("Records:"));
      recordsContainer = new Container(new BoxLayout(BoxLayout.Y_AXIS));
      //recordsContainer.setScrollableY(true);
      hi.addComponent(recordsContainer);
      hi.addComponent(supported);
      updateRecords();
      hi.show();
  }

  /**
   * Update the UI with the records that are currently loaded.
   */
  private void updateRecords() {
     recordsContainer.removeAll();
     if (records != null) {
        for (Map m : records) {
            recordsContainer.addComponent(new SpanLabel((String)m.get("title")));
        }
     } else {
        recordsContainer.addComponent(new SpanLabel("Put the app in the background, wait 10 seconds, then open it again.  The app should background fetch some data from the Slashdot RSS feed and show it here."));
     }
     if (Display.getInstance().getCurrent() != null) {
        Display.getInstance().getCurrent().revalidate();
     }
  }

  public void stop() {
     current = Display.getInstance().getCurrent();
     if(current instanceof Dialog) {
        ((Dialog)current).dispose();
        current = Display.getInstance().getCurrent();
     }
  }

  public void destroy() {
  }

  /**
   * This method will be called in the background by the platform.  It will
   * load the RSS feed.  Note:  This only runs when the app is in the background.
   * @param deadline
   * @param onComplete 
   */
  @Override
  public void performBackgroundFetch(long deadline, Callback<Boolean> onComplete) {
     supported.setText("deadline: " + deadline + "; timeMillis: " + System.currentTimeMillis());
     onComplete.onSucess(Boolean.TRUE);
  }

}

1 个答案:

答案 0 :(得分:1)

setPreferredBackgroundFetchInterval javadoc指出:

  

设置两次后台获取之间的首选时间间隔。这只是一个首选间隔,不能保证。某些平台(例如iOS)会控制何时以及是否允许后台获取。 该数字仅用作指导原则