zoukankan      html  css  js  c++  java
  • 03 Django之视图函数

    一.Django的视图函数view

      一个视图函数(类),简称视图,是一个简单的Python函数(类),它接受WEB请求并返回Web响应.

      响应可以是一张网页的HTML内容,一个重定向,一个404错误,一个XML文档,或者一张图片.

      无论视图本身包含什么逻辑,都要返回响应.代码写在哪里也无所谓,只要它在你当前项目目录下面.除此之外没有更多要求了---可以说"没有什么神奇的地方".为了将代码放在某处,大家约定成俗将视图放置在项目(project)或应用程序(app)目录中的名为views.py的文件中.

      一个简单的视图

      下面是一个以HTML文档的形式返回当前日期和时间的视图:

    from django.http import HttpResponse
    import datetime
    
    def current_datetime(request):
        now = datetime.datetime(request):
        html = "<html><body>It is now %s.</body></html>" % now
        return HttpResponse(html)
    

      代码的逐行解释:

    1)首先从django.http模块导入HyypResponse类,以及Python的datetime库

    2)接着定义了current_datetime函数.它就是视图函数.每个视图函数都使用HttpResponse对象作为第一个参数,并且通常称为request.

     注意:视图函数的名称不重要;不需要用一个统一的命名方式来命名,以便让Django识别它我们将其命名为current_datetime,因为这个名称能够比较准确地反映出它实现的功能.

    3)这个视图会返回一个HttpResponse对象,其中包含生成的响应.每个视图函数都负责返回一个HttpResponse对象

     Django使用请求和响应对象来通过系统传递状态.

     当浏览器向服务端请求一个页面时,Django创建一个HttpResponse对象,该对象包含关于请求的元数据.然后,Django加载相应的视图,将这个HTTPResponse对象作为第一个参数传递给视图函数.

     每个视图负责返回一个HttpResponse对象.

      views.py(视图层),熟练掌握两个对象:请求对象(request)和响应对象(HTTPResponse)

    二.CBV和FBV

      FBV(function base views)  就是在视图中使用函数处理请求

      CBV(class base views)  就是在视图里使用类处理请求

      Python是一个面向对象的编程语言,如果只用函数来开发,有很多面向对象的优点就错失了(继承,封装,多态).所以Django在后来加入了Class-Based-VieW.可以让我们用类写view.

      优点:

      1.提高代码的复用性,可以使用面向对象的技术,比如Mixin(多继承)

      2.可以用不同的函数针对不同的HTTp方法处理,而不是通过很对if判断,提高代码的可读性

        如果我们要写一个处理GET方法的view,用函数写的话是下面这样

    from django.http import HttpResponse
      
    def my_view(request):
         if request.method == 'GET':
                return HttpResponse('OK')
    

      如果用class-based-view写的话,就是下面:

    from django.http import HttpResponse
    from  django.views import View
    
    class MyView(View):
        def get(self,request):
            return HttpRsponse('ok')
    

      Django的URL 是将一个请求分配给可调用函数的,而不是一个class.针对这个问题,class-based view提供了一个as_view()静态方法(也就是类方法),调用这个类方法,会创建一个类的实例,然后通过这个实例调用dispatch()方法,dispath()方法会根据request的method的不同调用相应的方法来处理request(如get(),或者post()),到这里,这些方法和function-based-view差不多了,要接收request,得到了一个response返回.返回方法没有定义,会抛出HttpResponseNotAllowed异常。

      注意:使用CBV时,urls.py中也做对应的修改:

    # urls.py
    from django.conf.urls import url
    from myapp.views import MyView #引入我们在views.py里面创建的类
      
    urlpatterns = [
         url(r'^index/$', MyView.as_view()),
    ]
    

      CBV传参,和FBV类似,有名分组,无名分组

       url写法:无名分组的

    url(r'^cv/(d{2})/', views.Myd.as_view(),name='cv'),
     url(r'^cv/(?P<n>d{2})/', views.Myd.as_view(name='xxx'),name='cv'),
    #如果想给类的name属性赋值,前提你的Myd类里面必须有name属性(类属性,
    定义init方法来接受属性行不通,但是可以自行研究一下,看看如何行通,意义不大),
    并且之前类里面的name属性的值会被覆盖掉

      类写法:

    class Myd(View):
        name = 'sb'
    
        def get(self,request,n):
            print('get方法执行了')
            print('>>>',n)
            return render(request,'cvpost.html',{'name':self.name})
    
        def post(self,request,n):
            print('post方法被执行了')
            return HttpResponse('post')

     四.给视图加装饰器

      使用装饰器装饰FBV

      FBV本身就是一个函数,就是Python通用装饰器的加法.

    def wrapper(func):
        def inner(*args,**kwargs):
            start_time = time.time()
            ret = func(*args,**kwargs)
            end_time = time.time()
            print("used:",end_time - start_time)
            return ret
        return inner
    
    #FBV版本添加班级
    @wrapper
    def add_class(request):
        if request.method == "POST":
            class_name = request.POST.get("calss_name")
            models.Classes.objects.create(name=class_name)
            return redirect("/class_list/")
        return render(request,"add_class.heml")
    

      使用装饰器装饰CBV

      类中的方法与独立函数不完全相同,因此不能直接将函数装饰器应用于类中的方法 ,我们需要先将其转换为方法装饰器。

      Django中提供了method_decorator装饰器用于将函数装饰器转换为方法装饰器。

    from django.views import View
    from django.utils.decorators import method_decorator
    
    class AddClass(View):
    
        @method_decorator(wrapper)
        def get(self, request):
            return render(request, "add_class.html")
    
        def post(self, request):
            class_name = request.POST.get("class_name")
            models.Classes.objects.create(name=class_name)
            return redirect("/class_list/")

     五.request对象

      当一个页面被请求时,Django就会创建一个包含本次请求原信息(请求报文中的请求行、首部信息、内容主体等)的HttpRequest对象。

      Django会将这个对象自动传递给响应的视图函数,一般视图函数约定俗成的使用request参数承接这个对象.

      Django会将这个对象自动传递给相应的视图函数,一般视图函数约定俗成的使用request参数承接这个对象.

      请求相关的常用值

    1)path_info   返回用户访问URL,不包括域名

    2)method      请求中使用的HTTP方法的字符串表示,全大写表示.

    3)get              包含所有Http GET参数的类字典对象

    4)post             包含所有HTTP POST参数类字典对象

    5)body     请求体,byte类型 request.POST的数据就是从body里面提取到的

       要处理表单数据的时候,推荐还是使用HttpRequest.POST.

      另外,我们还可以用Python的类文件方法去操作它.

     2.HttpRequest.path

      一个字符串,表示请求的路径组件(不含域名)。
      例如:"/music/bands/the_beatles/"
    3.HttpRequest.method
      一个字符串,表示请求使用的HTTP 方法。必须使用大写。
      例如:"GET"、"POST"

     六.响应  response

      响应对象有三种形式:

      (1) HTTPResponse()

      (2) render()

      (3)redirect()

      HTTPResponse()括号内直接跟一个具体的字符串作为响应体,比较直接很简单.

      Django shortcut function

        render()

        

        结合一个给定的模板和一个给定的上下文字典,并返回一个渲染后的HTTPResponse对象.

          参数:

            request: 用于生成相应的请求对象.

            template_name: 要使用的模板的完整名称,可选参数

            context:添加到模板上下文的一个字典。默认是一个空字典。如果字典中的某个值是可调用的,视图将在渲染模板之前调用它

       例子:

    from django.shortcuts import render
    
    def my_view(request):
        # 视图的代码写在这里
        return render(request, 'myapp/index.html', {'foo': 'bar'})

        redirect():给浏览器了一个30x的状态码

        参数是:

          1/ 一个模型,将调用模型的get_absolute_url() 函数

          2/ 一个视图,可以带有参数:将使用urlresolvers.revese来反向解析名称

          3/一个绝对的或相对的URL,将原封不动的作为重定向的位置

        默认返回一个临时的重定向;传递permanent=True可以返回一个永久的重定向.

        示例:

        可以用多种方法使用redirect()函数.

        传递一个视图的名称

    def my_view(request):
        ...
        return redirect('some-view-name', foo='bar')
    

        传递要重定向到的一个具体的网址

    def my_view(request):
        ...
        return redirect('/some/url/')
    

        看一个例子

            index.html文件,内容如下

    <!DOCTYPE html>
    <html lang="en">
    <head>
        <meta charset="UTF-8">
        <title>Title</title>
    </head>
    <body>
    <div>这是index页面</div>
    <h1>{{ name }}</h1>
    
    </body>
    </html>
    

            login.html文件,内容如下

    <!DOCTYPE html>
    <html lang="en">
    <head>
        <meta charset="UTF-8">
        <title>Title</title>
    </head>
    <body>
    
    <div>
        <form action="{% url 'xxx' %}" method="post">
            用户名:<input type="text" name="username">
            密码:<input type="password" name="password">
            <input type="submit">
        </form>
    
    </div>
    
    </body>
    </html>
    

            urls.py里面的内容:

    from django.shortcuts import render,HttpResponse,redirect
    
    def index(request):
        return render(request,'index.html',{'name':'ce'})
    
    def login(request):
        method = request.method
        
        if method == 'GET':
            return render(request,'login.html')
        else:
            username = request.POST.get('username')
            password = request.POST.get('password')
            if username == 'ce' and password == '123':
                return redirect('/index/')  #重新定向到/index/路径,这也是发送了一个请求,
    别忘了在上面引入这个redirect类,和render,HttpResponse在同一个地方引入 else: return HttpResponse('失败')

        上面几个文件搞好之后,我们重启Django项目,然后登陆页面的输入网址,注意,你输入的网址端口要和你启动的django项目的端口一样。

      但是如果我们在函数里面写的render来返回内容,两者有什么不同呢?

    from django.shortcuts import render,HttpResponse,redirect
    
    # Create your views here.
    
    def index(request):
    
        return render(request,'index.html',{'name':'chao'})
    
    def login(request):
        method = request.method
    
        if method == 'GET':
            return render(request,'login.html')
        else:
            username = request.POST.get('username')
            password = request.POST.get('password')
            if username == 'chao' and password == '123':
                return redirect('/index/')    #重定向到/index/路径,这也是发送一个请求,别忘了在上面引入这个redirect类,和render,HTTPResponse在一个地方引入
                #如果直接用render来返回页面,是一次响应就返回来页面,两者是有区别的,如果你用render返回index.html页面,那么这个页面里面的模板渲染语言里面需要的数据你怎么搞,如果这些数据就是人家index那个函数里面独有的呢,你怎么搞,有人可能就响了,我把所有的数据都拿过来不就行了吗,首先如果数据量很大的话,是不是都重复了,并且你想想如果用户登陆完成之后,你们有进行跳转,那么如果网速不太好,卡一下,你想刷新一下你的页面,你是不是相当于又发送了一个login请求,你刷新完之后,是不是还要让你输入用户名和密码,你想想是不是,所有咱们一般在登陆之后都做跳转。
                #redirect本质上也是一个HttpResponse的操作,看看源码就知道了
                # return HttpResponse('success')
            else:
                return HttpResponse('失败')
  • 相关阅读:
    在客户端判断上传文件大小(不支持opera)
    javascript冒泡排序(javascript算法学习)
    提高js性能方法(让js只判断一次)
    ie8本地预览报错的解决
    负载均衡获取客户端IP
    遍历变量里面所有的属性
    HttpWebrequest的使用
    C#调用EnyimMemcached
    MySQL国内省市直辖区
    Log4Net使用
  • 原文地址:https://www.cnblogs.com/a2534786642/p/10439944.html
Copyright © 2011-2022 走看看