在实际Python web开发过程中,我们经常会碰到这样的应用场景。当用户试图访问某个页面或评论某个页面时,我们会要求其先登录,然后在用户在登录后自动跳转到用户试图访问的页面。小编我今天就来总结下Django中如何实现用户登录后跳转回前一页的几种方法,希望对大家有所帮助。
方法一: 静态模板中手动添加next参数实现跳转
假如我们负责管理用户注册登录的app叫myaccount, 其包括下面4个urls。我们要稍后用到它们。
from django.urls import re_path
from . import views
app_name = 'myaccount'
urlpatterns = [
re_path(r'^account/register/$', views.register, name='account_register'),
re_path(r'^account/login/$', views.login, name='account_login'),
re_path(r'^user/(?P<pk>\d+)/profile/$', views.profile, name='profile'),
re_path(r'^logout/$', views.logout, name='account_logout'),
]
这时静态模板中我们可以手动添加next参数,指向跳转页面链接。比如下例代码中,用户登录后会自动跳转到添加文章页面(blog:article_create)。
{% if request.user.is_authenticated %}
<p> <a href="{% url 'blog:article_create' %}">添加文章</a></p>
{% else %}
<p>请<a href="{% url 'myaccount:account_login' %}?next=
{% url 'blog:article_create' %}">登录</a>后再添加文章。</p>
{% endif %}
{% endblock %}
上面代码等同于下段代码。区别在于一个使用了命名的url,一个使用了硬编码的url(小编并不推荐硬编码,因为其不利于传递参数)。
{% if request.user.is_authenticated %}
<p> <a href="{% url 'blog:article_create' %}">添加文章</a></p>
{% else %}
<p>请<a href="/account/login/?next={% url 'blog:article_create' %}">
登录</a>后再添加文章。</p>
{% endif %}
{% endblock %}
或者
{% if request.user.is_authenticated %}
<p> <a href="{% url 'blog:article_create' %}">添加文章</a></p>
{% else %}
<p>请<a href="/account/login/?next=/blog/article/create/">
登录</a>后再添加文章。</p>
{% endif %}
{% endblock %}
我们负责登录的视图login函数如下所示,该函数很重要的一件事就是处理通过next参数传递过来的跳转链接。当有next参数时,登录后跳转到next指向页面。如果没有next参数时,用户登录后跳转到profile页面。下面这段代码阅读性很高,请仔细体会。
def login(request):
next = request.GET.get('next', '')
if request.method == 'POST':
form = LoginForm(request.POST)
if form.is_valid():
username = form.cleaned_data['username']
password = form.cleaned_data['password']
user = auth.authenticate(username=username, password=password)
if user is not None and user.is_active:
auth.login(request, user)
if next == "":
return HttpResponseRedirect(reverse('myaccount:profile', args=[user.id]))
else:
return HttpResponseRedirect(next)
else:
# 登陆失败
return render(request, 'account/login.html', {'form': form,})
else:
form = LoginForm()
return render(request, 'account/login.html', {'form': form})
那么问题来了,本例中我们明确知道客户登录后要跳转到创建文章(article_create)页面,所以可以直接在模板中通过next指定跳转链接。如果我们不知道用户尝试访问的前一个页面是什么,我们该如何操作呢?这时我们可以用request.path获取前一页面,如下所示。这是目前最好的解决方案。
<p>请<a href="{% url 'myaccount:acount_login' %}?next=
{% firstof request.path '/' %}">登录</a>后再访问。</p>
如果你使用django-allauth负责用户注册登录, 你可以直接使用{% url 'account_login' %}获取登录链接,前面不需要加myaccount:。
方法二: 使用login_required装饰器
用户尝试访问的页面总是对应某个视图,这时我们可以给这个视图加上login_required装饰器。对于基于函数的视图(Functional Based View, 以def开头), 你只需要在函数头部加上@login_required即可。
from django.contrib.auth.decorators import login_required
@login_required
def article_detail(request, pk):
...
login_required主要做以下事情。使用login_required前请确保你在settings.ply里设置了LOGIN_URL。
如果你使用自定义的用户管理app实现登录,你还需要修改视图中的login函数来处理url传递过来的next参数(见前文login函数代码),否则不会实现跳转。如果你使用Django自带的Auth登录模块或Django-allauth登录模块,你则不需要编写自己的login函数来处理next参数,因为它们可以自动处理next传递的参数并实现跳转。
当你使用基于类的视图(Class Based Views, 以class开头)时,你需要按如下方法使用method_decorator的这个装饰器。其作用是把类伪装成函数,然后再应用login_required这个装饰器。
from django.utils.decorators import method_decorator
@method_decorator(login_required, name='dispatch')
class ArticleDetailView(DetailView):
model = Article
方法三: 视图views.py中实现跳转
在视图views.py中你可以使用HttpResponseDirect, redirect和reverse方法实现跳转(如下代码所示)。更多内容见Django基础(10): URL重定向的HttpResonseDirect, redirect和reverse方法详解。
def my_review(request):
if not request.user.is_authenticated():
return HttpResponseRedirect('/accounts/login/?next=/blog/article/15/')
else:
return HttpResponseRedirect('/blog/article/15/')
该方法的最大弊端是你需要知道用户之前尝试访问的urls是哪个,所以并不推荐。
小结
有多种方法可以实现用户在登录后跳转回前一页。小编我更推荐静态模板自定义next参数,其次是使用login_required装饰器。掌握了这些登录跳转的方法,我们就可以灵活地控制用户的访问权限和自由跳转了。