2008-09-03 2 views
193

Я создал некоторый JSON, и я пытаюсь вытащить его в объект в JavaScript. Я продолжаю получать ошибки. Вот что у меня есть:Как обрабатывать строки в JSON?

var data = '{"count" : 1, "stack" : "sometext\n\n"}'; 
var dataObj = eval('('+data+')'); 

Это дает мне ошибку:

unterminated string literal 

С JSON.parse(data), я вижу подобные сообщения об ошибках: "Unexpected token ↵" в Chrome, и "unterminated string literal" в Firefox и IE.

Когда я вывожу \n после sometext, ошибка в обоих случаях. Я не могу понять, почему \n делает eval и JSON.parse терпит неудачу.

+7

Попробуйте использовать реальный синтаксический анализатор json вместо eval. – Eric 2011-03-04 08:08:45

ответ

257

Я думаю, это то, что вы хотите:

var data = '{"count" : 1, "stack" : "sometext\\n\\n"}'; 

(Вы должны избежать «\» в строке (превращая его в двойной - «\»), в противном случае она станет новой строки в . источник JSON, а не данные в формате JSON)

+63

Это, конечно, правильно, но я хотел бы добавить причину этого: спецификация JSON на http://www.ietf.org/rfc/rfc4627.txt содержит это предложение в разделе 2.5: «Все Символы Юникода могут быть помещены в кавычки, за исключением символов, которые должны быть экранированы: кавычка, обратное солидус и управляющие символы (U + 0000 до U + 001F). " Поскольку новая строка является управляющим символом, она должна быть экранирована. – 2013-04-25 10:48:02

+0

Согласно www.json.org JSON принимает последовательность управления «\ n» в строках - и если вы пытаетесь JSON.parse (['' a \\ na "']) [1] .charCodeAt(); который покажет 10 - это был «Linefeed» в последний раз, когда я проверил. --- Кстати: перестань кричать! – BlaM 2015-11-11 07:25:09

2

Вы можете посмотреть в этот C# функции, чтобы избежать строки:

http://www.aspcode.net/C-encode-a-string-for-JSON-JavaScript.aspx

public static string Enquote(string s) 
{ 
    if (s == null || s.Length == 0) 
    { 
     return "\"\""; 
    } 
    char   c; 
    int   i; 
    int   len = s.Length; 
    StringBuilder sb = new StringBuilder(len + 4); 
    string  t; 

    sb.Append('"'); 
    for (i = 0; i < len; i += 1) 
    { 
     c = s[i]; 
     if ((c == '\\') || (c == '"') || (c == '>')) 
     { 
      sb.Append('\\'); 
      sb.Append(c); 
     } 
     else if (c == '\b') 
      sb.Append("\\b"); 
     else if (c == '\t') 
      sb.Append("\\t"); 
     else if (c == '\n') 
      sb.Append("\\n"); 
     else if (c == '\f') 
      sb.Append("\\f"); 
     else if (c == '\r') 
      sb.Append("\\r"); 
     else 
     { 
      if (c < ' ') 
      { 
       //t = "000" + Integer.toHexString(c); 
       string t = new string(c,1); 
       t = "000" + int.Parse(tmp,System.Globalization.NumberStyles.HexNumber); 
       sb.Append("\\u" + t.Substring(t.Length - 4)); 
      } 
      else 
      { 
       sb.Append(c); 
      } 
     } 
    } 
    sb.Append('"'); 
    return sb.ToString(); 
} 
+1

Вам нужно немного почистить этот код ... (Не компилируется.) – 2011-03-29 09:31:42

+2

Почему это происходит `` `? – nothingisnecessary 2015-01-12 23:18:27

19

Вам понадобится функция, которая заменяет \n на \\n в случае, если data не является строковым литералом.

function jsonEscape(str) { 
    return str.replace(/\n/g, "\\\\n").replace(/\r/g, "\\\\r").replace(/\t/g, "\\\\t"); 
} 

var data = '{"count" : 1, "stack" : "sometext\n\n"}'; 
var dataObj = JSON.parse(jsonEscape(data)); 

Результирующая dataObj будет

Object {count: 1, stack: "sometext\n\n"} 
+2

вам нужно избежать ваших escape-символов (например, `.replace (" \\ n "," \\\\ n ")`), и я также предложил бы использовать регулярное выражение для замены нескольких экземпляров (например, `.replace (/ \ n/g, "\\\\ n") `) – musefan 2012-03-12 09:14:29

0

я столкнулся с этой проблемой, делая класс в PHP4 для эмуляции json_encode (доступен в PHP5). Вот что я придумал:

class jsonResponse { 
    var $response; 

    function jsonResponse() { 
     $this->response = array('isOK'=>'KO','msg'=>'Undefined'); 
    } 

    function set($isOK, $msg) { 
     $this->response['isOK'] = ($isOK) ? 'OK' : 'KO'; 
     $this->response['msg'] = htmlentities($msg); 
    } 

    function setData($data=null) { 
     if(!is_null($data)) 
      $this->response['data'] = $data; 
     elseif(isset($this->response['data'])) 
      unset($this->response['data']); 
    } 

    function send() { 
     header('Content-type: application/json'); 
     echo '{"isOK":"'.$this->response['isOK'].'","msg":'.$this->parseString($this->response['msg']); 
     if(isset($this->response['data'])) 
      echo ',"data":'.$this->parseData($this->response['data']); 
     echo '}'; 
    } 

    function parseData($data) { 
     if(is_array($data)) { 
      $parsed = array(); 
      foreach ($data as $key=>$value) 
       array_push($parsed, $this->parseString($key).':'.$this->parseData($value)); 
      return '{'.implode(',', $parsed).'}'; 
     } else 
      return $this->parseString($data); 
    } 

    function parseString($string) { 
      $string = str_replace("\\", "\\\\", $string); 
      $string = str_replace('/', "\\/", $string); 
      $string = str_replace('"', "\\".'"', $string); 
      $string = str_replace("\b", "\\b", $string); 
      $string = str_replace("\t", "\\t", $string); 
      $string = str_replace("\n", "\\n", $string); 
      $string = str_replace("\f", "\\f", $string); 
      $string = str_replace("\r", "\\r", $string); 
      $string = str_replace("\u", "\\u", $string); 
      return '"'.$string.'"'; 
    } 
} 

Я следовал правилам here упоминалось. Я использовал только то, что мне нужно, но я считаю, что вы можете адаптировать его к вашим потребностям на используемом вами языке. Проблема в моем случае была не о новостях, как я изначально думал, а о том, что/не удалось избежать. Надеюсь, это помешает кому-то еще от маленькой головной боли, я выяснил, что я сделал неправильно.

2

Привет я использовал эту функцию, чтобы лишить символ новой строки или другие символы в данных для анализа JSON данные:

function normalize_str($str) { 

    $invalid = array('Š'=>'S', 'š'=>'s', 'Đ'=>'Dj', 'đ'=>'dj', 'Ž'=>'Z', 'ž'=>'z', 
    'Č'=>'C', 'č'=>'c', 'Ć'=>'C', 'ć'=>'c', 'À'=>'A', 'Á'=>'A', 'Â'=>'A', 'Ã'=>'A', 
    'Ä'=>'A', 'Å'=>'A', 'Æ'=>'A', 'Ç'=>'C', 'È'=>'E', 'É'=>'E', 'Ê'=>'E', 'Ë'=>'E', 
    'Ì'=>'I', 'Í'=>'I', 'Î'=>'I', 'Ï'=>'I', 'Ñ'=>'N', 'Ò'=>'O', 'Ó'=>'O', 'Ô'=>'O', 
    'Õ'=>'O', 'Ö'=>'O', 'Ø'=>'O', 'Ù'=>'U', 'Ú'=>'U', 'Û'=>'U', 'Ü'=>'U', 'Ý'=>'Y', 
    'Þ'=>'B', 'ß'=>'Ss', 'à'=>'a', 'á'=>'a', 'â'=>'a', 'ã'=>'a', 'ä'=>'a', 'å'=>'a', 
    'æ'=>'a', 'ç'=>'c', 'è'=>'e', 'é'=>'e', 'ê'=>'e', 'ë'=>'e', 'ì'=>'i', 'í'=>'i', 
    'î'=>'i', 'ï'=>'i', 'ð'=>'o', 'ñ'=>'n', 'ò'=>'o', 'ó'=>'o', 'ô'=>'o', 'õ'=>'o', 
    'ö'=>'o', 'ø'=>'o', 'ù'=>'u', 'ú'=>'u', 'û'=>'u', 'ý'=>'y', 'ý'=>'y', 'þ'=>'b', 
    'ÿ'=>'y', 'Ŕ'=>'R', 'ŕ'=>'r', "`" => "'", "´" => "'", '"' => ',', '`' => "'", 
    '´' => "'", '"' => '\"', '"' => "\"", '´' => "'", "&acirc;€™" => "'", "{" => "", 
    "~" => "", "–" => "-", "'" => "'","  " => " "); 

    $str = str_replace(array_keys($invalid), array_values($invalid), $str); 

    $remove = array("\n", "\r\n", "\r"); 
    $str = str_replace($remove, "\\n", trim($str)); 

     //$str = htmlentities($str,ENT_QUOTES); 

    return htmlspecialchars($str); 
} 


echo normalize_str($lst['address']); 
2

Вы могли бы просто убежать вашу строку на сервере при записи значения поля JSon и экранирования в нем, когда например, получение значения в клиентском браузере.

Реализация javascript всех основных браузеров имеет команду unescape.

Пример: на сервере:

response.write "{""field1"":""" & escape(RS_Temp("textField")) & """}" 

в браузере:

document.getElementById("text1").value = unescape(jsonObject.field1) 
3

Согласно спецификации: http://www.ecma-international.org/publications/files/ECMA-ST/ECMA-404.pdf

A string is a sequence of Unicode code points wrapped with quotation marks 
(U+0022). All characters may be placed within the quotation marks except for the 
characters that must be escaped: quotation mark (U+0022), reverse solidus 
(U+005C), and the control characters U+0000 to U+001F. There are two-character 
escape sequence representations of some characters. 

Таким образом, вы не можете пройти 0x0A или 0x0C коды напрямую. Запрещено! Spec предлагает использовать управляющие последовательности для некоторых хорошо определенных кодов из U+0000 в U+001F:

\f represents the form feed character (U+000C). 
\n represents the line feed character (U+000A). 

В большинстве языков программирования использует \ для цитирования вы должны избежать синтаксической побег (двойной побег - один раз для языка/платформы, один раз Json):

jsonStr = "{ \"name\": \"Multi\\nline.\" }"; 
Смежные вопросы