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('失败')
posted @ 2019-02-26 20:32  small_caier  阅读(264)  评论(0编辑  收藏  举报