42

我正在创建一个使用 EventBus 将异步广播发布到其他类的 Android 应用程序,但在执行过程中我遇到了错误。

MainActivity.java

import android.content.Intent;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import com.google.android.gms.maps.model.LatLng;
import org.greenrobot.eventbus.EventBus;
import org.greenrobot.eventbus.Subscribe;
import org.greenrobot.eventbus.ThreadMode;


public class MainActivity extends AppCompatActivity {

    //Globals
    public String uname = null;
    public double lat = 0;
    public double lng = 0;

    //Get GUI handles
    public Button sendButton; //
    public EditText username;
    public Button MapButton; //
    public EditText LatBox;
    public EditText LngBox;


    protected void onDestroy() {
        super.onDestroy();
        EventBus.getDefault().unregister(this);
    }

    @Override
    protected void onCreate(Bundle savedInstanceState) {

        //register EventBus
        EventBus.getDefault().register(this);

        super.onCreate(savedInstanceState);
        //set GUI for MainActivity
        setContentView(R.layout.activity_main);

        //get handlers
        LatBox = (EditText)findViewById(R.id.LatBox);
        LngBox = (EditText)findViewById(R.id.LngBox);

        MapButton = (Button)findViewById(R.id.locationButton);
        //Call the class which will handle finding coordinates
        MapButton.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Intent MapIntent = new Intent(getApplicationContext(), MapClass.class);
                startActivityForResult(MapIntent, 0);
            }
        });

        sendButton = (Button)findViewById(R.id.Submit);
        //Set action for Button
        sendButton.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {

                //Get username from user
                username  = (EditText)findViewById(R.id.UsernameText);
                uname = username.getText().toString();

                //Generate intent to start IntentService
                Intent i = new Intent(getApplicationContext(), Register.class);

                //Put the extra field of username
                i.putExtra("username", uname);
                i.putExtra("latitude", lat);
                i.putExtra("longitude", lng);
                i.putExtra("type", "meetup.be2015.gcm_meetup.MAIN_ACTIVITY");

                //Start the IntentService on a different thread
                startService(i);
            }
        });

    }


    @Subscribe(threadMode = ThreadMode.MAIN)
    public void onEvent(LatLng currentPos){

        LatBox.setText(String.valueOf(currentPos.latitude));
        LngBox.setText(String.valueOf(currentPos.longitude));

        lat = currentPos.latitude;
        lng = currentPos.longitude;
    }
}

MapClass.java

import android.app.IntentService;
import android.content.Intent;
import android.location.Location;
import android.net.Uri;
import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.util.Log;
import com.google.android.gms.appindexing.Action;
import com.google.android.gms.appindexing.AppIndex;
import com.google.android.gms.common.ConnectionResult;
import com.google.android.gms.common.api.GoogleApiClient;
import com.google.android.gms.location.LocationServices;
import com.google.android.gms.maps.GoogleMap;
import com.google.android.gms.maps.OnMapReadyCallback;
import com.google.android.gms.maps.model.LatLng;
import org.greenrobot.eventbus.EventBus;
import org.greenrobot.eventbus.Subscribe;

public class MapClass extends AppCompatActivity implements OnMapReadyCallback, GoogleApiClient.ConnectionCallbacks, GoogleApiClient.OnConnectionFailedListener {

    private GoogleApiClient mGoogleApiClient;
    private GoogleMap mgoogleMap;
    private LatLng latLng;
    private GoogleApiClient client;

    @Override
    public void onMapReady(GoogleMap googleMap) {
        mgoogleMap = googleMap;
        mgoogleMap.setMyLocationEnabled(true);      //Sets location to current position
        buildGoogleApiClient();
        mGoogleApiClient.connect();
    }

    protected synchronized void buildGoogleApiClient() {
        mGoogleApiClient = new GoogleApiClient.Builder(this)
                .addConnectionCallbacks(this)
                .addOnConnectionFailedListener(this)
                .addApi(LocationServices.API)
                .build();
    }

    @Override
    public void onDestroy() {
        super.onDestroy();
        if (EventBus.getDefault().isRegistered(this)) {
            EventBus.getDefault().unregister(this);
        }
    }

    @Override
    public void onConnected(Bundle bundle) {
        Location MLastLocation = LocationServices.FusedLocationApi.getLastLocation(mGoogleApiClient);
        if (MLastLocation != null) {
            latLng = new LatLng(MLastLocation.getLatitude(), MLastLocation.getLongitude());

            //Post the LatLng to MainActivity
            EventBus.getDefault().post(latLng);

            //Send sticky event to Register and MyGcmListenerService
            EventBus.getDefault().postSticky(latLng);

        } else {
            Log.d("onConnected", "Value of LatLng is NULL");
            latLng = new LatLng(0, 0);   //equator
        }
    }

    @Override
    public void onConnectionSuspended(int i) {
        //Notify
        Log.d("ConnectionSuspended", "Connection Suspended. Status: " +   i);
        mgoogleMap.clear();
        mGoogleApiClient.disconnect();
    }

    @Override
    public void onConnectionFailed(ConnectionResult connectionResult) {
        //Notify
        Log.d("ConnectionFailed", "Connection Failed. Status: " + connectionResult.toString());
        mgoogleMap.clear();
        mGoogleApiClient.disconnect();
    }

    @Subscribe
    public void onEvent() {
        Log.d("EVENT", "EVENT");
    }

    @Override
    public void onStart() {
        super.onStart();
        if (!EventBus.getDefault().isRegistered(this)) {
            EventBus.getDefault().register(this);
        }


    @Override
    public void onStop() {
        super.onStop();
        if (EventBus.getDefault().isRegistered(this)) {
            EventBus.getDefault().unregister(this);
        }

    }
}

LogCat 显示以下内容:

03-08 22:54:56.970 8570-8570/meetup.be2015.gcm_meetup E/AndroidRuntime: FATAL EXCEPTION: main
java.lang.RuntimeException: Unable to start activity ComponentInfo{meetup.be2015.gcm_meetup/meetup.be2015.gcm_meetup.MapClass}:
org.greenrobot.eventbus.EventBusException: Subscriber class meetup.be2015.gcm_meetup.MapClass 
and its super classes have no public methods with the @Subscribe annotation
at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2118)
at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2143)
at android.app.ActivityThread.access$700(ActivityThread.java:140)
at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1237)
at android.os.Handler.dispatchMessage(Handler.java:99)
at android.os.Looper.loop(Looper.java:174)
at android.app.ActivityThread.main(ActivityThread.java:4952)
at java.lang.reflect.Method.invokeNative(Native Method)
at java.lang.reflect.Method.invoke(Method.java:511)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1027)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:794)
at dalvik.system.NativeStart.main(Native Method)
Caused by: org.greenrobot.eventbus.EventBusException: Subscriber class meetup.be2015.gcm_meetup.MapClass 
and its super classes have no public methods with the @Subscribe annotation
at org.greenrobot.eventbus.SubscriberMethodFinder.findSubscriberMethods(SubscriberMethodFinder.java:67)
at org.greenrobot.eventbus.EventBus.register(EventBus.java:136)
at meetup.be2015.gcm_meetup.MapClass.onStart(MapClass.java:91)
at android.app.Instrumentation.callActivityOnStart(Instrumentation.java:1178)
at android.app.Activity.performStart(Activity.java:5198)
at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2091)
at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2143) 
at android.app.ActivityThread.access$700(ActivityThread.java:140) 
at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1237) 
at android.os.Handler.dispatchMessage(Handler.java:99) 
at android.os.Looper.loop(Looper.java:174) 
at android.app.ActivityThread.main(ActivityThread.java:4952) 
at java.lang.reflect.Method.invokeNative(Native Method) 
at java.lang.reflect.Method.invoke(Method.java:511) 
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1027) 
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:794) 
at dalvik.system.NativeStart.main(Native Method) 

为什么会这样?难道我做错了什么?

4

15 回答 15

149

如果您在构建中使用 proguard,请确保这些行在您的 proguard 配置文件中。

-keepattributes *Annotation*
-keepclassmembers class ** {
    @org.greenrobot.eventbus.Subscribe <methods>;
}
-keep enum org.greenrobot.eventbus.ThreadMode { *; }
于 2016-09-08T09:39:29.860 回答
20

我认为这是因为 MapClass.java 中的 onEvent 没有参数。您可以尝试使用预期的参数吗?

于 2016-03-10T09:33:09.580 回答
16

我遇到了同样的问题,经过长时间的研究,每个案例都得到了解决方案。此问题是由于您尝试将事件总线注册为的类中缺少 @Subscribe 公共方法 onEvent() EventBus.getDefault().register(this)。如果您使用 Event bus 注册一个类,则此功能的存在是强制性的

这可能有两种情况

  1. 使用 progruad :progruad 可能会修改 onEvent() 方法的名称,因为事件总线无法找到它。将这些行放入您的 progruad 规则中

    -keepattributes注解

    -keepclassmembers 类 ** {

    @org.greenrobot.eventbus.Subscribe ;

    }

    -保持枚举 org.greenrobot.eventbus.ThreadMode { *;

}

  1. 如果您没有使用 progruard,那么您的类肯定缺少带有 @Subscribe 注释的 onEvent() 方法。对于 EventBus 版本 3.0.0,此方法注释是强制性的,因此请仔细检查您的类中是否存在此方法。
于 2017-05-10T11:17:55.077 回答
11

以防万一你的代码和我的一样:p

我不得不将方法设置为public因为它当前是private.

于 2017-06-24T12:20:49.553 回答
10

ProGuard

ProGuard 会混淆方法名称,并且可能会删除未被调用的方法(死代码删除)。因为没有直接调用订阅者方法,所以 ProGuard 假定它们未被使用。因此,如果您启用 ProGuard 缩小,您必须告诉 ProGuard 保留这些订阅者方法。

在您的 ProGuard 配置文件 (proguard.cfg) 中使用以下规则来防止订阅者被删除:

-keepattributes *Annotation*
-keepclassmembers class * {
   @org.greenrobot.eventbus.Subscribe <methods>;
}
-keep enum org.greenrobot.eventbus.ThreadMode { *; }
 
# Only required if you usenter code heree AsyncExecutor
-keepclassmembers class * extends org.greenrobot.eventbus.util.ThrowableFailureEvent {
    <init>(java.lang.Throwable);
}
于 2019-04-29T07:21:31.850 回答
5

在我的情况下,我收到了这个错误,因为我没有在我注册 EventBus 的类上写 @Subscribe。

于 2016-10-21T03:14:12.947 回答
5

在我的情况下onEvent()私有的并被放置在child class中。

但是register()并在父类unregister()中被调用。

解决办法是onEvent() 公开

于 2018-10-11T11:03:36.067 回答
4

如果您使用 proguard,您将不会在调试模式下遇到此问题。我在发布版本中遇到了这个问题。在 proguard-rules.pro 文件中添加以下代码后,解决了我的问题。

-keepattributes *Annotation*
-keepclassmembers class ** {
    @org.greenrobot.eventbus.Subscribe <methods>;
 }
-keep enum org.greenrobot.eventbus.ThreadMode { *; }
于 2019-08-17T06:12:36.583 回答
3

在旁注中,从Google 的EventBus实现切换到这个后,我遇到了同样的错误。这个错误让我发疯了,因为 Google 的 EventBus 也有一个 @Subscribe 注释,而我使用的是那个而不是 greenrobot 提供的那个。

在此处输入图像描述

好吧,这对我来说是一个非常愚蠢的错误,但如果我能帮助像我这样的人,我很高兴。

于 2018-11-05T22:19:02.220 回答
3

可以帮助某人!我的情况,我忘了在下面添加一行buildTypes

proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'

于 2020-05-13T03:23:43.393 回答
2

以防万一它会对某人有所帮助,在我的情况下,我忘记将参数传递给接收方法,其他一切都很好。当没有参数传递给接收函数/方法时,在这种情况下会抛出此异常。

于 2018-12-13T09:48:34.063 回答
2

只需从私人娱乐切换到公共娱乐,您就会看到魔法。

于 2020-12-09T20:36:54.097 回答
0

从答案更新,因为 proguard 不再是默认值。它变成了R8。关闭 R8 以确保。在颤动中,这将是:

颤振构建 apk --no-shrink ...

于 2020-11-06T14:45:15.207 回答
0

1.转到导入并删除此行:
import com.google.common.eventbus.Subscribe;
2.并添加:
import org.greenrobot.eventbus.Subscribe;
3.删除@Subscribe并再次添加。
4.添加时确保您使用org.greenrobot.eventbus导入@Subscribe。

于 2022-01-12T06:15:18.420 回答
-2

我正在使用来自错误包的订阅注释。

它应该是 import org.greenrobot.eventbus.Subscribe

我正在使用 import com.squareup.otto.Subscribe

于 2020-02-14T07:22:13.600 回答