[python] json.dumps에 utf-8 텍스트를 UTF8로 저장하고 이스케이프 시퀀스로 저장하지 않습니다.


2 Answers

케이크처럼 쉽다.

파일에 쓰려면

import codecs
import json

with codecs.open('your_file.txt', 'w', encoding='utf-8') as f:
    json.dump({"message":"xin chào việt nam"}, f, ensure_ascii=False)

stdin으로 인쇄하려면

import codecs
import json
print(json.dumps({"message":"xin chào việt nam"}, ensure_ascii=False))
Question

샘플 코드 :

>>> import json
>>> json_string = json.dumps("ברי צקלה")
>>> print json_string
"\u05d1\u05e8\u05d9 \u05e6\u05e7\u05dc\u05d4"

문제는 사람이 읽을 수있는 것이 아닙니다. 내 (똑똑한) 사용자는 JSON 덤프로 텍스트 파일을 확인하거나 편집하려고합니다. (그리고 나는 XML을 사용하지 않을 것입니다)

\ uXXXX 대신 utf-8 json 문자열로 객체를 직렬화하는 방법이 있습니까?

이 도움이되지 않습니다 :

>>> output = json_string.decode('string-escape')
"\u05d1\u05e8\u05d9 \u05e6\u05e7\u05dc\u05d4"

작품은,하지만 어떤 하위 개체는 파이썬 유니 코드 및 utf - 8 아니라면 쓰레기를 버릴거야 :

>>> #### ok:
>>> s= json.dumps( "ברי צקלה", ensure_ascii=False)    
>>> print json.loads(s)   
ברי צקלה

>>> #### NOT ok:
>>> d={ 1: "ברי צקלה", 2: u"ברי צקלה" }
>>> print d
{1: '\xd7\x91\xd7\xa8\xd7\x99 \xd7\xa6\xd7\xa7\xd7\x9c\xd7\x94', 
 2: u'\xd7\x91\xd7\xa8\xd7\x99 \xd7\xa6\xd7\xa7\xd7\x9c\xd7\x94'}
>>> s = json.dumps( d, ensure_ascii=False, encoding='utf8')
>>> print json.loads(s)['1']
ברי צקלה
>>> print json.loads(s)['2']
××¨× ×¦×§××

json.dumps 문서를 검색했지만 유용한 것을 찾을 수 없습니다.

편집 - 솔루션 (?) :

Martijn Pieters의 의견과 답변을 요약 해 보겠습니다.

(편집 : 세바스찬 주석의 생각과 약 1 년 후 제 2의 생각)

  1. json.dumps에는 기본 제공 솔루션이 없습니다.

  2. 나는 모든 문자열을 JSON-ed되기 전에 UTF8 유니 코드로 변환해야 할 것이다. 중첩 된 객체에서 문자열을 재귀 적으로 변환하는 Mark의 함수를 사용합니다.

  3. 내가 준 예제는 내 컴퓨터 및 IDE 환경에 너무 많이 의존하며 모든 컴퓨터에서 동일하게 실행되지 않습니다.

모두 감사합니다 :)




다음은 내 이해 var 읽기 위와 구글입니다.

# coding:utf-8
r"""
@update: 2017-01-09 14:44:39
@explain: str, unicode, bytes in python2to3
    #python2 UnicodeDecodeError: 'ascii' codec can't decode byte 0xe4 in position 7: ordinal not in range(128)
    #1.reload
    #importlib,sys
    #importlib.reload(sys)
    #sys.setdefaultencoding('utf-8') #python3 don't have this attribute.
    #not suggest even in python2 #see:http://.com/questions/3828723/why-should-we-not-use-sys-setdefaultencodingutf-8-in-a-py-script
    #2.overwrite /usr/lib/python2.7/sitecustomize.py or (sitecustomize.py and PYTHONPATH=".:$PYTHONPATH" python)
    #too complex
    #3.control by your own (best)
    #==> all string must be unicode like python3 (u'xx'|b'xx'.encode('utf-8')) (unicode 's disappeared in python3)
    #see: http://blog.ernest.me/post/python-setdefaultencoding-unicode-bytes

    #how to Saving utf-8 texts in json.dumps as UTF8, not as \u escape sequence
    #http://.com/questions/18337407/saving-utf-8-texts-in-json-dumps-as-utf8-not-as-u-escape-sequence
"""

from __future__ import print_function
import json

a = {"b": u"中文"}  # add u for python2 compatibility
print('%r' % a)
print('%r' % json.dumps(a))
print('%r' % (json.dumps(a).encode('utf8')))
a = {"b": u"中文"}
print('%r' % json.dumps(a, ensure_ascii=False))
print('%r' % (json.dumps(a, ensure_ascii=False).encode('utf8')))
# print(a.encode('utf8')) #AttributeError: 'dict' object has no attribute 'encode'
print('')

# python2:bytes=str; python3:bytes
b = a['b'].encode('utf-8')
print('%r' % b)
print('%r' % b.decode("utf-8"))
print('')

# python2:unicode; python3:str=unicode
c = b.decode('utf-8')
print('%r' % c)
print('%r' % c.encode('utf-8'))
"""
#python2
{'b': u'\u4e2d\u6587'}
'{"b": "\\u4e2d\\u6587"}'
'{"b": "\\u4e2d\\u6587"}'
u'{"b": "\u4e2d\u6587"}'
'{"b": "\xe4\xb8\xad\xe6\x96\x87"}'

'\xe4\xb8\xad\xe6\x96\x87'
u'\u4e2d\u6587'

u'\u4e2d\u6587'
'\xe4\xb8\xad\xe6\x96\x87'

#python3
{'b': '中文'}
'{"b": "\\u4e2d\\u6587"}'
b'{"b": "\\u4e2d\\u6587"}'
'{"b": "中文"}'
b'{"b": "\xe4\xb8\xad\xe6\x96\x87"}'

b'\xe4\xb8\xad\xe6\x96\x87'
'中文'

'中文'
b'\xe4\xb8\xad\xe6\x96\x87'
"""



Peters 'python 2 해결 방법은 딱딱한 경우에 실패합니다.

d = {u'keyword': u'bad credit  \xe7redit cards'}
with io.open('filename', 'w', encoding='utf8') as json_file:
    data = json.dumps(d, ensure_ascii=False).decode('utf8')
    try:
        json_file.write(data)
    except TypeError:
        # Decode data to Unicode first
        json_file.write(data.decode('utf8'))

UnicodeEncodeError: 'ascii' codec can't encode character u'\xe7' in position 25: ordinal not in range(128)

3 행의 .decode ( 'utf8') 부분에서 충돌이 발생했습니다. 나는 ascii의 특별한 케이스뿐만 아니라 그 단계를 피함으로써 프로그램을 훨씬 단순하게 만들어 문제를 해결했습니다 :

with io.open('filename', 'w', encoding='utf8') as json_file:
  data = json.dumps(d, ensure_ascii=False, encoding='utf8')
  json_file.write(unicode(data))

cat filename
{"keyword": "bad credit  çredit cards"}



Related