Retrofit 2:response.body()为null,但状态码为200

时间:2016-06-28 23:30:25

标签: java android web-services retrofit2

我不确定为什么会发生这种情况,我试图从中获取数据:

season_matches = pd.read_csv(io.StringIO(rv.decode('utf-8')))
season_matches = season_matches.replace(np.nan, None)
season_matches = season_matches.to_dict()

使用以下代码:

TypeError: cannot replace [nan] with method pad on a DataFrame

和班级:

https://api.stackexchange.com/2.2/questions?order=desc&sort=creation&site=stackoverflow&tagged=android

片段:

public interface StackOverflowAPI {

    @GET("/2.2/questions?order=desc&sort=creation&site=stackoverflow")
    Call<StackQuestions> loadQuestions(@Query("tagged") String tags);

}

Gradle依赖项:

public class StackQuestion {

    @SerializedName("title")
    private String title;

    @SerializedName("view_count")
    private int numViews;

    @SerializedName("is_answered")
    private boolean solved;


    public String getTitle() {
        return title;
    }

    public void setTitle(String title) {
        this.title = title;
    }

    public int getNumViews() {
        return numViews;
    }

    public void setNumViews(int numViews) {
        this.numViews = numViews;
    }

    public boolean isSolved() {
        return solved;
    }

    public void setSolved(boolean solved) {
        this.solved = solved;
    }

}


public class StackQuestions {
    List<StackQuestion> questions;
}

执行代码后,我已经登录了logcat,如下:

public class AlphaFragment extends Fragment implements Callback<StackQuestions> {

    private StackOverflowAPI stackOverflowAPI;

    private EditText    etxtQuestionToSearch;
    private Button      btnSearchQuestion;

    private RecyclerView                mRecyclerView;
    private RecyclerView.Adapter        mStackQuestionsAdapter;
    private RecyclerView.LayoutManager  mLayoutManager;

    private ArrayList<StackQuestion> mStackQuestionsList;

    private final static String TAG = "AlphaFragment";

    public AlphaFragment() {}


    // ------------------------ FRAGMENT LIFECYCLE ------------------------ //

    @Override
    public void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        init();
    } // end : onCreate Method

    @Override
    public View onCreateView(LayoutInflater inflater,
                             ViewGroup container,
                             Bundle savedInstanceState) {
        View view = inflater.inflate(R.layout.fragment_alpha, container, false);

        etxtQuestionToSearch = (EditText) view.findViewById(R.id.etxt_stack_topic);
        btnSearchQuestion = (Button) view.findViewById(R.id.button_stack_search);
        mRecyclerView = (RecyclerView) view.findViewById(R.id.rcvw_stackquestions_list);

        // Recycler View SETUP
        mLayoutManager = new LinearLayoutManager(getActivity());
        mRecyclerView.setLayoutManager(mLayoutManager);
        mRecyclerView.setItemAnimator(new DefaultItemAnimator());

        btnSearchQuestion.setOnClickListener(new View.OnClickListener() {
            public void onClick(View v) {
                searchStackQuestion();
            } // end : onClick Method
        });

        return view;
    } // end : onCreateView Method


    // ------------------------- CALLBACK METHODS ------------------------- //

    @Override
    public void onResponse(Call<StackQuestions> call,
                           Response<StackQuestions> response) {
        if (response == null) {
            return;
        }

        List<StackQuestion> resultList = response.body().questions;

        // X:ERROR: .questions == NULL, so Toast appears
        if (resultList == null || resultList.isEmpty()) {
            Toast.makeText(getActivity(), "Results for the query are null/empty", Toast.LENGTH_LONG).show();
            return;
        }

        // Add resulting data from web service
        for (StackQuestion question : resultList) {
            mStackQuestionsList.add(question);
        }

        // Assign data to recycler view
        mStackQuestionsAdapter = new StackQuestionsAdapter(
            mStackQuestionsList,
            getActivity(),
            new RVTapListener() {
                @Override
                public void onItemClick(View v, int position) {
                    Toast.makeText(getActivity(), "Now clicked, but not implemented " + new Date(), Toast.LENGTH_SHORT).show();
                } // end : onItemClick Method
            }
        );
        mRecyclerView.setAdapter(mStackQuestionsAdapter);
    } // end : onResponse Method

    @Override
    public void onFailure(Call<StackQuestions> call,
                          Throwable t) {
        Toast.makeText(getActivity(), t.getLocalizedMessage(), Toast.LENGTH_LONG).show();
    } // end : onFailure Method


    // ------------------------- FRAGMENT METHODS ------------------------- //

    private void init() {
        Gson gson = new GsonBuilder()
            .setDateFormat("yyyy-MM-dd'T'HH:mm:ssZ")
            .create();

        HttpLoggingInterceptor interceptor = new HttpLoggingInterceptor();
        interceptor.setLevel(HttpLoggingInterceptor.Level.BODY);
        OkHttpClient client = new OkHttpClient.Builder().addInterceptor(interceptor).build();

        Retrofit retrofit = new Retrofit.Builder()
            .baseUrl("https://api.stackexchange.com")
            .addConverterFactory(GsonConverterFactory.create(gson))
            .client(client)
            .build();

        stackOverflowAPI = retrofit.create(StackOverflowAPI.class);
    } // end : init Method

    private void searchStackQuestion() {
        mStackQuestionsList = new ArrayList<>();

        Call<StackQuestions> call = stackOverflowAPI.loadQuestions("android");
        call.enqueue(this);
    } // end : searchStackQuestion Method

}    

一切都很顺利,但我在AlphaFragment的这一部分得到了NullPointerException:

compile 'com.google.code.gson:gson:2.7'
compile 'com.android.support:design:24.0.0'
compile 'com.squareup.retrofit2:retrofit:2.1.0'
compile 'com.android.support:cardview-v7:24.0.0'
compile 'com.android.support:appcompat-v7:24.0.0'
compile 'com.android.support:recyclerview-v7:24.0.0'
compile 'com.squareup.retrofit2:converter-gson:2.1.0'
compile 'com.squareup.okhttp3:logging-interceptor:3.3.1'

可能是什么问题?我有D/OkHttp: <-- 200 OK https://api.stackexchange.com/2.2/questions?order=desc&sort=creation&site=stackoverflow&tagged=android (1136ms) D/OkHttp: Cache-Control: private D/OkHttp: Content-Type: application/json; charset=utf-8 D/OkHttp: Access-Control-Allow-Origin: * D/OkHttp: Access-Control-Allow-Methods: GET, POST D/OkHttp: Access-Control-Allow-Credentials: false D/OkHttp: X-Content-Type-Options: nosniff D/OkHttp: Date: Tue, 28 Jun 2016 22:56:26 GMT D/OkHttp: {"items":[{"tags":["android"],"owner":{"reputation":1,"user_id" ... D/OkHttp: er_id":1163471,"user_type":"registered","accept_rate":93,"profil ... D/OkHttp: e?type=large","display_name":"Vahid Zadeh","link":"http://stacko ... D/OkHttp: n_id":38086882,"link":"stackoverflow.com/questions/380868 ... D/OkHttp: ","animation-listener"],"owner":{"reputation":23,"user_id":43462 ... D/OkHttp: <-- END HTTP (19295-byte body) List<StackQuestion> resultList = response.body().questions; // X:ERROR: .questions == NULL, so Toast appears if (resultList == null || resultList.isEmpty()) { Toast.makeText(getActivity(), "Results for the query are null/empty", Toast.LENGTH_LONG).show(); return; } ,那里有JSON信息;但是,在尝试检索它之后,会出现status code

提前致谢...

2 个答案:

答案 0 :(得分:1)

我认为您的模型中存在错误。

尝试:

public class StackQuestions {
    @SerializedName("items")
    List<StackQuestion> questions;
}

答案 1 :(得分:0)

Retrofit默认使用gson来解析json,所以要成功解析json,你需要在同一个层次结构中定义模型。

我想这样的事情会起作用:

ReactDOM.render( 
   React.createElement('div', null, [...yourComponentsHere]),
   document.getElementById('container')
)