看我多久能学会Django(二):视图与网址

本文章来自自强学堂:http://www.ziqiangxuetang.com/django/django-tutorial.html

Django中网址是写在urls.py文件中,用正则表达式对应views.py中的一个函数(或者generic类),来个小栗子。

第一步:新建项目(project),项目名字:mysite

django-admin startproject  mysite 

运行成功后,我们看下目录结构:

.
├── manage.py
└── mysite
    ├── __init__.py
    ├── settings.py
    ├── urls.py
    └── wsgi.py

1 directory, 5 files

我们会发现执行命令后,新建了一个 mysite 目录,其中还有一个 mysite 目录,这个子目录 mysite 中是一些项目的设置 settings.py 文件,总的urls配置文件 urls.py 以及部署服务器时用到的 wsgi.py 文件, init.py 是python包的目录结构必须的,与调用有关。

我们在外层的mysite目录下创建应用(app)

第二步:新建一个应用(app),名称learn

python manage.py  startapp   learn 

我们可以看到mysite中多个一个 learn 文件夹,其中有以下文件。

# tree  learn 
learn
├── admin.py
├── apps.py
├── __init__.py
├── migrations
│   └── __init__.py
├── models.py
├── tests.py
└── views.py

1 directory, 7 files

把我们新定义的app加到settings.py中的INSTALL_APPS中
修改 mysite/mysite/settings.py

INSTALLED_APPS = [
    'django.contrib.admin',
    'django.contrib.auth',
    'django.contrib.contenttypes',
    'django.contrib.sessions',
    'django.contrib.messages',
    'django.contrib.staticfiles',
 
    'learn',
]

注意:新建的 app 如果不加到 INSTALL_APPS 中的话, django 就不能自动找到app中的模板文件(app-name/templates/下的文件)和静态文件(app-name/static/中的文件)

第三步:定义视图函数(访问页面时的内容)

我们在learn这个目录中,把views.py打开,修改其中的源代码,改成下面的

  #coding:utf-8
  from django.shortcuts import render
  
  # Create your views here.
  from  django.http import HttpResponse
  
  def index(request):
      return HttpResponse(u'你好,Django')

声明编码为utf-8, 因为我们在代码中用到了中文,如果不声明就报错.

引入HttpResponse,它是用来向网页返回内容的,就像Python中的 print 一样,只不过 HttpResponse 是把内容显示到网页上。

我们定义了一个index()函数,第一个参数必须是 request,与网页发来的请求有关,request 变量里面包含get或post的内容,用户浏览器,系统等信息在里面

第四步:定义视图函数相关的URL(网址) (即规定 访问什么网址对应什么内容)

我们打开 mysite/mysite/urls.py 这个文件, 修改其中的代码:

 from django.conf.urls import url
 from django.contrib import admin
 from learn import views as learn_views
 
 
 urlpatterns = [
     url(r'^$',learn_views.index),
     url(r'^admin/', admin.site.urls),
 ]

在终端上运行 python manage.py runserver 我们会看到类似下面的信息:

Performing system checks...

System check identified no issues (0 silenced).

You have 13 unapplied migration(s). Your project may not work properly until you apply the migrations for app(s): admin, auth, contenttypes, sessions.
Run 'python manage.py migrate' to apply them.
March 02, 2017 - 20:08:48
Django version 1.10.5, using settings 'mysite.settings'
Starting development server at http://0.0.0.0:8000/
Quit the server with CTRL-BREAK.

结果:

Paste_Image.png

Django 视图与网址进阶

栗子一:在网页上做加减法

1,采用 /add/?a=4&b=5

django-admin   startproject  addtest
cd  addtest
python manage.py  startapp  addapp

上面的步骤创建了项目并且添加了app,我们看下目录结构:

# tree   addtest/
addtest/
├── addapp
│   ├── admin.py
│   ├── apps.py
│   ├── __init__.py
│   ├── migrations
│   │   └── __init__.py
│   ├── models.py
│   ├── tests.py
│   └── views.py
├── addtest
│   ├── __init__.py
│   ├── __pycache__
│   │   ├── __init__.cpython-35.pyc
│   │   └── settings.cpython-35.pyc
│   ├── settings.py
│   ├── urls.py
│   └── wsgi.py
└── manage.py

4 directories, 14 files

首先修改settings.py文件,将我们的app加入INSTALLED_APPS。
接着修改addapp/views.py文件:

from django.shortcuts import render

from django.http import HttpResponse

# Create your views here.

def add(request):
    a = request.GET['a']
    b = request.GET['b']
    c = int(a)+int(b)
    return  HttpResponse(str(c))

最后修改addtest/urls.py

from addapp import  views as views_add

urlpatterns = [
    url(r'^add/$',views_add.add,name='add'),
    url(r'^admin/', admin.site.urls),
]

运行程序,并在浏览器中输入:http://127.0.0.1:8000/add/ 就可以看到

Paste_Image.png

之所以这样,是因为我们并没有传值进去,当我们在链接后面加上?a=4&b=5,即访问 http://127.0.0.1:8000/add/?a=4&b=5

Paste_Image.png

当我们改变a和b的值的时候,结果会随之改变。

栗子二:采用/add/3/4/ 这样的网址的方式

首先,我们修改addapp/views.py,添加一个新的函数add2():

from django.shortcuts import render

from django.http import HttpResponse

# Create your views here.

def add(request):
    a = request.GET['a']
    b = request.GET['b']
    c = int(a)+int(b)
    return  HttpResponse(str(c))

def add2(request,a,b):
    c = int(a)+int(b)
    return  HttpResponse(str(c))

然后修改addtest/urls.py

from addapp import  views as views_add

urlpatterns = [
    url(r'^add/$',views_add.add,name='add'),
    url(r'^add/(\d+)/(\d+)/$',views_add.add2,name='add2'),  #修改了之一行
    url(r'^admin/', admin.site.urls),
]

我们可以看到网址中多了 (\d+), 正则表达式中 \d 代表一个数字,+ 代表一个或多个前面的字符,写在一起 \d+ 就是一个或多个数字,用括号括起来的意思是保存为一个子组,每一个子组将作为一个参数,被views.py中的对应视图函数接收。

运行之后,访问 http://127.0.0.1:8000/add/3/4/ 结果为:

Paste_Image.png

Django URL name详解

在上面的urls.py中 url(r'^add/(\d+)/(\d+)/$',views_add.add2,name='add2'),这里的name是用来做什么的呢?
简单的讲,name可以用在templates,models,views ......中得到对应的网址,相当于给网址起了个名字,只要名字不变,网址变了也能通过名字获取到。

什么鬼,还是不明白,看看栗子吧,首先新建一个首页的视图和url

1,修改addapp/views.py

from django.shortcuts import render

from django.http import HttpResponse

# Create your views here.
def index(request):
    return render(request,'home.html') #render是渲染模板

还有之前我们已经做过的,把addapp加到settings.py 中。这样,使用render的时候,Django 会自动找到 INSTALLED_APPS 中列出的各个 app 下的 templates 中的文件。

2,新建templates文件

我们在addapp中新建一个templates文件夹,在templates中新建一个home.html
在home.html中写入以下内容:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Django</title>
</head>
<body>
<a href="/add/4/5">计算4+5</a>
</body>
</html>

接着修改urls.py:

from addapp import  views as views_add

urlpatterns = [
    url(r'^$',views_add.index,name='home'),
    url(r'^add/$',views_add.add,name='add'),
    url(r'^add/(\d+)/(\d+)/$',views_add.add2,name='add2'),
    url(r'^admin/', admin.site.urls),
]

最后我们运行程序,在浏览器中访问 http://127.0.0.1:8000/ 可以看到:

Paste_Image.png

我们计算加法的时候用的是 /add/4/5/ ,后来需求发生变化,比如改成 /4_add_5/,但在网页中,代码中很多地方都写死的 /add/4/5/,比如模板中可能是这么写的 :

<a href="/add/4/5/">计算 4+5</a>

如果这样写“死网址”,会使得在改了网址(正则)后,模板(template),视图(views.py,用以用于跳转),模型(models.py,可以用用于获取对象对应的地址)用了此网址的,都得进行相应的更改。

下面我们讨论,如何修改这种问题:
首先我们先说一下如何用 Python 代码获取对应的网址(可以用在 views.py,models.py等各种需要转换得到网址的地方):

from django.urls import reverse
>>> reverse('add2', args=(4,5))
u'/add/4/5/'
>>> reverse('add2', args=(444,555))
u'/add/444/555/'

reverse接收url中的name作为第一个参数,我们在代码中可以通过reverse()来获取对应的网址(这个网址可以用来跳转,也可以用来计算相关页面的地址,只要对应的url的name不变,就不用改变代码中的网址。

在网页模板中可以方便的使用:

不带参数的:
{% url 'name' %}
带参数的:参数可以是变量名
{% url 'name' 参数 %}
 
例如:
<a href="{% url 'add2' 4 5 %}">link</a>

上面代码渲染成的最终页面是:

<a href="/add/4/5/">link</a>

这样就可以通过 {% url 'add2' 4 5 %} 获取到对应的网址 /add/4/5/

当 urls.py 进行更改,前提是不改 name(这个参数设定好后不要轻易改),获取的网址也会动态地跟着变,比如改成:

url(r'^new_add/(\d+)/(\d+)/$',views_add.add2,name='add2'),

注意看重点 add 变成了 new_add,但是后面的 name='add2' 没改,这时 {% url 'add2' 4 5 %} 就会渲染对应的网址成 /new_add/4/5/

用在 views.py 或 models.py 等地方的 reverse函数,同样会根据 name 对应的url获取到新的网址。

想要改网址的时候,修改 urls.py 中的正则表达式部分(url 参数第一部分),name 不变的前提下,其它地方都不需要修改。

如果用户收藏夹中收藏的URL是旧的,如何让以前的 /add/3/4/自动跳转到现在新的网址呢?

具体思路是,在 views.py 写一个跳转的函数:

from django.http import HttpResponseRedirect   #注意这行是新增的
from django.shortcuts import render

from django.http import HttpResponse
from django.urls import reverse                           #注意这行是新增的

# Create your views here.
def old_add2_redirect(request,a,b):
    return  HttpResponseRedirect(
        reverse('add2',args=(a,b))
    )

在urls.py中:

urlpatterns = [
    url(r'^$',views_add.index,name='home'),
    url(r'^add/(\d+)/(\d+)/$',views_add.old_add2_redirect),
    url(r'^new_add/(\d+)/(\d+)/$',views_add.add2,name='add2'),
    url(r'^admin/', admin.site.urls),
]

这样就能解决:假如用户收藏夹中有 /add/4/5/ ,访问时就会自动跳转到新的 /new_add/4/5/ 了

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,324评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,303评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,192评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,555评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,569评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,566评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,927评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,583评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,827评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,590评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,669评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,365评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,941评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,928评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,159评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,880评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,399评论 2 342

推荐阅读更多精彩内容