web-dev-qa-db-ja.com

PHP配列をjson_encodeする方法は引用符なしのキー

いくつかのデータを含む円グラフを( Flot で)プロットしようとしています

_var data = <?php echo json_encode($data)?>
_

それから私が得る結果はこれです:

_var data = [
{"label":"Crear Usuario", "data":"2"},
{"label":"Impresoras", "data":"1"},
{"label":"Problema Correo", "data":"1"},
{"label":"Requisicion Equipo", "data":"1"},
{"label":"Sitio Web", "data":"1"}
]
_

ここでの問題は、引用符なしでlabeldataが必要なことです。すでにjson_encode($data, JSON_NUMERIC_CHECK);を試しましたが、数字から引用符を削除するだけです。

次の形式が必要です。

_var data = [
    {label:"Crear Usuario",data:2}, ...
_
19
Danny

最初に、データの値が文字列ではなく整数になるように、phpで配列を生成する必要があります。

私はあなたのjson_encode()からあなたの配列をエミュレートしました、私はそれがこのように見えると思います(またはそれはすべきです):

$array =  array(
                array("label" => "Crear Usuario",   "data" => 2),
                array("label" => "Impresoras",      "data" => 1),
                array("label" => "Problema Correo", "data" => 1),
                array("label" => "Requisicion Equipo", "data" => 1),
                array("label" => "Sitio Web", "data" => 1)
            );

    $data = json_encode($array);
  • 2と1は引用符で囲まれていないため、整数であることに注意してください。

次に、JavaScriptでJSON.parse()を実行して、実際にその出力をjsonオブジェクトに変換します。

<script>
    var data = '<?php echo $data; ?>';
    var json = JSON.parse(data);
    console.log(json);
    console.log(json[0]);
</script>
  • var data = ...は単一引用符で囲まれているため、phpからのエコーを文字列としてキャッチすることに注意してください

Console.log()はこれを私に出力します:

[Object, Object, Object, Object, Object] // First console.log(): one object with the 5 Objects. 
Object {label: "Crear Usuario", data: 2} // secons console log (json[0]) with the first object 

必要なもののように見えますが、私は正しいのでしょうか?

13
aleation

引用符付きと引用符なしのキーに違いはありません。 Flotは文字列ではなく数値を必要とするため、問題は実際のデータ値を引用することです。

Json_encode関数は、提供するデータのタイプに基づいて引用するかどうかを決定します。この場合、$ dataを作成するために実行している操作が整数ではなく文字列値を生成しているように見えます。これらの操作を再検討するか、PHPに明示的に伝えて、(int)または(float)キャスト、またはintval/floatval関数を使用して、それらを数値として解釈する必要があります。

6
DNS

次のようなものを試してください:

function buildBrokenJson( array $data ) {

   $result = '{';

   $separator = '';
   foreach( $data as $key=>$val ) {
      $result .= $separator . $key . ':';

      if( is_int( $val ) ) {
         $result .= $val;
      } elseif( is_string( $val ) ) {
         $result .= '"' . str_replace( '"', '\"', $val) . '"';
      } elseif( is_bool( $val ) ) {
         $result .= $val ? 'true' : 'false';
      } else {
         $result .= $val;
      }

      $separator = ', ';
   }

   $result .= '}';

   return $result;
}

そして実行すると

$a = array("string"=>'Crear "Usuario', 'foo'=>':', "int"=>2, "bool"=>false);
var_dump( buildBrokenJson($a) );

それは与えます:

string(54) "{string:"Crear \"Usuario", foo:":", int:2, bool:false}"
3
Marcin Orlowski

TL; DR:引用符の欠如は、Chromeが文字列ではなくJSONオブジェクトであることを示しています。Header( 'Content-Type:application/json; charset = UTF8'); PHPのAJAX=応答で、実際の問題を解決します。

詳細:この問題を解決する一般的な理由は、返されたAJAXデータの処理をデバッグしているときにこの違いを見つけることです。

私の場合、Chromeのデバッグツールを使用して違いを確認しました。レガシーシステムに接続すると、成功したChromeは、デバッガーによる応答のキーの周りに引用符が表示されていないことを示しました。これにより、オブジェクトを使用せずにオブジェクトとしてすぐに処理できるようになりましたJSON.parse()呼び出し新しいAJAX=宛先をデバッグすると、応答に引用符が表示され、変数はオブジェクトではなく文字列でした。
AJAX応答をレガシーシステムで実際に外部から見たときに実際に問題に気づきましたDIDキーの周りに引用符があります。これはChrome開発ツールが示したものではありません。唯一の違いは、レガシーシステムにコンテンツタイプを指定するヘッダーがあったことです。これを新しい(WordPress)システムに追加し、呼び出しは元のスクリプトと完全に互換性があり、success関数は解析を必要とせずにオブジェクトとして応答を処理できました。これで、リンク先URLを変更することなく、レガシーシステムと新しいシステムを切り替えることができます。

0
Brian Layman

私はここにキーの引用符なしでphpでjsonをフォーマットするクラスを作成しました

class JsonFormatter
{
static $result = '';
static $separator = '';

public static function iterateArray($data) : string
{
    static::$result .= '[';
    static::$separator = '';
    foreach ($data as $key => $val) {
        if (is_int($val)) {

        } elseif (is_string($val)) {
            static::$result .= '"' . str_replace('"', '\"', $val) . '"';
        } elseif (is_bool($val)) {
            static::$result .= $val ? 'true' : 'false';
        } elseif (is_object($val)) {
            static::iterateObject($val);
            static::$result .= ', ';
        } elseif (is_array($val)) {
            static::iterateArray($val);
            static::$result .= ', ';
        } else {
            static::$result .= $val;
        }
        if (!is_int($val)) {
            static::$separator = ', ';
        }
    }

    static::$result .= ']';
    return static::$result;
}

public static function iterate($data)
{
    if (is_array($data)) {
        static::iterateArray($data);
    } elseif (is_object($data)) {
        static::iterateObject($data);
    }
    return static::$result;
}

public static function iterateObject($data)
{
    static::$result .= '{';
    static::$separator = '';
    foreach ($data as $key => $val) {

        static::$result .= static::$separator . $key . ':';

        if (is_int($val)) {
            static::$result .= $val;
        } elseif (is_string($val)) {
            static::$result .= '"' . str_replace('"', '\"', $val) . '"';
        } elseif (is_bool($val)) {
            static::$result .= $val ? 'true' : 'false';
        } elseif (is_object($val)) {
            static::iterate($val, true);
            static::$result .= ', ';
        } elseif (is_array($val)) {
            static::iterateArray($val, true);
            static::$result .= ', ';
        } else {
            static::$result .= $val;
        }
        static::$separator = ', ';
    }
    static::$result .= '}';
    return static::$result;
}

}

あなたは今呼ぶことができます

$jsonWithoutKeyQuotes  = JsonFormatter::iterate($data);

marcin Orlowskiに感謝

0
Ekene Madunagu