将JSON数据保存到Django模型中

5

我试图将JSON数据保存到模型中。我已经获取了所有必要的数据,如何将它们保存到模型中?

views.py

def book_api(request):
if request.method == 'POST':
    search = request.POST['textfield']
    url = 'https://www.googleapis.com/books/v1/volumes?q=' + search
    print(url)
    r = requests.get(url).json()
    book_info = {
        'title': r['items'][0-2]['volumeInfo']['title'],
        'description': r['items'][2]['volumeInfo']['description'],
        'author_name': r['items'][0-2]['volumeInfo']['authors'],
        'genres': r['items'][0-2]['volumeInfo']['categories'],

    }
    print(book_info)

    return redirect('index')
else:
    return render(request, 'api/book_api.html')

models.py

class Genres(models.Model):
    genres = models.CharField(max_length=200)

    def __str__(self):
        return self.genres


class Authors(models.Model):
    author_name = models.CharField(max_length=200)

    def __str__(self):
        return self.author_name


class Books(models.Model):
    title = models.CharField(max_length=300)
    description = models.TextField()
    authors = models.ForeignKey(Authors, on_delete=models.CASCADE)
    genres = models.ForeignKey(Genres, on_delete=models.CASCADE)

    def __str__(self):
      return self.title

我尝试执行Books.objects.save(**book_info),但是出现了错误:

'Manager'对象没有'save'属性


已解决 https://dev59.com/vVoV5IYBdhLWcg3wivP9#62634107 - hassanzadeh.sd
2个回答

11

如果您从使用谷歌 API 获得的答案在每次查询中包含不同的属性,并且您想要将它们全部存储,则必须使用 JSONField

from django.db import models
from django.core.serializers.json import DjangoJSONEncoder
import json

class JSONField(models.TextField):
    """
    JSONField es un campo TextField que serializa/deserializa objetos JSON.
    Django snippet #1478

    Ejemplo:
        class Page(models.Model):
            data = JSONField(blank=True, null=True)

        page = Page.objects.get(pk=5)
        page.data = {'title': 'test', 'type': 3}
        page.save()
    """
    def to_python(self, value):
        if value == "":
            return None

        try:
            if isinstance(value, str):
                return json.loads(value)
        except ValueError:
            pass
        return value

    def from_db_value(self, value, *args):
        return self.to_python(value)

    def get_db_prep_save(self, value, *args, **kwargs):
        if value == "":
            return None
        if isinstance(value, dict):
            value = json.dumps(value, cls=DjangoJSONEncoder)
        return value

在你的models.py文件中:

class Book(Model.models):
    info = JSONField(null=True, blank=True)
    ...

在您的views.py文件中:

 ...
 book.info = json.loads(r.content)
 book.save()

但是如果你需要始终保存相同的属性,对于你的情况来说是titledescriptionauthor_namegenres

Book.objects.create(title = r['items'][0-2]['volumeInfo']['title'], description = r['items'][2]['volumeInfo']['description'], ......)

非常感谢,我卡在那里了。你为什么使用'r'?我收到NameError:名称'r'未定义。 - Georgi Ivanov Dimitrov
1
在你的问题中,你定义了“r”为“r = requests.get(url).json()”,所以我使用了“r”。 - Jota

1
Django使用ORM(对象关系映射)。因此,'manage.py makemigrations'命令将把models.Model实例转换为SQL。
因此,您可以查看您使用的数据库是否具有此字段。
在Postgresql中,您可以使用JSON字段。其他一些数据库也可能具有此字段。
对于Postgresql,您可以访问: PostgreSQL的JSONField 对于MYSQL:MySQL的JSONField

你的错误: 在:Books.objects.save(**book_info) 首先,在Python中,**表示一个字典, 将其转换为以下形式:

Books.objects.save(genere='类型', author='作者名字', ....)

文本字段的作用是保存文本。 什么是文本?是一个长而无限制的字符串。 如何将字典转换为字符串?调用json.dumps(dictionary)

将返回的值保存到您的文本字段中。

如何从字符串再次生成字典?调用json.loads(string) 结果就是字典。

还有一些限制,比如无法转储datetime或numpy ndarray……


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接