当我们向服务器post数据时,表单 or JSON

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了当我们向服务器post数据时,表单 or JSON相关的知识,希望对你有一定的参考价值。

参考技术A Web前端开发中,以post方式向服务器发送请求的操作再平常不过。发送post请求时,若需要传递数据,数据是包含在消息主体中的。

传递数据的方式有两种:表单和json。使用表单时,数据类型是普通的文本数据,或文件(大型二进制数据或者包含非ASCII字符的数据)。而json是指序列化后的 JSON 字符串。

对于表单方式,消息主体的编码方式有三种:

x-www-form-urlencoded 这种方式只能传输普通文本数据,后来为了提供文件上传的功能,提出了 form-data 方式,可以实现文本数据和文件混合传输,所谓 multipart 。

对于json而言,消息主体的编码方式是: application/json

消息主体的编码方式通过 Content-Type 字段标明。有时候前端传递具体的 Content-Type 可能会导致请求失败,因为有可能后端设计接口的时候,预先设定了传递的数据类型,它也会使用对应的解码方式。如果前端对此不知情,使用了错误的编码格式,反而会导致请求失败。这个时候去掉 Content-Type 设置,问题可能就解决了。

参考资料:

使用 POST 方法和 HttpURLConnection 从 Android 向 PHP 服务器发送 JSON 对象

【中文标题】使用 POST 方法和 HttpURLConnection 从 Android 向 PHP 服务器发送 JSON 对象【英文标题】:Send a JSON Object from Android to PHP server with POST method and HttpURLConnection 【发布时间】:2015-11-13 21:16:08 【问题描述】:

我正在尝试在我的 Android 应用和本地网络上的 WampServer 之间建立通信。

当我想从服务器读取数据时,我已经成功了,但是当我尝试向服务器发送数据时出现问题。

我正在使用服务来建立通信:

public class SynchronisationService extends Service 
@Override
public IBinder onBind(Intent intent) 
    return null;


@Override
public int onStartCommand(Intent intent, int flags, int startId) 
    super.onStartCommand(intent, flags, startId);

    new Thread(new Runnable() 
        @Override
        public void run() 
            try 
                URL url = new URL("http://192.168.37.23/happiness_barometer/php_input.php");
                HttpURLConnection connection = (HttpURLConnection) url.openConnection();
                connection.setDoOutput(true);
                connection.setDoInput(false);
                connection.setRequestMethod("POST");
                connection.connect();
                OutputStream outputStream = connection.getOutputStream();
                OutputStreamWriter writer = new OutputStreamWriter(outputStream);
                JSONObject jsonObject = new JSONObject();
                jsonObject.put("rate", 1);
                writer.write(URLEncoder.encode(jsonObject.toString(), "UTF-8"));
                writer.flush();
                writer.close();

             catch (Exception e) 
                Log.v("EXCEPTION", e.getMessage());
            
        
    ).start();


    stopSelf();

    return flags;

还有我的 php 文件:

 <?php 

    try 
    
        $bdd = new PDO('mysql:host=localhost;dbname=happiness_barometer;charset=utf8', 'utilisateur', '');
     catch (Exception $e) 
    
        die('Erreur : '.$e->getMessage());
    

    $sql = $bdd->prepare(
    'INSERT INTO rates (rate, comment, category, day, month, year, hour, minute, day_of_week, week, rate_number) 
    VALUES (:rate, :comment, :category, :day, :month, :year, :hour, :minute, :day_of_week, :week, :rate_number)');
    if (!empty($_POST['rate'])) 
        $sql->execute(array(
            'rate' => $_POST['rate'],
            'comment' => '',
            'category' => 'pro',
            'day' => 19,
            'month' => 8,
            'year' => 2015,
            'hour' => 18,
            'minute' => 3,
            'day_of_week' =>3,
            'week' => 33,
            'rate_number' => 2));
    
?>

当我运行我的应用程序时,我的数据库中没有添加任何内容。 我觉得$_POST['rate']里面什么都没有。

请告诉我我的代码有什么问题?

【问题讨论】:

你能发布你的堆栈跟踪(logcat) @vinay Maneti :我的 logcat 中有大约 200 行。哪些行很重要? 与你的 SynchronisationService 和活动类相关的 没有与 SynchronisationService 相关的行...(对不起,我只是使用 logcat 来查看异常发生时) 【参考方案1】:

最后,我成功地将 JSONObject 发送到我的服务器。

我将此代码用于 android 部分:

new Thread(new Runnable() 
        @Override
        public void run() 
            OutputStream os = null;
            InputStream is = null;
            HttpURLConnection conn = null;
            try 
                //constants
                URL url = new URL("http://192.168.43.64/happiness_barometer/php_input.php");
                JSONObject jsonObject = new JSONObject();
                jsonObject.put("rate", "1");
                jsonObject.put("comment", "OK");
                jsonObject.put("category", "pro");
                jsonObject.put("day", "19");
                jsonObject.put("month", "8");
                jsonObject.put("year", "2015");
                jsonObject.put("hour", "16");
                jsonObject.put("minute", "41");
                jsonObject.put("day_of_week", "3");
                jsonObject.put("week", "34");
                jsonObject.put("rate_number", "1");
                String message = jsonObject.toString();

                conn = (HttpURLConnection) url.openConnection();
                conn.setReadTimeout( 10000 /*milliseconds*/ );
                conn.setConnectTimeout( 15000 /* milliseconds */ );
                conn.setRequestMethod("POST");
                conn.setDoInput(true);
                conn.setDoOutput(true);
                conn.setFixedLengthStreamingMode(message.getBytes().length);

                //make some HTTP header nicety
                conn.setRequestProperty("Content-Type", "application/json;charset=utf-8");
                conn.setRequestProperty("X-Requested-With", "XMLHttpRequest");

                //open
                conn.connect();

                //setup send
                os = new BufferedOutputStream(conn.getOutputStream());
                os.write(message.getBytes());
                //clean up
                os.flush();

                //do somehting with response
                is = conn.getInputStream();
                //String contentAsString = readIt(is,len);
             catch (IOException e) 
                e.printStackTrace();
             catch (JSONException e) 
                e.printStackTrace();
             finally 
                //clean up
                try 
                    os.close();
                    is.close();
                 catch (IOException e) 
                    e.printStackTrace();
                

                conn.disconnect();
            
        
    ).start();

这个在 php 部分:

$json = file_get_contents('php://input');
$obj = json_decode($json);
$rate = $obj->'rate';
$comment = $obj->'comment';
$category = $obj->'category';
$day = $obj->'day';
$month = $obj->'month';
$year = $obj->'year';
$hour = $obj->'hour';
$minute = $obj->'minute';
$day_of_week = $obj->'day_of_week';
$week = $obj->'week';
$rate_number = $obj->'rate_number';

try 

    $bdd = new PDO('mysql:host=localhost;dbname=happiness_barometer;charset=utf8', 'utilisateur', '');
 catch (Exception $e) 

    die('Erreur : '.$e->getMessage());


$sql = $bdd->prepare(
'INSERT INTO rates (rate, comment, category, day, month, year, hour, minute, day_of_week, week, rate_number) 
VALUES (:rate, :comment, :category, :day, :month, :year, :hour, :minute, :day_of_week, :week, :rate_number)');
if (!empty($rate)) 
    $sql->execute(array(
        'rate' => $rate,
        'comment' => $comment,
        'category' => $category,
        'day' => $day,
        'month' => $month,
        'year' => $year,
        'hour' => $hour,
        'minute' => $minute,
        'day_of_week' => $day_of_week,
        'week' => $week,
        'rate_number' => $rate_number));

希望对其他人有所帮助;)

【讨论】:

@Anaïs 我使用相同的代码,我得到的响应 httpUrlConnection.getResponse() 是 HttpURLConnection.HTTP_OK 但 php 不接受 json 数据。请帮忙 您将 os 声明为 OutputStream,但将其分配为 BufferedOutputStream。不确定您使用的是什么 API,但它不适用于当前版本。 嗨。我试试你的代码。但什么都没有显示。请帮我。 .php 文件有问题吗?【参考方案2】:

检查此代码:

public class UniversalNetworkConnection 

    public static String postJSONObject(String myurl, JSONObject parameters) 
        HttpURLConnection conn = null;
        try 
            StringBuffer response = null;
            URL url = new URL(myurl);
            conn = (HttpURLConnection) url.openConnection();
            conn.setReadTimeout(10000);
            conn.setConnectTimeout(15000);
            conn.setRequestProperty("Content-Type", "application/json");
            conn.setDoOutput(true);
            conn.setRequestMethod("POST");
            OutputStream out = new BufferedOutputStream(conn.getOutputStream());
            BufferedWriter writer = new BufferedWriter(new OutputStreamWriter(out, "UTF-8"));
            writer.write(parameters.toString());
            writer.close();
            out.close();
            int responseCode = conn.getResponseCode();
            System.out.println("responseCode" + responseCode);
            switch (responseCode) 
                case 200:
                    BufferedReader in = new BufferedReader(new InputStreamReader(conn.getInputStream()));
                    String inputLine;
                    response = new StringBuffer();
                    while ((inputLine = in.readLine()) != null) 
                        response.append(inputLine);
                    
                    in.close();
                    return response.toString();
            
         catch (IOException ex) 
            ex.printStackTrace();
         finally 
            if (conn != null) 
                try 
                    conn.disconnect();
                 catch (Exception ex) 
                    ex.printStackTrace();
                
            
        
        return null;
    


在 PHP 端:

<?php
    $json = file_get_contents('php://input');
    $obj = json_decode($json);
    print_r($obj);
    print_r("this is a test");
?>

【讨论】:

我尝试了这段代码,但也没有发生任何事情:没有异常,没有错误,logcat 中的 System.out 没有。 @Anaïs 检查 wamp -> apache -> 访问日志。检查连接是否已建立 @Bonatti I 检查访问日志,本地网络上的客户端(IP 地址:192.168.36.212)“打开”了 php_output 文件,但 php_input 文件从未“打开”。我还发现我认为是 408 错误。有用吗? @Anaïs 408 很可能是服务器在预期的时间内没有收到完整的请求。请使用 cURL 或其他方法执行来自不同来源的请求,这可用于排除服务器端的任何故障。如果服务器没有错误,那么您的 Android 设备上缺少请求部分......即至少 Content-TypeContent-Lengthcharset @Harsha 对我来说听起来像是外包!

以上是关于当我们向服务器post数据时,表单 or JSON的主要内容,如果未能解决你的问题,请参考以下文章

如何向php服务器发送数据为json的post请求

angularjs 在使用 POST 向服务器发送数据时导致 json 格式错误

表单提交中get和post方式的区别

使用 JQUERY UI 向 Django 发送 POST 数据

邮递员原始数据有效,但表单数据不适用于节点中的 POST 请求

当连接是移动数据 xamarin 表单时,无法在服务器上发出 Put 和 Post 请求