bindService force关闭

时间:2011-08-12 14:59:13

标签: android service binding android-intent

我对Android编程和Java一般都很新,但我作为系统级C编码器有丰富的经验。我想创建一个可以被我计划开发的一系列Android应用程序使用的API或库,并经过大量研究后决定绑定服务是Android的方式(我通常会创建一个.so)或.dll)。如果这个假设不正确,那么我的整个问题很可能没有实际意义。

无论如何,我在一个项目中创建了我的服务框架,并在另一个项目中创建了一个活动来测试服务。我通过调用Toast来查看服务的handleMessage()方法,看看它是否正常工作,然后在我的活动中添加了两个按钮:一个用于绑定服务,另一个用于解除绑定。经过大约一个小时无法运行我的活动,因为它无法看到我的服务,我想出了如何让Eclipse从活动中查看服务的源代码,并认为我在路上很顺利。然后我运行了活动。

一切看起来都不错。两个按钮,一个绑定,一个解除绑定。但是,当我按下绑定按钮时,它强制关闭应用程序。我使用调试器来查看发生了什么,并且在我的活动中强制关闭bindService()调用。当然,bindService()调用是从另一个在同一行创建新Intent()的例子中复制的,所以我将两行代码分开,当我在真的死时尝试创建意图。

以下是我的活动代码:

package net.william.android.myAPI.tester;

import net.william.android.myAPI.MyApiService;
import android.app.Activity;
import android.content.ComponentName;
import android.content.Context;
import android.content.Intent;
import android.content.ServiceConnection;
import android.os.Bundle;
import android.os.IBinder;
import android.view.View;
import android.widget.Toast;

public class MyApiTester extends Activity
{
    boolean mIsBound = false;

    private ServiceConnection mConnection = new ServiceConnection()
    {
        public void onServiceConnected(ComponentName className, IBinder service)
        {
            Toast.makeText(FaceClientTester.this, "Remote Service Connected", Toast.LENGTH_SHORT).show();
        }

        public void onServiceDisconnected(ComponentName className)
        {
            Toast.makeText(FaceClientTester.this, "Remote Service Disconnected", Toast.LENGTH_SHORT).show();
        }
    };

    public void doBindService(View view)
    {
        if (!mIsBound)
        {
            Toast.makeText(this, "Binding Service", Toast.LENGTH_SHORT).show();
            Intent myIntent = new Intent(MyApiTester.this, MyApiService.class); 
            bindService(myIntent, mConnection, Context.BIND_AUTO_CREATE);
            mIsBound = true;
        }
    }

    public void doUnbindService(View view)
    {
        if (mIsBound)
        {
            unbindService(mConnection);
            mIsBound = false;
            Toast.makeText(this, "Service Unbound", Toast.LENGTH_SHORT).show();
        }
    }

    @Override
    public void onCreate(Bundle savedInstanceState)
    {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        return;
    }
}

这是我服务的代码:

package net.william.android.myAPI

import android.app.Service;
import android.content.Intent;
import android.os.Handler;
import android.os.IBinder;
import android.os.Message;
import android.os.Messenger;
import android.widget.Toast;

public class MyApiService extends Service
{
    static final int API_FN_1   = 101;
    static final int API_FN_2   = 102;

    class MyApiHandler extends Handler
    {
        @Override
        public void handleMessage(Message msg)
        {
            switch (msg.what)
            {
                case API_FN_1:
                    ApiFn1();
                    break;
                case API_FN_2:
                    ApiFn2();
                    break;
                default:
                    super.handleMessage(msg);
                    break;
            }
        }
    }

    final Messenger myMessenger = new Messenger(new MyApiHandler());

    @Override
    public void onCreate()
    {
        Toast.makeText(this, "Service Started", Toast.LENGTH_SHORT).show();
    }

    @Override
    public void onDestroy()
    {
        Toast.makeText(this, "Service Stopped", Toast.LENGTH_SHORT).show();
    }

    @Override
    public IBinder onBind(Intent intent)
    {
        Toast.makeText(this, "Service Bound", Toast.LENGTH_SHORT).show();
        return myMessenger.getBinder();
    }

    private void ApiFn1()
    {
        Toast.makeText(this, "API Function 1", Toast.LENGTH_SHORT).show();
        return;
    }

    private void ApiFn2()
    {
        Toast.makeText(this, "API Function 2", Toast.LENGTH_SHORT).show();
        return;
    }
}

我更改了一些名称,但到目前为止,这确实是我的代码的核心。只是一堆Toast的调用,以便我可以看到事情正在发挥作用。

我在该主题上找到的所有教程都完全忽略了清单。昨晚我试着调整清单选项,认为这可能与它有关,但无济于事。无论如何,为了完整性,以下是活动的清单代码:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="net.william.android.myAPI.tester"
    android:versionCode="1"
    android:versionName="1.0">
    <uses-sdk android:minSdkVersion="10" />
    <application android:icon="@drawable/icon" android:label="@string/app_name">
        <activity android:name=".MyApiTester" android:label="@string/app_name">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
    </application>
</manifest>

和服务:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="net.william.android.myAPI"
    android:versionCode="1"
    android:versionName="1.0">
    <uses-sdk android:minSdkVersion="10" />
    <application android:icon="@drawable/icon" android:label="@string/app_name">
        <service android:name=".MyApiService"></service>
    </application>
</manifest>

我尝试在<service>块中添加以下内容,但似乎没有帮助:

            <intent-filter>
                <action android:name="net.william.android.myAPI.MyApiService" />
            </intent-filter>

无论如何,这是我在你的论坛上的第一篇文章,所以如果信息太多或太少,或者如果我错过了一些非常简单的解决方法,我会在网上的某个地方道歉。我大部分时间都在阅读有关Google所提供服务的所有文档,并在论坛上发帖后阅读,所以,在这一点上,我很容易忽略一些事情。提前感谢任何有时间帮助的人!

1 个答案:

答案 0 :(得分:1)

如果要将这些文件作为两个单独的APK文件分发,如通过一对清单所示,则不能使用将Java类作为参数的Intent构造函数。这甚至编译的事实表明您的项目设置有些可怕 - 从构建时间的角度来看,您的客户端不应位于服务的国家英里范围内。

在您的<intent-filter>广告中添加<service>广告,并在您的客户端中使用该结构来创建它。{/ p>

例如,在这对sample projects中,服务具有:

Intent

,因此客户使用:

<service android:name=".BshService">
    <intent-filter>
        <action android:name="com.commonsware.android.advservice.IScript" />
    </intent-filter>
</service>