Как сделать класс JSON сериализуемым для использования в сеансе Django - PullRequest
0 голосов
/ 12 января 2019

Я пытаюсь сохранить в сессии Django следующий довольно произвольный класс:

class QuizInfo:
    def __init__(self):
        self.i_active_question = INVALID_PQA_ID  # integer
        self.sequence = []

В свою очередь, каждый элемент списка QuizInfo.sequence является экземпляром AnsweredQuestion:

class AnsweredQuestion:
    def __init__(self, i_question, i_answer):
        self.i_question = i_question
        self.i_answer = i_answer

Я пытаюсь сохранить такую ​​структуру данных в HttpRequest.session, как показано ниже:

qi = QuizInfo()
qi.i_active_question = 5  # e.g.
qi.sequence.append(AnsweredQuestion(1, 2))  # e.g.
qi.sequence.append(AnsweredQuestion(3, 4))  # e.g.
quiz_id = 7  # e.g.
request.session['quiz_infos'] = {quiz_id : qi}

Итак, request.session['quiz_infos'] - это словарь int ключей и QuizInfo значений.

Я получаю следующую ошибку:

Traceback:

File "C:\Users\Sarge\Envs\PqaWeb1\lib\site-packages\django\core\handlers\exception.py" in inner
  34.             response = get_response(request)

File "C:\Users\Sarge\Envs\PqaWeb1\lib\site-packages\django\utils\deprecation.py" in __call__
  93.             response = self.process_response(request, response)

File "C:\Users\Sarge\Envs\PqaWeb1\lib\site-packages\django\contrib\sessions\middleware.py" in process_response
  58.                             request.session.save()

File "C:\Users\Sarge\Envs\PqaWeb1\lib\site-packages\django\contrib\sessions\backends\db.py" in save
  83.         obj = self.create_model_instance(data)

File "C:\Users\Sarge\Envs\PqaWeb1\lib\site-packages\django\contrib\sessions\backends\db.py" in create_model_instance
  70.             session_data=self.encode(data),

File "C:\Users\Sarge\Envs\PqaWeb1\lib\site-packages\django\contrib\sessions\backends\base.py" in encode
  96.         serialized = self.serializer().dumps(session_dict)

File "C:\Users\Sarge\Envs\PqaWeb1\lib\site-packages\django\core\signing.py" in dumps
  87.         return json.dumps(obj, separators=(',', ':')).encode('latin-1')

File "c:\programs\python37\Lib\json\__init__.py" in dumps
  238.         **kw).encode(obj)

File "c:\programs\python37\Lib\json\encoder.py" in encode
  199.         chunks = self.iterencode(o, _one_shot=True)

File "c:\programs\python37\Lib\json\encoder.py" in iterencode
  257.         return _iterencode(o, 0)

File "c:\programs\python37\Lib\json\encoder.py" in default
  179.         raise TypeError(f'Object of type {o.__class__.__name__} '

Exception Type: TypeError at /
Exception Value: Object of type QuizInfo is not JSON serializable

Как сделать структуру данных JSON-сериализуемой?

Я видел другие вопросы, однако ответы там, в основном, предлагают реализовать метод передачи в качестве параметра. Но где это сделать в Джанго?

Ответы [ 2 ]

0 голосов
/ 13 января 2019

Здесь является возможным решением.

Сериализаторы Django удобны, когда дело доходит до сериализации / десериализации данных.

Вы также можете бросить взгляд на это . Проверьте также десериализация .

0 голосов
/ 12 января 2019

Используйте __dict__, чтобы изменить это в словарь. В вашем случае, хотя у вас есть вложенные объекты, которые также необходимо преобразовать в словари. Это можно найти в ответе jsbueno здесь . По сути, вы должны перебирать свой объект, чтобы проверить наличие вложенных объектов, например:

def my_dict(obj):
   if not  hasattr(obj,"__dict__"):
       return obj
   result = {}
   for key, val in obj.__dict__.items():
       if key.startswith("_"):
           continue
       element = []
       if isinstance(val, list):
           for item in val:
               element.append(my_dict(item))
       else:
           element = my_dict(val)
       result[key] = element
   return result

Edit:

Затем вы можете вывести результат в строку, например,

import json

json.dumps(result)

Надеюсь, это поможет!

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