Retrofit2 не отправляет данные POST

Извините, если мой английский плохой! Я использую модификацию 2 для получения данных, зависящих от данных POST. Я получаю данные с сервера, но у меня проблема с отправкой данных. Я пытался использовать разные аннотации (@Field, @Body с Object, @Body с данными HashMap), но каждая из них не работала. Вот мой код Java:

build.gradle

apply plugin: 'com.android.application'

android {
    compileSdkVersion 28
    defaultConfig {
        applicationId "com.test.retrofitpostdatagetjson"
        minSdkVersion 19
        targetSdkVersion 28
        versionCode 1
        versionName "1.0"
        testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
    }
    buildTypes {
        release {
            minifyEnabled false
            proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
        }
    }
}

dependencies {
    implementation fileTree(dir: 'libs', include: ['*.jar'])
    implementation 'com.android.support:appcompat-v7:28.0.0'
    implementation 'com.android.support.constraint:constraint-layout:1.1.3'

    implementation 'com.google.code.gson:gson:2.8.4'

    implementation 'com.squareup.retrofit2:retrofit:2.5.0'

    implementation 'com.squareup.retrofit2:converter-gson:2.5.0'

    testImplementation 'junit:junit:4.12'
    androidTestImplementation 'com.android.support.test:runner:1.0.2'
    androidTestImplementation 'com.android.support.test.espresso:espresso-core:3.0.2'
}

APIClient.java

import retrofit2.Retrofit;
import retrofit2.converter.gson.GsonConverterFactory;

public class APIClient {
    private static final String BASE_URL = "https://myurl.ru/";
    private static Retrofit retrofit;

    public static Retrofit getClient() {
        if (retrofit == null) {
            retrofit = new Retrofit.Builder()
                    .baseUrl(BASE_URL)
                    .addConverterFactory(GsonConverterFactory.create())
                    .build();
        }
        return retrofit;
    }
}

ApiUtils.java

public class ApiUtils {

    private ApiUtils() {}

    public static APIInterface getAPIService() {

        return APIClient.getClient().create(APIInterface.class);
    }
}

APIИнтерфейс.java

import com.test.retrofitpostdatagetjson.model.DataResponse;
import retrofit2.Call;
import retrofit2.http.Body;
import retrofit2.http.POST;

public interface APIInterface {
    @POST("test_json_with_post")
    Call<DataResponse> createData(@Body DataResponse data);

}

DataResponse.java

import com.google.gson.annotations.Expose;
import com.google.gson.annotations.SerializedName;
import java.io.Serializable;

public class DataResponse implements Serializable {

    @SerializedName("numb")
    @Expose
    private int numb;
    @SerializedName("name")
    @Expose
    private String name;

    @SerializedName("emp")
    @Expose
    public String value;
    @SerializedName("status")
    @Expose
    public String status;

    public DataResponse(int numb, String name) {
        this.numb = numb;
        this.name = name;
    }

    public int getNumb() {
        return numb;
    }

    public String getName() {
        return name;
    }

    public String getValue() {
        return value;
    }

    public String getStatus() {
        return status;
    }

    public void setNumb(int numb) {
        this.numb = numb;
    }

    public void setName(String name) {
        this.name = name;
    }

    public void setValue(String value) {
        this.value = value;
    }

    public void setStatus(String status) {
        this.status = status;
    }

    @Override
    public String toString() {
        return "DataResponse{" +
                "numb = " + numb +
                ", name='" + name + '\'' +
                ", value='" + value + '\'' +
                ", status='" + status + '\'' +
                '}';
    }
}

MainActivity.java

import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.util.Log;
import android.widget.Toast;
import retrofit2.Call;
import retrofit2.Callback;
import retrofit2.Response;
import com.test.retrofitpostdatagetjson.api.APIInterface;
import com.test.retrofitpostdatagetjson.model.DataResponse;
import static com.test.retrofitpostdatagetjson.api.ApiUtils.getAPIService;

public class MainActivity extends AppCompatActivity {

    APIInterface apiInterface;
    int numb = 0;
    String name = "test row";

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        apiInterface = getAPIService();

        if (CommonMethod.isNetworkAvailable(MainActivity.this))
            sendPost(numb, name);
        else
            CommonMethod.showAlert("Internet Connectivity Failure", MainActivity.this);

    }

    private void sendPost(int numb, String name) {

        final DataResponse data = new DataResponse(numb, name);
        Log.w("retroTest", "sent    -->  " + data.toString());
        Call<DataResponse> call1 = apiInterface.createData(data);
        call1.enqueue(new Callback<DataResponse>() {
            @Override
            public void onResponse(Call<DataResponse> call, Response<DataResponse> response) {
                DataResponse dataResponse = response.body();
                if (dataResponse != null) {
                    Log.w("retroTest", "received    -->  " + dataResponse.toString());
                }
            }

            @Override
            public void onFailure(Call<DataResponse> call, Throwable t) {
                Toast.makeText(getApplicationContext(), "onFailure called ", Toast.LENGTH_SHORT).show();
                call.cancel();
            }
        });
    }

}

Вот мои журналы:

W/retroTest: sent  -->  DataResponse{numb=0, name='test row', value='null', status='null'}
W/retroTest: received  -->  DataResponse{numb=0, name='null', value='null', status='2'}

мой файл PHP на сервере:

<?php
header("Content-type: application/json; charset=utf-8");

$inputJSON = file_get_contents('php://input');
$input = json_decode($inputJSON, TRUE);

$numb = $input['numb'];
$name = $input['name'];

if (isset($_POST['numb']) && !empty($_POST['numb'])) {   
  $numb = $_POST['numb']; }
if (isset($_POST['name']) && !empty($_POST['name'])) {   
  $name = $_POST['name']; }

if (!$numb) {
$json = json_encode( array(
"emp" => $name,
"status" => "2"));
} else {
$json = json_encode( array(
"emp" => "nothing",
"status" => "2"));
}

$myfile = fopen("testfile.txt", "w");
fwrite($myfile, $inputJSON);
fwrite($myfile, "\n");
fwrite($myfile, $numb);
fwrite($myfile, "\n");
fwrite($myfile, $name);
fclose($myfile);


echo $json;
?>

Файл testfile.txt тоже пустой, но при попытке отправить POST через Postman все работает! Retrofit2 не отправляет данные POSTRetrofit2 не отправляет данные POST

Является ли URL-адрес в почтальоне таким же, как «/test_json_with_post»? Кроме того, я не вижу кода, написанного для журнала "отправлено -->"

uneq95 30.01.2019 13:02

Возможный дубликат Как сделать запрос POST с помощью модификации 2?

Prim 30.01.2019 13:04

uneq95, да url в почтальоне такой же, также я получаю данные с него в своем приложении, проблема только в отправке данных из приложения. О журнале «отправлено ->» - извините, я изменил эти строки после того, как вставил код MainActivity в свой вопрос. Исправлено сейчас.

Maria 30.01.2019 13:52

Прим, ответы на этот вопрос мне не помогли, возможно причина моей проблемы в моем коде..

Maria 30.01.2019 14:01
Пользовательский скаляр GraphQL
Пользовательский скаляр GraphQL
Листовые узлы системы типов GraphQL называются скалярами. Достигнув скалярного типа, невозможно спуститься дальше по иерархии типов. Скалярный тип...
Как вычислять биты и понимать побитовые операторы в Java - объяснение с примерами
Как вычислять биты и понимать побитовые операторы в Java - объяснение с примерами
В компьютерном программировании биты играют важнейшую роль в представлении и манипулировании данными на двоичном уровне. Побитовые операции...
Поднятие тревоги для долго выполняющихся методов в Spring Boot
Поднятие тревоги для долго выполняющихся методов в Spring Boot
Приходилось ли вам сталкиваться с требованиями, в которых вас могли попросить поднять тревогу или выдать ошибку, когда метод Java занимает больше...
Полный курс Java для разработчиков веб-сайтов и приложений
Полный курс Java для разработчиков веб-сайтов и приложений
Получите сертификат Java Web и Application Developer, используя наш курс.
0
4
2 017
2

Ответы 2

Вы пытались отправить тело как JsonObject? Всегда работает для меня. Если нет, то я бы предположил, что что-то не так с серверным портом.

Что-то вроде этого:

public interface APIInterface {
    @POST("test_json_with_post")
    Call<DataResponse> createData(@Body JsonObject data);

}
JsonObject obj = new Gson().toJson(yourDataObject);
...
apiInterface.createData(obj);

Я не мог сделать именно так, как вы написали, потому что Android Studio сообщила мне «несовместимые типы. Требуется: com.google.gson.JsonObject Found: java.lang.String». Я делаю это: JSONObject obj = new JSONObject(new Gson().toJson(data)); Но это все еще не решило мою проблему

Maria 30.01.2019 14:09

@Maria JSONObject и JsonObject — это два разных объекта. Во-первых, это реализация библиотеки Java по умолчанию, во-вторых, реализация библиотеки Gson. Вероятно, вам следует использовать второй. Проверьте свой импорт и избавьтесь от JSONObject :)

Yury Dombaev 30.01.2019 14:27

Юрий Домбаев, я так понимаю, что JSONObject и JsonObject это два разных объекта. Как вы видите в моем комментарии, я пытаюсь использовать com.google.gson.JsonObject и получаю ошибку (несовместимые типы), потому что метод toJson(Object src) возвращает String.. И я не понимаю, как правильно это исправить)

Maria 30.01.2019 14:38

@ Мария Ой, извините, кажется, я дал вам неправильный метод. Можешь попробовать этот? JsonElement jsonElement = gson.toJsonTree(javaObject); JsonObject jsonObject = (JsonObject) jsonElement;

Yury Dombaev 30.01.2019 15:28

Я узнал причину своей проблемы.. Это так просто.. Все дело в косой черте в конце URL. Я использую index.php, поэтому, когда я отправляю POST на свой URL-адрес, я должен заканчивать его косой чертой.

Вместо этого:

 @POST("test_json_with_post")

Я должен написать это:

 @POST("test_json_with_post/")

в моем APIInterface.java. И все работает!

Надеюсь, кому-нибудь когда-нибудь поможет)

Другие вопросы по теме