我目前正在为我的武术组织开发一个应用程序,在该应用程序中,我希望它显示一个带有大头针的地图,以显示每个武术道场在我所在的州的位置。我正在使用片段进行此操作,因为在应用程序中,您可以通过抽屉式开关访问地图。我的LocationFragment
扩展了SupportMapFragment
并实现了OnMapReadyCallback
。但是,当我打开“位置”标签时,地图开始加载,然后崩溃。这是通过Android监视器出现的错误:
java.lang.IllegalArgumentException: PinsApiInterface.getStreams: Must have either a return type or Callback as last argument.
at retrofit.RestMethodInfo.methodError(RestMethodInfo.java:107)
at retrofit.RestMethodInfo.parseResponseType(RestMethodInfo.java:270)
at retrofit.RestMethodInfo.<init>(RestMethodInfo.java:97)
at retrofit.RestAdapter.getMethodInfo(RestAdapter.java:213)
at retrofit.RestAdapter$RestHandler.invoke(RestAdapter.java:236)
at java.lang.reflect.Proxy.invoke(Proxy.java:393)
at $Proxy0.getStreams(Unknown Source)
at com.mycompany.kfcomapplication.fragments.LocationFragment.onMapReady(LocationFragment.java:85)
at com.google.android.gms.maps.SupportMapFragment$zza$1.zza(Unknown Source)
at com.google.android.gms.maps.internal.zzt$zza.onTransact(Unknown Source)
at android.os.Binder.transact(Binder.java:387)
at zu.a(:com.google.android.gms.DynamiteModulesB:82)
at maps.ad.t$5.run(Unknown Source)
at android.os.Handler.handleCallback(Handler.java:739)
at android.os.Handler.dispatchMessage(Handler.java:95)
at android.os.Looper.loop(Looper.java:158)
at android.app.ActivityThread.main(ActivityThread.java:7224)
at java.lang.reflect.Method.invoke(Native Method)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1230)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1120)
我知道该错误与我通过Retrofit2访问的PinsApiInterface.getStreams有关,但是我不确定如何解决此问题。我一直在跟Tutsplus的家伙们一起学习,以及他们如何在他们的Zoo应用程序中实现地图(我正在查看的代码位于here中)。在他们的教程中,他们使用的是Retrofit的旧版本,在
public void success()
下使用public void failure()
和PinsApiInterface.getStreams()
作为替代。对于Retrofit2,已将其更改为onResponse()
和onFailure()
。我已经搜索了多个不同的线程来解决此崩溃问题,但是我没有任何运气。我希望我可以在如何解决此问题上获得一些帮助。我是在Android中使用Retrofit2的新手,所以我们将不胜感激。以下是我的LocationFragment.java
和我的PinsApiInterface.java
的代码。这是
LocationFragment.java
:public class LocationFragment extends SupportMapFragment implements OnMapReadyCallback {
public static LocationFragment getInstance() {
LocationFragment fragment = new LocationFragment();
return fragment;
}
@Override
public void onViewCreated(View view, @Nullable Bundle savedInstanceState) {
super.onViewCreated(view, savedInstanceState);
getMapAsync(this);
}
@Override
public void onMapReady(final GoogleMap googleMap) {
CameraPosition position = CameraPosition.builder()
.target(new LatLng(42.921966, -85.718533))
.zoom( 16f )
.bearing( 0.0f )
.tilt( 0.0f )
.build();
googleMap.animateCamera(CameraUpdateFactory.newCameraPosition(position), null);
googleMap.setMapType(GoogleMap.MAP_TYPE_HYBRID);
googleMap.setTrafficEnabled(true);
googleMap.getUiSettings().setZoomControlsEnabled(true);
MarkerOptions options = new MarkerOptions().position(new LatLng(42.921966, -85.718533));
options.title("KFCOM");
options.icon(BitmapDescriptorFactory.defaultMarker(BitmapDescriptorFactory.HUE_BLUE));
googleMap.addMarker(options);
googleMap.setOnMarkerClickListener(new GoogleMap.OnMarkerClickListener() {
@Override
public boolean onMarkerClick(Marker marker) {
marker.showInfoWindow();
return true;
}
});
RestAdapter adapter = new RestAdapter.Builder()
.setEndpoint("https://gist.githubusercontent.com/anonymous/101ffbf7e1aed60b7caf7d3d5418bfde/raw/43b5d1fa6862fd1dce84044821cdf1a9d48b6ca2")
.build();
PinsApiInterface pinsApiInterface = adapter.create(PinsApiInterface.class );
//This line is what is causing the crash.
pinsApiInterface.getStreams(new Callback<List<Pin>>() {
@Override
public void onResponse(Call<List<Pin>> pins, Response<List<Pin>> response) {
if(!isAdded() || pins == null || pins.equals("") )
{
return;
}
for( Pin pin : response.body()) {
MarkerOptions options = new MarkerOptions().position( new LatLng(pin.getLatitude(), pin.getLongitude()));
options.title(pin.getName());
options.icon(BitmapDescriptorFactory.defaultMarker(BitmapDescriptorFactory.HUE_GREEN));
googleMap.addMarker(options);
}
}
@Override
public void onFailure(Call<List<Pin>> call, Throwable t) {
}
});
}
}
这是
PinsApiInterface.java
:public interface PinsApiInterface {
@GET( "/Pins.json" )
void getStreams(Callback<List<Pin>> callback );
}
最佳答案
看起来您正在将改造和改造2混合在一起。 RestAdapter
是Retrofit 1,而Callback
与onResponse
和onError
则是Retrofit2。
首先,查看您的gradle文件中的依赖项,您将需要以下内容-
compile 'com.squareup.retrofit2:retrofit:2.1.0'
compile 'com.squareup.retrofit2:converter-gson:2.1.0'
并删除任何形式的库-'com.squareup.retrofit ...'。那就是引入旧的库,这也是混合使用两者时没有出现编译时错误的部分原因。
对于您的服务,不再指定回调,它具有一个基于
Call
的新接口。 Call
允许您在发出请求时而不是在定义请求时选择同步还是异步。 -public interface PinsApiInterface {
@GET( "/Pins.json" )
Call<List<Pin>> getStreams();
}
替换此旧的改装代码-
RestAdapter adapter = new RestAdapter.Builder()
.setEndpoint("https://gist.githubusercontent.com/anonymous/101ffbf7e1aed60b7caf7d3d5418bfde/raw/43b5d1fa6862fd1dce84044821cdf1a9d48b6ca2")
.build();
PinsApiInterface pinsApiInterface = adapter.create(PinsApiInterface.class );
借助构建服务接口的新方法-
Retrofit retrofit = new Retrofit.Builder()
.baseUrl("https://gist.githubusercontent.com/anonymous/101ffbf7e1aed60b7caf7d3d5418bfde/raw/43b5d1fa6862fd1dce84044821cdf1a9d48b6ca2")
.addConverterFactory(GsonConverterFactory.create())
.build();
PinsApiInterface pinsApiInterface = retrofit.create(PinsApiInterface.class);
最后,进行异步调用的新方法-
Call<List<Pin>> myCall = pinsApiInterface.getStreams();
myCall.enqueue(new Callback<List<Pin>>() {
@Override
public void onResponse(Call<List<Pin>> call, Response<List<Pin>> response) {
// your response code
}
@Override
public void onFailure(Call<List<Pin>> call, Throwable t) {
// your failure code
}
});