jQuery.parseJSON выдает ошибку «Invalid JSON» из-за экранированной одинарной кавычки в JSON - PullRequest
200 голосов
/ 16 февраля 2010

Я делаю запросы на свой сервер, используя jQuery.post(), и мой сервер возвращает объекты JSON (например, { "var": "value", ... }). Однако, если какое-либо из значений содержит одинарную кавычку (правильно экранированную как \'), jQuery не сможет проанализировать допустимую строку JSON, иначе Вот пример того, что я имею в виду ( сделано в консоли Chrome ):

data = "{ \"status\": \"success\", \"newHtml\": \"Hello \\\'x\" }";
eval("x = " + data); // { newHtml: "Hello 'x", status: "success" }

$.parseJSON(data); // Invalid JSON: { "status": "success", "newHtml": "Hello \'x" }

Это нормально? Нет ли способа правильно передать одинарную кавычку через JSON?

Ответы [ 7 ]

324 голосов
/ 16 февраля 2010

В соответствии со схемой конечного автомата на веб-сайте JSON допускаются только экранированные символы двойных кавычек, а не одинарные. Символы одинарных кавычек не должны быть экранированы:

http://www.json.org/string.gif


Обновление - Больше информации для интересующихся:

<ч />

Дуглас Крокфорд конкретно не говорит, почему спецификация JSON не допускает экранированные одинарные кавычки в строках. Однако во время обсуждения JSON в Приложении E JavaScript: Хорошие части он пишет:

Цели JSON были минимальными, переносимыми, текстовыми и представляли собой подмножество JavaScript. Чем меньше нам нужно договориться о взаимодействии, тем легче нам взаимодействовать.

Так что, возможно, он решил разрешить определять строки только с помощью двойных кавычек, поскольку это еще одно правило, с которым должны согласиться все реализации JSON. В результате невозможно, чтобы одиночный символ кавычки в строке случайно завершил строку, потому что по определению строка может заканчиваться только символом двойной кавычки. Следовательно, нет необходимости разрешать экранирование символа одинарной кавычки в формальной спецификации.

<ч /> Пропуская немного глубже, реализация Крокфорда org.json JSON для Java более допустима, а делает допускает символы одинарных кавычек:

Тексты, создаваемые методами toString, строго соответствуют правилам синтаксиса JSON. Конструкторы более снисходительны в текстах, которые они примут:

...

  • Строки могут быть заключены в кавычки '(одинарные кавычки).

Это подтверждается исходным кодом JSONTokener . Метод nextString принимает экранированные символы одинарных кавычек и обрабатывает их как символы двойных кавычек:

public String nextString(char quote) throws JSONException {
    char c;
    StringBuffer sb = new StringBuffer();
    for (;;) {
        c = next();
        switch (c) {

        ...

        case '\\':
            c = this.next();
            switch (c) {

            ...

            case '"':
            case '\'':
            case '\\':
            case '/':
                sb.append(c);
                break;
        ...

В верхней части метода находится информативный комментарий:

Формальный формат JSON не допускает строки в одинарных кавычках, но реализация может принимать их.

Так что некоторые реализации будут принимать одинарные кавычки - но вы не должны полагаться на это. Многие популярные реализации довольно ограничительны в этом отношении и будут отклонять JSON, который содержит строки в одинарных кавычках и / или экранированные одинарные кавычки.

<ч />

Наконец, чтобы связать это с первоначальным вопросом, jQuery.parseJSON сначала пытается использовать собственный анализатор JSON браузера или загруженную библиотеку, такую ​​как json2.js , где это применимо (что примечание - библиотека, на которой основана логика jQuery, если JSON не определено). Таким образом, jQuery может быть таким же разрешающим, как и базовая реализация:

parseJSON: function( data ) {
    ...

    // Attempt to parse using the native JSON parser first
    if ( window.JSON && window.JSON.parse ) {
        return window.JSON.parse( data );
    }

    ...

    jQuery.error( "Invalid JSON: " + data );
},

Насколько мне известно, эти реализации придерживаются только официальной спецификации JSON и не принимают одинарные кавычки, следовательно, jQuery также не допускается.

16 голосов
/ 16 сентября 2011

Если вам нужна одиночная кавычка внутри строки, поскольку \ 'не определено спецификацией, используйте \u0027, см. http://www.utf8 -chartable.de / для всех из них

edit: извините за неправильное использование обратных слов в комментариях. Я имел в виду обратную косую черту. Я хочу сказать, что в случае, если у вас есть вложенные строки внутри других строк, я думаю, что может быть более полезным и читабельным использовать юникод вместо множества обратных слешей, чтобы избежать одной кавычки. Однако, если вы не вложены, проще вставить туда просто старую цитату.

4 голосов
/ 30 октября 2014

Я понимаю, в чем проблема, и когда я смотрю на спецификации, становится ясно, что одиночные кавычки без экранирования должны быть правильно проанализированы.

Я использую функцию jQuery.parseJSON jquery для анализа строки JSON, но все еще получаю ошибку синтаксического анализа, когда в данных, подготовленных с помощью json_encode, есть одинарная кавычка.

Может ли быть ошибкой в ​​моей реализации, которая выглядит следующим образом (PHP - на стороне сервера):

$data = array();

$elem = array();
$elem['name'] = 'Erik';
$elem['position'] = 'PHP Programmer';
$data[] = json_encode($elem);

$elem = array();
$elem['name'] = 'Carl';
$elem['position'] = 'C Programmer';
$data[] = json_encode($elem);

$jsonString = "[" . implode(", ", $data) . "]";

Последний шаг - сохранение строки в кодировке JSON в переменной JS:

<script type="text/javascript">
employees = jQuery.parseJSON('<?=$marker; ?>');
</script>

Если я использую «» вместо «», он все равно выдаст ошибку.

РЕШЕНИЕ:

Единственное, что мне помогло, - это использовать битовую маску JSON_HEX_APOS для преобразования одинарных кавычек следующим образом:

json_encode($tmp, JSON_HEX_APOS);

Есть ли другой способ решения этой проблемы? Мой код неправильный или плохо написан?

Спасибо

3 голосов
/ 22 ноября 2012

Когда вы отправляете одинарную цитату в запросе

empid = " T'via"
empid =escape(empid)

Когда Вы получаете значение, включающее одинарную кавычку

var xxx  = request.QueryString("empid")
xxx= unscape(xxx)

Если вы хотите найти / вставить значение, которое включает в себя одну кавычку в запросе xxx=Replace(empid,"'","''")

2 голосов
/ 12 декабря 2014

Устранение аналогичной проблемы с использованием CakePHP для вывода скриптового блока JavaScript с использованием нативного PHP json_encode. $contractorCompanies содержит значения с одинарными кавычками и, как объяснено выше и ожидается, json_encode($contractorCompanies) не экранирует их, потому что это действительный JSON.

<?php $this->Html->scriptBlock("var contractorCompanies = jQuery.parseJSON( '".(json_encode($contractorCompanies)."' );"); ?>

Добавляя addlashes () вокруг строки, закодированной в JSON, вы избегаете кавычек, позволяющих Cake / PHP выводить правильный код JavaScript в браузер. Ошибки JS исчезают.

<?php $this->Html->scriptBlock("var contractorCompanies = jQuery.parseJSON( '".addslashes(json_encode($contractorCompanies))."' );"); ?>
0 голосов
/ 14 апреля 2017

Я пытался сохранить объект JSON из XHR-запроса в атрибут data5 * HTML5. Я попробовал многие из вышеперечисленных решений безуспешно.

В конечном итоге я заменил одинарную кавычку ' на код &#39;, используя регулярное выражение после вызова метода stringify () следующим образом:

var productToString = JSON.stringify(productObject);
var quoteReplaced = productToString.replace(/'/g, "&#39;");
var anchor = '<a data-product=\'' + quoteReplaced + '\' href=\'#\'>' + productObject.name + '</a>';
// Here you can use the "anchor" variable to update your DOM element.
0 голосов
/ 16 февраля 2010

Интересно. Как вы генерируете свой JSON на стороне сервера? Используете ли вы библиотечную функцию (например, json_encode в PHP) или вы строите строку JSON вручную?

Единственное, что привлекает мое внимание, - это спасительный апостроф (\'). Поскольку вы используете двойные кавычки, как вы и должны делать, вам не нужно избегать одинарных кавычек. Я не могу проверить, действительно ли это является причиной вашей ошибки jQuery, так как я еще не обновился до версии 1.4.1.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...