我正在为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);
}
}
答案 0 :(得分:1)
setPreferredBackgroundFetchInterval
javadoc指出:
设置两次后台获取之间的首选时间间隔。这只是一个首选间隔,不能保证。某些平台(例如iOS)会控制何时以及是否允许后台获取。 该数字仅用作指导原则。