Django框架基础篇 -- 请求HttpRequest 知识点 -- 提取不同类型数据的方式

目录

1.提取查询字符串数据(query string)

2. 提取请求体中数据

2.1. 表单类型请求体数据(Form Data)

2.2. 非表单类型请求体数据(Non-Form Data): JSON 数据

3. URL路径参数 -->> 提取URL路径中的特定部分数据

4. 请求头


Django的请求 HttpRequest

  • 首先,我们要明确 HttpRequest 的用处: 当用户发送请求时携带参数后端需要使用,而不同的发送参数的方式对应不同的提取参数的方式
  • 上述就是我们要使用 HttpRequest 做的目的,提取不同发送的参数(后端服务器要获取前端发送过来的参数)
  • 关于请求对象的 QueryDict 对象

QueryDict 对象是 django 中的一种特殊的字典。

QueryDict 对象可以存储一对一类型数据,也可以存储一对多类型数据。

获取 QueryDict 中的数据:  如果获取一个值,调用 QueryDict.get( key )来获取。

                                          如果获取多个值,调用 QueryDict.getlist( key ) 获取这个键对应的所有值, 存放在列表中返回。

注意点: 对于一对多类型数据来讲,如果获取一个值的时候,它是获取列表中的最后一个值,而不是其它值。


1.提取查询字符串数据(query string)

  • 获取请求路径中的查询字符串参数,形如: ? key1=value1 & key2=value2
  • 通过 request.GET 属性获取,并返回一个 QueryDict 类型的对象

案例:

基本流程:

1. 创建一个Django新的子应用 ;创建命名: django-admin startapp Request_1 

2. 注册子应用,在总工程下的settings.py 下的 INSTALLED_APPS 列表中添加 "Request_1"

3. 在视图(views)中编写 提取查询字符串数据 代码

from django.shortcuts import render
from django.views import View
from django import http

# Create your views here.

"""
    # 1. 查询字符串数据(query string):,一般是问号后面的数据
    形如: ?key1=value1&key2=value2
    比如: http://www.meiduo.site/list/115/1/?sort=price 中的 ?sort=price
"""
"""重要提示: 提取查询字符串参数不区分请求方式,即使客户端进行POST方式的请求,依然可以通过request.GET获取请求中的查询字符串参数。"""

# @案例代码:
""" 
    用户访问 -->> http://127.0.0.1:8000/querystring/?name=zxc&age=18 
    我们需要提取 ?name=zxc&age=18
"""
class QSParamView(View):
    """测试 提取查询字符串参数
    用户访问 -->> http://127.0.0.1:8000/querystring/?name=zxc&age=18
    """
    def get(self,request):
        # 获取查询字符串参数 name、age
        name = request.GET.get("name")
        age = request.GET.get("age")


        return http.HttpResponse(content=f"查询字符串参数:{name} ; {age}")

    def post(self,request):

        name = request.GET.get("name","Mr.lin")
        age = request.GET.get("age",18)

        return http.HttpResponse(content=f"查询字符串参数:{name} ; {age}")


4. 注册子路由,在我们的子应用下创建一个urls.py文件,写入一下代码

from django.urls import path
from . import views


# 注册子路由
urlpatterns = [
    # 测试提取查询字符串参数:http://127.0.0.1:8000/querystring/?name=zxc&age=18
    path("querystring/", views.QSParamView.as_view())
]

5. 注册总路由,在我们的总工程下的 urls.py 的 urlpatterns 列表中注册,代码如下

from django.contrib import admin
from django.urls import path,include

urlpatterns = [
    # 测试提取查询字符串参数:http://127.0.0.1:8000/querystring/?name=zxc&age=18
    path("",include("Request_Response.urls"))
]

6. 尝试运行一下,运行命令: python mange.py runserver,是否成功。


2. 提取请求体中数据

  • 可以发送的请求体数据的请求方式有: POST、 PUT、 PATCH 、DELETE。
  • 常见的请求体数据格式有:  表单类型数据 和 JSON字符串类型,我们要区别对待。

2.1. 表单类型请求体数据(Form Data)

   -->> request.POST.get() 属性获取,并返回 QueryDict 对象

案例:

1. 代码如下(写在views.py 中)

class FormDtataParamView(View):
    """
    测试提取表单类型的请求体参数
    http://127.0.0.1:8000/formdata/
    """

    def post(self,request):
        # 获取表单类型请求体参数中的username、password
        # 使用 request.POST.get() 来获取
        username = request.POST.get("username","Mr.lin")
        password = request.POST.get("password","123")

        return http.HttpResponse(content=f"提取到的表单类型请求体的参数:username:{username};password:{password}")
    # 结果:提取到的表单类型请求体的参数:Mr.lin;123

2.  注册子路由(列表中最后一行代码才是哦),切记:注册一次总路由,就可以不用注册了,子路由需要注册一下

from django.urls import path
from . import views


# 注册子路由
urlpatterns = [
    # 测试提取查询字符串参数:http://127.0.0.1:8000/querystring/?name=zxc&age=18
    path("querystring/", views.QSParamView.as_view()),

    #测试提取表单类型的请求体参数 http://127.0.0.1:8000/formdata/
    path("formdata/",views.FormDtataParamView.as_view()),
]

3. 尝试运行: 

        结果: 提取到的表单类型请求体的参数:Mr.lin;123

 

2.2. 非表单类型请求体数据(Non-Form Data): JSON 数据

  • 重要知识点:  

1. 非表单类型请求体数据,Django无法解析,通过 request.body 属性获取最原始的请求体数据  -->> json_str = request.body

2. JSON数据 进行解析 -->> 使用 json模块 将 原始的JSON数据 转 字典(dict) -->> json_dict = json.loads(json_str)

3. 提取JSON数据中的参

变量1 = json_dict.get("key1")  # key1 表示 json_dict 中的 key
变量2 = json_dict.get("key2")  # key2 表示 json_dict 中的 key

4. request.body 获取原始数据类型是: bytes类型


案例:

  • 需求:

1. 用户使用 post请求访问 并 携带JSON数据如下

{
    "username":"Mr.lin",
    "password":"123"
}

 2. 提取 JSON数据 代码如下(写在views.py 中)

class JSONParamView(View):
    """测试提取非表单类型请求体参数
    http://127.0.0.1:8000/json/
    """
    def post(self,request):
        # 获取请求体中原始的JSON数据
        json_str = request.body
        # 使用json模块将原始的JSON数据转字典
        json_dict = json.loads(json_str)

        # 提取JSON数据中的参数
        username = json_dict.get("username")
        password = json_dict.get("password")

        return http.HttpResponse(content='非表单类型请求体参数:%s--%s' % (username, password))

3. 子路由(最后一行代码才是哦

from django.urls import path
from . import views


# 注册子路由
urlpatterns = [
    # 测试提取查询字符串参数:http://127.0.0.1:8000/querystring/?name=zxc&age=18
    path("querystring/", views.QSParamView.as_view()),

    #测试提取表单类型的请求体参数 http://127.0.0.1:8000/formdata/
    path("formdata/",views.FormDtataParamView.as_view()),

    # 测试提取非表单类型请求体参数  http://127.0.0.1:8000/json/
    path("json/",views.JSONParamView.as_view()),
]

4. 运行结果:

3. URL路径参数 -->> 提取URL路径中的特定部分数据

  • 在 path() 或者 re_path() 都可以提取路径参数

url路径参数,我们需要在子路由urls.py中提取  -->> <int:age> 就是我们提取的路径参数 18

注意点: 提取的路径参数时,使用的关键字,必须和视图(views)中的参数名一样 -->> <int:age>/ 的age == views 中的 age

  • <int:age>是什么?为什么要这样使用?-->> 路由转换器
  • 路由转换器:就是django默认封装的正则表达式,是用于path()

         默认的转换器的位置: django.urls.converters.py

DEFAULT_CONVERTERS = {
    'int': IntConverter(), # 匹配正整数,包含0
    'path': PathConverter(), # 匹配任何非空字符串,包含了路径分隔符
    'slug': SlugConverter(), # 匹配字母、数字以及横杠、下划线组成的字符串
    'str': StringConverter(), # 匹配除了路径分隔符(/)之外的非空字符串,这是默认的形式
    'uuid': UUIDConverter(), # 匹配格式化的uuid,如 075194d3-6885-417e-a8a8-6c931e272f00
}

案例1

需求1:  “http://127.0.0.1:8000/url_param1/18/” 取出url路径中的数据 18

流程:

1.  注册子路由 并 获取路径中的数据(列表中最后一行代码才是哦)

from django.urls import path
from . import views


# 注册子路由
urlpatterns = [
    # 测试提取查询字符串参数:http://127.0.0.1:8000/querystring/?name=zxc&age=18
    path("querystring/", views.QSParamView.as_view()),

    #测试提取表单类型的请求体参数 http://127.0.0.1:8000/formdata/
    path("formdata/",views.FormDtataParamView.as_view()),

    # 测试提取非表单类型请求体参数  http://127.0.0.1:8000/json/
    path("json/",views.JSONParamView.as_view()),

    #测试path()提取普通路径参数 http://127.0.0.1:8000/url_param1/18/
    path("url_param1/<int:age>/",views.URLParam1View.as_view())  # <int:age>/ 就是我们提取的路径参数 18
]

2. 在views.py中编写接收 路径参数 代码

# 提取 路径中的 18
class URLParam1View(View):
    """测试path()提取普通路径参数
    http://127.0.0.1:8000/url_param1/18/
    """
    def get(self,request,age):
        """
        :param age: 子路由提取的关键字参数
        """
        print("测试path()提取普通路径参数:%s" % age)
        return http.HttpResponse("测试path()提取普通路径参数:%s" % age)

3. 运行结果 测试path()提取普通路径参数:18

案例2 (这里有两个方法 path() and re_path() 获取url 路径中的电话号码,随便挑选一个用就OK,两种方法实现的是同一个任务)

需求2:"http://127.0.0.1:8000/url_param2/18500001111/" -->> 提取url 路径中的电话号码 18500001111

>> 问题产出: 在实现需求2中,我们要注意一个点:在默认的路由转换器中,未实现手机号码的路由转换器,所以无法直接使用默认的路由转换器进行转换。

>> 解决方案:

方案1: 使用path()的时候需要我们自定义路由转换器,这里稍微复杂一点,注意理清思路。

  • 自定义路由转换器流程:

1. 在总工程目录下,随便新建 converters.py 文件,写入以下代码

class MobileConverter:
  """自定义路由转换器:匹配手机号"""
  # 匹配手机号码的正则
  regex = '1[3-9]\d{9}'

  def to_python(self, value):
      # 将匹配结果传递到视图内部时使用
      return int(value)

  def to_url(self, value):
      # 将匹配结果用于反向解析传值时使用
      return str(value)

2. 在总路由中注册已经写好的自定义路由转换器(仔细看代码注释)

from django.contrib import admin
from django.urls import path,include,register_converter

from converters import MobileConverter

# 注册自定义路由转换器
# register_converter(自定义路由转换器, '别名')
register_converter(MobileConverter,'mobile')

urlpatterns = [
    # 自带的后台管理系统的总路由:可以忽略
    path('admin/', admin.site.urls),
]

3. 在views.py中写入以下代码

# 例如2: http://127.0.0.1:8000/url_param2/18500001111/
# 提取电话号码 18500001111
class URLParam2View(View):
    """测试path()中自定义路由转换器提取路径参数:手机号
    http://127.0.0.1:8000/url_param2/18500001111/
    """
    def get(self,request,phone_num):
        """
        :param phone_num: 路由提取的关键字参数
        """
        print("测试path()提取路径参数手机号:%s" % phone_num)
        return http.HttpResponse(content="测试path()提取路径参数手机号:%s" % phone_num)

4. 注册子路由 并 获取路径中的数据(列表中最后一行代码才是哦)

from django.urls import path
from . import views


# 注册子路由
urlpatterns = [
    # 测试提取查询字符串参数:http://127.0.0.1:8000/querystring/?name=zxc&age=18
    path("querystring/", views.QSParamView.as_view()),

    #测试提取表单类型的请求体参数 http://127.0.0.1:8000/formdata/
    path("formdata/",views.FormDtataParamView.as_view()),

    # 测试提取非表单类型请求体参数  http://127.0.0.1:8000/json/
    path("json/",views.JSONParamView.as_view()),

    #测试path()提取普通路径参数 http://127.0.0.1:8000/url_param1/18/
    path("url_param1/<int:age>/",views.URLParam1View.as_view()),  # <int:age>/ 就是我们提取的路径参数 18

    # 测试path()中自定义路由转换器提取路径参数:手机号 http://127.0.0.1:8000/url_param2/18500001111/
    path("url_param2/<mobile:phone_num>/", views.URLParam2View.as_view()), # <mobile:phone_num>/ 就是我们提取的18500001111
]

5. 运行结果:

方案2: 使用re_path()提取路径参数。(注意点: 正则匹配时,严格按照以 ^开头,以 $结尾)

  • 基本流程

1. 在views.py(视图)中写入一下代码

class URLParam3View(View):
    """测试re_path()提取路径参数
    http://127.0.0.1:8000/url_param3/18500001111/
    """

    def get(self, request, phone_num):
        """
        :param phone_num: 路由提取的关键字参数
        """
        return http.HttpResponse('测试re_path()提取路径参数:%s' % phone_num)

2. 注册子路由 并 获取路径中的数据(列表中最后一行代码才是哦)

>> 知识小天窗:

            >> 解析为什么使用这样的正则匹配(?P<phone_num>1[0-9]\d{9}) ?

            >> 1. 我们获取路径中的参数时,我们需要一个关键字来接受手机号,而这个关键字是作为给视图(views)传参用的,

                     所以我们需要给手机号码起别名 ?P<phone_num>

            >> 2. 需要用正则表达式中的分组来包裹只属于电话号码的部分 (?P<phone_num>1[0-9]\d{9})

from django.urls import path,re_path
from . import views


# 注册子路由
urlpatterns = [
    # 测试提取查询字符串参数:http://127.0.0.1:8000/querystring/?name=zxc&age=18
    path("querystring/", views.QSParamView.as_view()),

    #测试提取表单类型的请求体参数 http://127.0.0.1:8000/formdata/
    path("formdata/",views.FormDtataParamView.as_view()),

    # 测试提取非表单类型请求体参数  http://127.0.0.1:8000/json/
    path("json/",views.JSONParamView.as_view()),

    #测试path()提取普通路径参数 http://127.0.0.1:8000/url_param1/18/
    path("url_param1/<int:age>/",views.URLParam1View.as_view()),  # <int:age>/ 就是我们提取的路径参数 18

    # 测试path()中自定义路由转换器提取路径参数:手机号 http://127.0.0.1:8000/url_param2/18500001111/
    path("url_param2/<mobile:phone_num>/", views.URLParam2View.as_view()), # <mobile:phone_num>/ 就是我们提取的18500001111

    # 测试re_path()提取路径参数 http://127.0.0.1:8000/url_param3/18500001111/
    re_path(r"^url_param3/(?P<phone_num>1[0-9]\d{9})/$", views.URLParam3View.as_view())
]

3. 运行结果:


4. 请求头

  1. 通过 request.META 属性获取 请求头headers 中的数据
  2. request.META类型为字典类型

>> 常见的请求头如:

  • CONTENT_LENGTH – The length of the request body (as a string).
  • CONTENT_TYPE – The MIME type of the request body.
  • HTTP_ACCEPT – Acceptable content types for the response.
  • HTTP_ACCEPT_ENCODING – Acceptable encodings for the response.
  • HTTP_ACCEPT_LANGUAGE – Acceptable languages for the response.
  • HTTP_HOST – The HTTP Host header sent by the client.
  • HTTP_REFERER – The referring page, if any.
  • HTTP_USER_AGENT – The client’s user-agent string.
  • QUERY_STRING – The query string, as a single (unparsed) string.
  • REMOTE_ADDR – The IP address of the client.
  • REMOTE_HOST – The hostname of the client.
  • REMOTE_USER – The user authenticated by the Web server, if any.
  • REQUEST_METHOD – A string such as "GET" or "POST".
  • SERVER_NAME – The hostname of the server.
  • SERVER_PORT – The port of the server (as a string).

具体使用:

class HeadersParamView(View):
    """测试提取请求头参数"""

    def get(self, request):
        # 获取请求头中文件的类型
        ret = request.META.get('CONTENT_TYPE')
        return http.HttpResponse('OK')

作为小白的我,希望大佬们多多支持,谢谢!!!

 

 

 

 

 

 

 

  • 2
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值