用简单的英语,什么是Django通用视图?

时间:2010-03-13 06:09:35

标签: python django django-generic-views

本页的前两段解释说,通用观点应该让我的生活更轻松,更少单调,让我对女性更具吸引力(我编写了最后一篇):

https://docs.djangoproject.com/en/1.4/topics/generic-views/

我只是为了改善我的生活,但通用观点实际上做了什么?似乎有很多流行语被抛出,这比他们解释的要多得多。

Ruby on Rails中的通用视图是否类似于scaffolding?介绍中的最后一个要点似乎表明了这一点。这是一个准确的陈述吗?

3 个答案:

答案 0 :(得分:19)

Django通用视图只是视图函数(常规的旧python函数),它们在Web应用程序中非常常见。

根据您正在构建的应用程序类型,它们可以帮助您避免编写大量非常简单的视图。

例如,direct_to_template通用视图只是使用RequestContext呈现模板(这意味着模板可以访问请求的信息,如当前用户等)。

举个简单的例子,你可以写下这样的东西:

# urls.py
url('^some-url/$', some_view)

# views.py
def some_view(request):
    return render_to_response('template_name.html', context_instance=RequestContext(request))

就这样:

# urls.py
url('^some-url/$', direct_to_template, {'template': 'template_name.html'})

# views.py doesn't need any code for this view anymore

对于常见操作,例如“显示模型列表”或“将模型添加到数据库”,还有更复杂的通用视图。

此外,由于通用视图只是函数,因此当您需要与通用案例略有不同的内容时,可以在自己的视图函数中调用它们来执行“大部分工作”。

答案 1 :(得分:4)

通用视图允许您编写更短的代码。

比较

from django.http import HttpResponse, HttpResponseRedirect, Http404
from django.shortcuts import render_to_response, get_object_or_404, redirect
from myapp.models import Context

def edit(request, item_id):
    object = get_object_or_404(Context, pk=item_id)

    if request.method == 'POST':
        form = ContextForm(request.POST, instance=object)
        if form.is_valid():
            form.save()
            return redirect('myapp-context-index')
    else:
        form = ContextForm(instance=object)

    return render_to_response("myapp/context/edit.html", {'object': object, 'form': form})

使用:

from django.core import urlresolvers
from django.views.generic.create_update import update_object
from myapp.models import Context

def edit(request, item_id):    
    return update_object(request,
        object_id=item_id,              
        form_class=ContextForm,            
        template_name="myapp/context/edit.html",
        post_save_redirect=urlresolvers.reverse("myapp-context-index")
    )

与普通视图一样,它们只是普通的功能。如果你愿意,可以在URLconf中完全配置视图,通过我发现这个用法更加清晰。

作为奖励,您还可以获得:

  • 登录身份验证检查(通过login_required=True
  • 来自django.contrib.messages的成功状态消息。
  • 检查错误的代码较少。
  • 当您提供ModelForm参数而不是model时,默认为form_class

template_name默认为“appname / model_form.html”,但这对我来说太过分了。


以下是他们共享的表单类:

class ContextForm(forms.ModelForm): 
    """The form for a context"""
    class Meta:
        model = Context
        exclude = ('collection',)

    def save(self, commit=True):
        """Overwritten save to force collection_id to a value"""
        model = super(ContextForm, self).save(commit=False)
        model.collection_id = 1
        if commit:
            model.save()
        return model

答案 2 :(得分:2)

回答第二个问题:不,通用视图与RoR中的脚手架无关。顾名思义,脚手架类似于代码生成。通用视图是另一回事。

我对通用视图的主要用法是更基本的render_to_response函数的更高级别替换。这是您使用render_to_response编写简单视图的方法:

def my_view(request):
    return render_to_response('my_template.html')

但这是非常基本的!例如,模板将无法访问请求上下文,除非您明确地传递它。

我更喜欢使用通用视图:

def my_view(request):
    return direct_to_template(request, template='my_template.html')

现在请求上下文将被传递!这只是一个开始。例如,当您想要显示列表或详细视图时,通用视图会派上用场。他们将管理查询数据库,并向用户发送消息等。

因此,通用视图是高级函数,可帮助您从视图创建响应。