Django 二——models(admin、ORM),一对一、一对多、多对多操作,all、values、value_list的对比...

内容概要

1.关系对象映射ORM

2.admin的配置(选修)

3、all()、values()、value_list()的对比

4、数据库操作(一对一、一对多、多对多)

5、HttpResponse和render的对比 

 


1.关系对象映射ORM                                                                                                

一、用于实现面向对象编程语言里不同类型系统的数据之间的转换,换言之,就是用面向对象的方式去操作数据库的创建表以及增删改查等操作。

 

 优点: 1 ORM使得我们的通用数据库交互变得简单易行,而且完全不用考虑该死的SQL语句。快速开发,由此而来。

            2 可以避免一些新手程序猿写sql语句带来的性能问题。

 

 缺点:1  性能有所牺牲,不过现在的各种ORM框架都在尝试各种方法,比如缓存,延迟加载登来减轻这个问题。效果很显著。

            2  对于个别复杂查询,ORM仍然力不从心,为了解决这个问题,ORM一般也支持写raw sql。

            3  通过QuerySet的query属性查询对应操作的sql语句

二、表(模型)的创建:

(1)、创建类,一个类指代一张表

(2)、创建字段,字段类型、长度、显示名称

一对一 

一对多(外键)

多对多

(3)、生成同步数据库的脚本:python manage.py makemigrations  

                               同步数据库:  python manage.py migrate   

例:

from django.db import models

class Publisher(models.Model):
    name = models.CharField(max_length=30, verbose_name="名称")
    address = models.CharField("地址", max_length=50)
    city = models.CharField('城市',max_length=60)
    state_province = models.CharField(max_length=30)
    country = models.CharField(max_length=50)
    website = models.URLField()
 
    class Meta:
        verbose_name = '出版商'
        verbose_name_plural = verbose_name
 
    def __str__(self):
        return self.name
 
class Author(models.Model):
    name = models.CharField(max_length=30)
    def __str__(self):
        return self.name
 
class AuthorDetail(models.Model):
    sex = models.BooleanField(max_length=1, choices=((0, '男'),(1, '女'),))
    email = models.EmailField()
    address = models.CharField(max_length=50)
    birthday = models.DateField()
    author = models.OneToOneField(Author)
 
class Book(models.Model):
    title = models.CharField(max_length=100)
    authors = models.ManyToManyField(Author)
    publisher = models.ForeignKey(Publisher)
    publication_date = models.DateField()
    price=models.DecimalField(max_digits=5,decimal_places=2,default=10)
    def __str__(self):
        return self.title

 

三、模型常用的字段类型参数:

<1> CharField
        #字符串字段, 用于较短的字符串.
        #CharField 要求必须有一个参数 maxlength, 用于从数据库层和Django校验层限制该字段所允许的最大字符数.

<2> IntegerField
       #用于保存一个整数.

等等

四、Field重要参数

  <1> null : 数据库中字段是否可以为空

    <2> blank: django的 Admin 中添加数据时是否可允许空值

    <3> default:设定缺省值
等等

五、表的操作(增删改查):

5.1、单表操作

--------基本操作:增删改查

1、增(create  ,  save)

from app01.models import *
 
    #create方式一:   Author.objects.create(name='Alvin')
 
    #create方式二:   Author.objects.create(**{"name":"alex"})
 
    #save方式一:     author=Author(name="alvin")
                    author.save()
 
    #save方式二:     author=Author()
                    author.name="alvin"
                    author.save()

2、删(delete)

Book.objects.filter(id=1).delete()
(3, {'app01.Book_authors': 2, 'app01.Book': 1})

3、改(update和save)

#---------------- update方法直接设定对应属性----------------
    models.Book.objects.filter(id=3).update(title="PHP")
 
#--------------- save方法会将所有属性重新设定一遍,效率低-----------
    obj=models.Book.objects.filter(id=3)[0]
    obj.title="Python"
    obj.save()

4、查(filter,value等)

查询相关API:
 
#  <1>filter(**kwargs):      它包含了与所给筛选条件相匹配的对象
 
#  <2>all():                 查询所有结果
 
#  <3>get(**kwargs):         返回与所给筛选条件相匹配的对象,返回结果有且只有一个,如果符合筛选条件的对象超过一个或者没有都会抛出错误。
 
#-----------下面的方法都是对查询的结果再进行处理:比如 objects.filter.values()--------
 
#  <4>values(*field):        返回一个ValueQuerySet——一个特殊的QuerySet,运行后得到的并不是一系列 model的实例化对象,而是一个可迭代的字典序列
                                      
#  <5>exclude(**kwargs):     它包含了与所给筛选条件不匹配的对象
 
#  <6>order_by(*field):      对查询结果排序
 
#  <7>reverse():             对查询结果反向排序
 
#  <8>distinct():            从返回结果中剔除重复纪录

 惰性机制:

所谓惰性机制:Publisher.objects.all()或者.filter()等都只是返回了一个QuerySet(查询结果集对象),它并不会马上执行sql,而是当调用QuerySet的时候才执行。

QuerySet特点:

       <1>  可迭代的

       <2>  可切片

 

--------进阶操作(双下划线):大于、小于、in、contains、range、等

# 获取个数
        #
        # models.Tb1.objects.filter(name='seven').count()

        # 大于,小于
        #
        # models.Tb1.objects.filter(id__gt=1)              # 获取id大于1的值
        # models.Tb1.objects.filter(id__gte=1)              # 获取id大于等于1的值
        # models.Tb1.objects.filter(id__lt=10)             # 获取id小于10的值
        # models.Tb1.objects.filter(id__lte=10)             # 获取id小于10的值
        # models.Tb1.objects.filter(id__lt=10, id__gt=1)   # 获取id大于1 且 小于10的值

        # in
        #
        # models.Tb1.objects.filter(id__in=[11, 22, 33])   # 获取id等于11、22、33的数据
        # models.Tb1.objects.exclude(id__in=[11, 22, 33])  # not in

        # isnull
        # Entry.objects.filter(pub_date__isnull=True)

        # contains
        #
        # models.Tb1.objects.filter(name__contains="ven")
        # models.Tb1.objects.filter(name__icontains="ven") # icontains大小写不敏感
        # models.Tb1.objects.exclude(name__icontains="ven")

        # range
        #
        # models.Tb1.objects.filter(id__range=[1, 2])   # 范围bettwen and

        # 其他类似
        #
        # startswith,istartswith, endswith, iendswith,

        # order by
        #
        # models.Tb1.objects.filter(name='seven').order_by('id')    # asc
        # models.Tb1.objects.filter(name='seven').order_by('-id')   # desc

        # group by
        #
        # from django.db.models import Count, Min, Max, Sum
        # models.Tb1.objects.filter(c1=1).values('id').annotate(c=Count('num'))
        # SELECT "app01_tb1"."id", COUNT("app01_tb1"."num") AS "c" FROM "app01_tb1" WHERE "app01_tb1"."c1" = 1 GROUP BY "app01_tb1"."id"

        # limit 、offset
        #
        # models.Tb1.objects.all()[10:20]

        # regex正则匹配,iregex 不区分大小写
        #
        # Entry.objects.get(title__regex=r'^(An?|The) +')
        # Entry.objects.get(title__iregex=r'^(an?|the) +')

        # date
        #
        # Entry.objects.filter(pub_date__date=datetime.date(2005, 1, 1))
        # Entry.objects.filter(pub_date__date__gt=datetime.date(2005, 1, 1))

        # year
        #
        # Entry.objects.filter(pub_date__year=2005)
        # Entry.objects.filter(pub_date__year__gte=2005)

        # month
        #
        # Entry.objects.filter(pub_date__month=12)
        # Entry.objects.filter(pub_date__month__gte=6)

        # day
        #
        # Entry.objects.filter(pub_date__day=3)
        # Entry.objects.filter(pub_date__day__gte=3)

        # week_day
        #
        # Entry.objects.filter(pub_date__week_day=2)
        # Entry.objects.filter(pub_date__week_day__gte=2)

        # hour
        #
        # Event.objects.filter(timestamp__hour=23)
        # Event.objects.filter(time__hour=5)
        # Event.objects.filter(timestamp__hour__gte=12)

        # minute
        #
        # Event.objects.filter(timestamp__minute=29)
        # Event.objects.filter(time__minute=46)
        # Event.objects.filter(timestamp__minute__gte=29)

        # second
        #
        # Event.objects.filter(timestamp__second=31)
        # Event.objects.filter(time__second=2)
        # Event.objects.filter(timestamp__second__gte=31)

进阶操作

--------其他操作:F、Q、原生SQL

# extra
    #
    # extra(self, select=None, where=None, params=None, tables=None, order_by=None, select_params=None)
    #    Entry.objects.extra(select={'new_id': "select col from sometable where othercol > %s"}, select_params=(1,))
    #    Entry.objects.extra(where=['headline=%s'], params=['Lennon'])
    #    Entry.objects.extra(where=["foo='a' OR bar = 'a'", "baz = 'a'"])
    #    Entry.objects.extra(select={'new_id': "select id from tb where id > %s"}, select_params=(1,), order_by=['-nid'])

    # F
    #
    # from django.db.models import F
    # models.Tb1.objects.update(num=F('num')+1)


    # Q
    #
    # 方式一:
    # Q(nid__gt=10)
    # Q(nid=8) | Q(nid__gt=10)
    # Q(Q(nid=8) | Q(nid__gt=10)) & Q(caption='root')
    # 方式二:
    # con = Q()
    # q1 = Q()
    # q1.connector = 'OR'
    # q1.children.append(('id', 1))
    # q1.children.append(('id', 10))
    # q1.children.append(('id', 9))
    # q2 = Q()
    # q2.connector = 'OR'
    # q2.children.append(('c1', 1))
    # q2.children.append(('c1', 10))
    # q2.children.append(('c1', 9))
    # con.add(q1, 'AND')
    # con.add(q2, 'AND')
    #
    # models.Tb1.objects.filter(con)


    # 执行原生SQL
    #
    # from django.db import connection, connections
    # cursor = connection.cursor()  # cursor = connections['default'].cursor()
    # cursor.execute("""SELECT * from auth_user where id = %s""", [1])
    # row = cursor.fetchone()

其他操作

  

 5.2、联表操作(利用双下划线和 _set 将表之间的操作连接起来):

    一对一

 一对多   见博客第4点、

 多对多   时见   http://www.cnblogs.com/tangtingmary/p/7994148.html    第12点。

 

2.admin的配置(选修)                                                                                         

  admin是django强大功能之一,它能共从数据库中读取数据,呈现在页面中,进行管理。默认情况下,它的功能已经非常强大,如果你不需要复杂的功能,它已经够用,但是有时候,一些特殊的功能还需要定制,比如搜索功能,下面这一系列文章就逐步深入介绍如何定制适合自己的admin应用。

步骤:

1.在models中建几张表(类)

      生成同步数据库的脚本:python manage.py makemigrations  

                      同步数据库:  python manage.py migrate   

       注意:在开发过程中,数据库同步误操作之后,难免会遇到后面不能同步成功的情况,解决这个问题的一个简单粗暴方法是把migrations目录下

         的脚本(除__init__.py之外)全部删掉,再把数据库删掉之后创建一个新的数据库,数据库同步操作再重新做一遍。           

 

2.创建超级管理员:python manage.py createsuperuser设置好用户名和密码后便可登录http://127.0.0.1:8080/admin/啦!

3.在app01的admin.py中进行配置

一  认识ModelAdmin

   管理界面的定制类,如需扩展特定的model界面需从该类继承。

二 注册medel类到admin的两种方式:

     <1>   使用register的方法

1
admin.site.register(Book,MyAdmin)

     <2>   使用register的装饰器

1
@admin .register(Book)

三 掌握一些常用的设置技巧

  •     list_display:     指定要显示的字段
  •     search_fields:  指定搜索的字段
  •     list_filter:        指定列表过滤器
  •     ordering:       指定排序字段

实例:

from django.contrib import admin
from app01.models import *  #导入

# Register your models here.

admin.site.register(Author)#该语句用来注册
admin.site.register(Publisher)#该语句用来注册

class MyAdmin(admin.ModelAdmin):
    list_display = ("title","price","publisher")#定制显示哪些字段
    search_fields = ("title", "publisher")#搜索插件,可根据title和publisher来搜索
    list_filter = ("publisher",)#过滤器
    ordering = ("price",)#排序
    # fieldsets = [
    #     (None, {'fields': ['title']}),
    #     ('price information', {'fields': ['price', "publisher"], 'classes': ['collapse']}),
    # ]


admin.site.register(Book,MyAdmin)#该语句用来注册,且将MyAdmin与Book绑定

3、all()、values()、value_list()的对比                                                                  

一对多操作:
  获取所有的学生--------获取DB数据

class Classes(models.Model):
	caption = models.CharField(max_length=32)


class Student(models.Model):
	name = models.CharField(max_length=32)
	email = models.CharField(max_length=32,null=True)
	cls = models.ForeignKey('Classes')

 原理讲解:

.all()
result = models.Student.object.all()
# QuerySet类型, select * from ....
# [obj,obj,obj,],Django内部有一个模块,可以转换
# obj.name, obj.email, obj.cls_id, obj.cls.id, obj.cls.caption
.all().values()
result = models.Student.object.all().values('id','name','email', 'cls_id', 'cls__id', 'cls__caption')
# QuerySet, select id.. from ...
# list(result)  -> QuerySet -> list, json.dumps()
# [ {'id': 1, 'name': ...,},{'id': 1, 'name': ...,},{'id': 1, 'name': ...,},{'id': 1, 'name': ...,} ]

  

.all().value_list()
result = models.Student.object.all().value_list('id','name','email', 'cls_id', 'cls__id', 'cls__caption')
# QuerySet, select id.. from ...
# list(result)  -> QuerySet -> list, json.dumps()
# [ (1,'root'..),(1,'root'..),(1,'root'..),(1,'root'..),(1,'root'..) ]

  

# PS: QuerySet,无法直接使用json.dumps()序列化
# QuerySet
  - Django内部有一个模块,可以转换
  - values, value_list, 转换成 list(result), json.dumps

获取数据:三种形式
请求数据:form(url),ajax

all()、values()、value_list()-------------------在模版语言显示数据(URL,Form表单的形式)

def index(request):
	result = models.Student.object.all()
	return render(request, 'index.html', {'reuslt': result})
	
{% for item in result%}
	{
      {item.id}} - {
      {item.name}} - {
      {item.cls.id}} - {
      {item.cls.caption}}
{% endfor %}

  

def index(request):
	result = models.Student.object.all().values('id','name','email', 'cls_id', 'cls__id', 'cls__caption')
	# [ {'id': 1, 'name': ...,},{'id': 1, 'name': ...,},{'id': 1, 'name': ...,},{'id': 1, 'name': ...,} ]
	return render(request, 'index.html', {'reuslt': result})
	
{% for item in result%}
	{
      {item.id}} - {
      {item.name}} - {
      {item.cls__id}} - {
      {item.cls__caption}}
{% endfor %}

  

def index(request):
	result = models.Student.object.all().value_list('id','name','email', 'cls_id', 'cls__id', 'cls__caption')
	# QuerySet, select id.. from ...
	# list(result)  -> QuerySet -> list, json.dumps()
	# [ (1,'root'..),(1,'root'..),(1,'root'..),(1,'root'..),(1,'root'..) ]
	return render(request, 'index.html', {'reuslt': result})
	
{% for item in result%}
	{
      {item.0}} - {
      {item.1}} - {
      {item.4}} - {
      {item.5}}
{% endfor %}

  

- Ajax请求

a. 
	$(function(){
		   $.ajax...
	})

b.
	def index(request):
		result = models.Student.object.all()
		# Django内部序列化工具
		result = Django内部序列化工具
		
		return HttpResponse(result)
		
		
	def index(request):
		result = models.Student.object.all().values('id','name','email', 'cls_id', 'cls__id', 'cls__caption')
		result = list(result)
		result_str = json.dumps(result)
		
		return HttpResponse(result_str)
		
	def index(request):
		result = models.Student.object.all().value_list('id','name','email', 'cls_id', 'cls__id', 'cls__caption')
		result = list(result)
		result_str = json.dumps(result)
		
		return HttpResponse(result_str)
		
c. 
	$.each(function(){
		# 构建html标签
		# 将标签添加到HTML指定位置
	
	})

4、数据库操作(一对一、一对多、多对多)                                                                  

  - 单表操作

- all
- filter   (过滤)内部也可以放字典
	models.tb.objects.filter(id=123)
	
	dic = {'id': 123, 'age__gt': 3}
	models.tb.objects.filter(**dic)
- count
- order_by
...

  - 一对多

1.(ForeignKey时,默认和主键id进行关联,也可以自定义nid(必须时唯一的unique=True)进行关联)
如:models.ForeignKey("Province", to_filed='nid')自定义和Province的nid进行关联

省
# id     name
  1      河北
  2      广东
  3      山东
class Province(models.Model):
	name = models.CharField(max_length=32,)
	# nid = models.Intergar(unique=True) # 唯一

市				  省
# id     name      pro
   1     东莞       2
   2     深圳       2
   3     惠州       2
   4     河源       2
   5     泰安       3
   6     青岛       3
   7     济南       3
   8     张家口     1
   9     邢台       1
class City(models.Model):
	name = models.CharField(max_length=32)
	pro = models.ForeignKey("Province", to_filed='id')(外键列,在数据库创建时的名称,自动会是pro_id,即【列名_id】)

---------------------------------------------------------------------
1、 正向查找(基于市查省份表,foreignkey在city里)	(_ _id)
	result = models.City.objects.all()
	result[0].pro.name	跨表查询
	
	models.City.objects.all().values('id','name','pro_id','pro__id','pro__name')	内部为字典#'pro__id','pro__name'是关联表里面的id和name
	models.City.objects.all().values_list('id','name','pro_id','pro__id','pro__name')	内部为元祖

2、反向查找	(_set)
	result = models.Province.objects.values('id','name', 'city__name')

	result = models.Province.objects.all()
	result[0]                  #  获取河北
	result[0].city_set.all()   #  获取河北下的所有市 张家口 、邢台
	
	for pro in result:
		a1 = pro.id
		a2 = pro.name
		a3 = pro.city_set.all()
		print(a1,a2,a3)
	=====> 多对多即使基于一对多来构造

  

  -多对多(自己创建关系表或Django默认帮我们创建)

class Book(models.Model):
    name =models.CharField(max_length=32)

class Author(models.Model):
    name = models.CharField(max_length=32)
    m = models.ManyToManyField('Book')

  

1.自己创建第三张表(后续操作方便):

	class Book(models.Model):
		name = ..
		
	class Author(models.Model):
		name = ..
		
	class A_to_B(models.Model): #这种方式就不能使用Django自带的clear、remove等方法
		bid = ForeignKey(Book)
		aid = ForeignKey(Author)
	====> 所有操作:化简为 单表操作 和 一对多操作
	
2.默认生成第三张表
	只能间接对第三张表进行操作(因为没有第三张表的类)
	
	# 正向查找
	# obj,人,金鑫
	# obj = models.Author.objects.get(id=1)
	#
	# # 金鑫所有的著作全部获取到
	# obj.m.all()

	# 反向查找
	# 金品买
	# obj = models.Book.objects.get(id=1)
	# # 金鑫,吴超
	# obj.author_set.all()
	# 10
	# author_list = models.Author.objects.all()
	# for author in author_list:
	#     print(author.name,author.m.all())

	# author_list = models.Author.objects.values('id','name','m', "m__name")
	# for item in author_list:
	#     print(item['id'],item['name'],'书籍ID:',item['m'],item['m__name'])

	# 添加

	# obj = models.Author.objects.get(id=1)
	# 第三张表中增加一个对应关系
	# 增加
	# obj.m.add(5)
	# obj.m.add(5,6)
	# obj.m.add(*[4,5])
	# 删除
	# obj.m.remove(5)
	# obj.m.remove(5,6)
	# obj.m.remove(*[5,6])
	# 清空
	# obj.m.clear()
	# 更新
	# obj.m.set([1,2,3])

	# 反向操作
	# obj = models.Book.objects.get(id=1)
	# obj.author_set.add(1)
	# obj.author_set.add(1,2,3,4)
	# ... 

补充:

select_related 和 prefetch_related 函数优化查询

1. select_related()

对于一对一字段(OneToOneField)和外键字段(ForeignKey),可以使用select_related 来对QuerySet进行优化

作用和方法

在对QuerySet使用select_related()函数后,Django会获取相应外键对应的对象,从而在之后需要的时候不必再查询数据库了。以上例说明,如果我们需要打印数据库中的所有市及其所属省份,最直接的做法是:

>>> citys = City.objects.all()  
>>> for c in citys:  
...   print c.province  
...  

这样会导致线性的SQL查询,如果对象数量n太多,每个对象中有k个外键字段的话,就会导致n*k+1次SQL查询。在本例中,因为有3个city对象就导致了4次SQL查询。

--------------------------------------------
如果我们使用select_related()函数:
>>> citys = City.objects.select_related().all()  
>>> for c in citys:  
...   print c.province  
...  

就只有一次SQL查询,显然大大减少了SQL查询的次数:

这里我们可以看到,Django使用了INNER JOIN来获得省份的信息。

小结

1.select_related主要针一对一和多对一关系进行优化。
2.select_related使用SQL的JOIN语句进行优化,通过减少SQL查询的次数来进行优化、提高性能。
3.可以通过可变长参数指定需要select_related的字段名。也可以通过使用双下划线“__”连接字段名来实现指定的递归查询。没有指定的字段不会缓存,没有指定的深度不会缓存,如果要访问的话Django会再次进行SQL查询。
4.也可以通过depth参数指定递归的深度,Django会自动缓存指定深度内所有的字段。如果要访问指定深度外的字段,Django会再次进行SQL查询。
5.也接受无参数的调用,Django会尽可能深的递归查询所有的字段。但注意有Django递归的限制和性能的浪费。
6.Django >= 1.7,链式调用的select_related相当于使用可变长参数。Django < 1.7,链式调用会导致前边的select_related失效,只保留最后一个。

2. prefetch_related()

对于多对多字段(ManyToManyField)和一对多字段,可以使用prefetch_related()来进行优化。或许你会说,没有一个叫OneToManyField的东西啊。实际上 ,ForeignKey就是一个多对一的字段,而被ForeignKey关联的字段就是一对多字段了。

作用和方法

prefetch_related()和select_related()的设计目的很相似,都是为了减少SQL查询的数量,但是实现的方式不一样。后者是通过JOIN语句,在SQL查询内解决问题。但是对于多对多关系,使用SQL语句解决就显得有些不太明智,因为JOIN得到的表将会很长,会导致SQL语句运行时间的增加和内存占用的增加。若有n个对象,每个对象的多对多字段对应Mi条,就会生成Σ(n)Mi 行的结果表。prefetch_related()的解决方法是,分别查询每个表,然后用Python处理他们之间的关系。继续以上边的例子进行说明,如果我们要获得张三所有去过的城市,使用prefetch_related()应该是这么做:

最佳实践:

1.prefetch_related主要针一对多和多对多关系进行优化。
2.prefetch_related通过分别获取各个表的内容,然后用Python处理他们之间的关系来进行优化。
3.可以通过可变长参数指定需要select_related的字段名。指定方式和特征与select_related是相同的。
4.在Django >= 1.7可以通过Prefetch对象来实现复杂查询,但低版本的Django好像只能自己实现。
5.作为prefetch_related的参数,Prefetch对象和字符串可以混用。
6.prefetch_related的链式调用会将对应的prefetch添加进去,而非替换,似乎没有基于不同版本上区别。
7.可以通过传入None来清空之前的prefetch_related。

小结

  1. 因为select_related()总是在单次SQL查询中解决问题,而prefetch_related()会对每个相关表进行SQL查询,因此select_related()的效率通常比后者高。
  2. 鉴于第一条,尽可能的用select_related()解决问题。只有在select_related()不能解决问题的时候再去想prefetch_related()。
  3. 你可以在一个QuerySet中同时使用select_related()和prefetch_related(),从而减少SQL查询的次数。
  4. 只有prefetch_related()之前的select_related()是有效的,之后的将会被无视掉。

 

 

 

 

5、HttpResponse和render的对比                                                                                

HttpResponse,传入的数据,直接返回给用户

render(request,'a.html', {'k':123})
  1、找到html模版,open函数打开获取到内存
  2、Django的模版引擎: html模版的内容 + 数据 => 渲染(替换) ===》 最终字符串
  3、HttpResponse(最终字符串)

6、补充

a.
def index(request):
  request.POST.get('k')
  # checkbox--name相同,、select(多选时)可用getlist
  request.POST.getlist('k')

b.模板语言里面(.html)调用方法不需要加括号

 

 

参考博客:

http://www.cnblogs.com/wupeiqi/articles/5246483.html

http://www.cnblogs.com/yuanchenqi/articles/6083427.html

转载于:https://www.cnblogs.com/tangtingmary/p/7954738.html

你可能感兴趣的:(python,数据库,json)