gpt4 book ai didi

android - 如何将一个简单的 Go 脚本作为 Android 服务运行?

转载 作者:IT王子 更新时间:2023-10-29 02:10:03 26 4
gpt4 key购买 nike

我有以下 Go 脚本 (testapp.go),我想将其作为后台服务继续运行:

package main

import(
"net/http"
"time"
"log"
"golang.org/x/mobile/app"
)

func main() {
app.Main(func(a app.App) {
for {
req, err := http.NewRequest( "GET", "http://0.0.0.0:88/fetch_news", strings.NewReader("topic.title") )
if err != nil {
log.Print(err)
}
cli := &http.Client{}
res, err := cli.Do(req)
if err != nil {
log.Print(err)
} else {
defer res.Body.Close()
body, _ := ioutil.ReadAll(res.Body)
returnStr := string(body)
// Do something with returnStr
}
time.Sleep(8 * time.Second)
}
})
}

我的 GoNativeActivity.java 如下所示:

package org.golang.app;
import android.app.Activity;
import android.app.NativeActivity;
import android.content.Context;
import android.content.pm.ActivityInfo;
import android.content.pm.PackageManager;
import android.content.Intent;
import android.os.Bundle;
import android.util.Log;
import android.view.KeyCharacterMap;

public class GoNativeActivity extends NativeActivity {
private static GoNativeActivity goNativeActivity;
public GoNativeActivity() {
super();
goNativeActivity = this;
}
String getTmpdir() {
return getCacheDir().getAbsolutePath();
}
int getRune(int deviceId, int keyCode, int metaState) {
try {
int rune = KeyCharacterMap.load(deviceId).get(keyCode, metaState);
if (rune == 0) {
return -1;
}
return rune;
} catch (KeyCharacterMap.UnavailableException e) {
return -1;
} catch (Exception e) {
Log.e("Go", "exception reading KeyCharacterMap", e);
return -1;
}
}
private void load() {
// Interestingly, NativeActivity uses a different method
// to find native code to execute, avoiding
// System.loadLibrary. The result is Java methods
// implemented in C with JNIEXPORT (and JNI_OnLoad) are not
// available unless an explicit call to System.loadLibrary
// is done. So we do it here, borrowing the name of the
// library from the same AndroidManifest.xml metadata used
// by NativeActivity.
try {
ActivityInfo ai = getPackageManager().getActivityInfo(
getIntent().getComponent(), PackageManager.GET_META_DATA);
if (ai.metaData == null) {
Log.e("Go", "loadLibrary: no manifest metadata found");
return;
}
String libName = ai.metaData.getString("android.app.lib_name");
System.loadLibrary(libName);
} catch (Exception e) {
Log.e("Go", "loadLibrary failed", e);
}
}
@Override
public void onCreate(Bundle savedInstanceState) {
load();
super.onCreate(savedInstanceState);
Intent intent = new Intent(this, org.golang.app.GoNativeService.class);
startService(intent);
}
}

我的 GoNativeService.java 如下所示:

package org.golang.app;
import android.app.PendingIntent;
import android.app.Service;
import android.content.Context;
import android.content.Intent;
import android.widget.Toast;

public class GoNativeService extends IntentService {
@Override
public void onStart(Intent intent, int startId) {
Toast.makeText(this,"Service started. onStart()", Toast.LENGTH_SHORT).show();
handleCommand(intent);
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
Toast.makeText(this,"Service started. onStartCommand()", Toast.LENGTH_SHORT).show();
handleCommand(intent);
// We want this service to continue running until it is explicitly
// stopped, so return sticky.
return START_STICKY;
}
}

我的 AndroidManifest.xml 文件如下所示:

<?xml version="1.0" encoding="utf-8"?>
<manifest
xmlns:android="http://schemas.android.com/apk/res/android"
package="org.golang.todo.testapp"
android:versionCode="1"
android:versionName="1.0">

<uses-permission android:name="android.permission.INTERNET" />

<application android:label="Testapp" android:debuggable="true">
<activity android:name="org.golang.app.GoNativeActivity"
android:label="Testactivity"
android:configChanges="orientation|keyboardHidden">
<meta-data android:name="android.app.lib_name" android:value="testapp" />
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>

<service
android:name="org.golang.app.GoNativeService"
android:enabled="true"
android:exported="true" />
</application>

</manifest>

这是我的工作环境:

GOARCH="amd64"
GOBIN="/root/go/bin"
GOCACHE="/root/.cache/go-build"
GOHOSTARCH="amd64"
GOHOSTOS="linux"
GOOS="linux"
GOPATH="/root/go"
GOROOT="/usr/lib/go-1.10"
GOTMPDIR=""
GOTOOLDIR="/usr/lib/go-1.10/pkg/tool/linux_amd64"

这是我的 $GOPATH/src/testapp 目录的样子:

testapp/
assets/
icon.png
AndroidManifest.xml
testapp.go

这是我的 $GOPATH/src/golang.org/x/mobile/app 目录的样子:

app/
android.c
app_test.go
darwin_armx.go
GoNativeActivity.java
GoNativeService.java
internal/
x11.go
android.go
darwin_amd64.go
darwin_armx.m
shiny.go
app.go
darwin_amd64.m
doc.go
x11.c

如何在 Android 上将我的 Go 脚本作为后台服务运行,以便在应用程序退出(MainActivity 关闭)时,后台服务将继续获取新闻消息?

注意:我在我的网络服务器上的端口 88 上监视请求。我可以在 GoNativeActivity 处于 Activity 状态时正常运行脚本,但我想知道如何构建一个启动后台服务。

任何能为我指明正确方向的答案或尝试,我们将不胜感激!

最佳答案

首先,有一个普遍的看法,即您真的不应该按照您的要求去做。正确的方法是使用 AlarmManager 发送一个广播来唤醒单个请求的服务。这样系统可以更好地优化功耗。为此,让您的 go 代码只处理一个请求,并在进行广播时从 Java 端调用它。

但是,要真正持久地运行该服务,您需要启动“前台服务”,如所述https://developer.android.com/guide/components/services .这又需要在您的服务运行时显示的通知。在这种情况下,您可能会发现设备也会自行关闭,在这种情况下,您需要考虑获取 https://developer.android.com/reference/android/os/PowerManager.WakeLock。 .除非您非常小心,否则执行此类操作会大大缩短电池生命周期。

我发现以这种方式启动 go 代码对于在 Android 本身上长时间运行的网络服务器(包括使用 websockets)来说已经足够了。

关于android - 如何将一个简单的 Go 脚本作为 Android 服务运行?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49461141/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com