我目前正在为我的武术组织制作一个应用程序,并且在应用程序中我希望它显示一个地图,其中的图钉显示了我所在州的每个武术道具的位置。我正在使用片段进行此操作,就像您在应用程序中从抽屉切换器访问地图一样。我的LocationFragment
扩展了SupportMapFragment
并实施了OnMapReadyCallback
。但是,当我打开位置选项卡时,地图开始加载然后崩溃。这是Android Monitor出现的错误:
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)
我知道错误与我的PinsApiInterface.getStreams有关,我通过Retrofit2访问,但我不知道如何解决问题。我跟随着Tutsplus的人们以及他们如何在他们的动物园应用程序中实现了一个地图(我正在查找的代码是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 );
}
答案 0 :(得分:2)
看起来你正在改造改装2。 RestAdapter
是改造1,但Callback
onResponse
和onError
是改造2。
首先,查看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
}
});