Python编程从入门到实践笔记


编程环境的安装请见我另两篇文章《Pycharm安装及破解方法》及《Pycharm配置和使用教程》,下面以你能正常使用pycharm为前提。

基础知识

变量及简单数据类型

变量名的使用

  1. 变量名只能包含字母,数字和下划线。可以以字母和下划线开头,但不能以数字开头。
  2. 变量名不能包含空格,可用下划线来分割单词,如:greeting_message。
  3. 不要将Python关键字和函数名用作变量名。
  4. 变量名应既简短又具有描述性。
  5. 慎用小写字母l和大写字母O,容易被看成1和0。

字符串

字符串是一串字符,引号括起的都是字符串,可以是单引号或双引号,如:

str = "ada lovelace"
str.title()                 #单词首字母大写
str.upper()                 #单词所有字母大写
str.lower()                 #单词所有字母小写
str = str1 + str2          #字符串相加
#字符串中出现\t是制表符,\n是换行符
str.rstrip()                #删除空格,不改变原变量

整型(浮点型同理)

加减乘除不必多说

3 ** 2     #乘方,结果是9
3 % 2      #求模,返回余数

注意,整型相除也是整型,会自行砍掉小数。

强制类型转换

str(age)         # 转变为字符串
int(age)         # 转变为整型

列表

列表由一系列特定元素排列而成,如:

bicycles = ['trek','cannondale','redline','specialized']
list[0]                  #表示第一个元素
list[-1]                 #表示最后一个元素
#列表的增删改
list.append('honda')     #在列表末尾添加字符串'honda'
list.insert(0,'ducati')  #在第一位插入字符串'ducati',后面的元素后移一位
del list[0]              #删除第一个元素
str = list.pop()         #删除列表末尾的元素,同时可以把这个元素赋值给另一个变量
str = list.pop(0)        #删除第一个元素,同时可以把这个元素赋值给另一个变量
list.remove('ducati')    #不清楚元素位置,知道元素值的删除方法
#列表的排列(排列时列表数据类型要保持一致)
list.sort()              #按字母顺序排列列表
list.sort(reverse = True)#按与字母顺序相反的顺序排列列表
sorted(list)             #不影响原有列表的排列顺序,直接返回一个排列好的列表
sorted(list,reverse = True)   #反向排列,不改变原有列表
list.reverse()           #将列表元素反转排列
#使用列表一部分
list[0:3]                #输出一个包含列表第1到3个元素的列表,没有list[3]
list[:4]                 #输出一个包含列表前四个元素的列表
list[2:]                 #输出一个包含列表第三个元素开始往后的元素的列表
list[-3:]                #输出一个包含列表最后三个元素的列表
#复制一个列表
list2 = list1[:]         #不能使用list2 = list1,因为这样它们的地址是一样的

len(list)                #确定列表长度
for i in list:           #打印整个列表
    print(i)
range(1,5)               #1到4的整数,是可迭代对象
range(2,11,2)            #2到10的整数,步长为2(2,4,6,8,10)
list(range(1,5))         #转换为列表
min(list_digits)         #找出数字列表的最小值
max(list_digits)         #找出数字列表的最大值
list = [value**2 for value in range(1,11)]    #一行代码生成列表

元组

不可修改的列表称为元组。

tuple = (200,50)             #两个元素的元组
tuple[0] = 10                #不合法 
tuple = (100,10)             #可以给储存元组的变量重新赋值,合法

if语句

#检查单个条件
if car == 'bmw':
    print(car.upper())
elif car == 'saf':
    print(car.lower())
else:
    print(car.title())
#检查多个条件
if age_0 >= 21 and age_1 >= 21
if age_0 >= 21 or age_1 >= 21
#检查特定值是否在列表中,特定值是否在字符串中同理
if 'value' in list
if 'value' not in list
#确定列表不是空的
if list

字典

字典是一系列的键值对:

dict = {'color':'green','points':5}
dict['color']         #访问字典中'color'所对应的值
#字典是动态结构,可随时在其中添加键值对,键值对的排列顺序和添加顺序不同,Python不关心键值对顺序,只关心键与值的联系
dict['x_position'] = 0
#修改字典中的值
dict['x_position'] = 10
#删除键值对
del dict['points']
#遍历字典
for key,value in dict.items():       #item方法返回一个键值对列表,依次是键,值,键,值。。。
    print(key/n)
    print(value)
for key in dict.keys():              #遍历字典所有键
    print(key);
for key in sorted(dict.keys()):      #按顺序遍历字典所有键
    print(key);   
for value in dict.values():
	print(value)
    
#嵌套,字典储存于列表或列表储存于字典
#字典储存于列表中可运用于游戏中产生的一群敌人(列表),每个敌人的数据都不相同(字典)
#列表储存于字典运用于一个特征由多个元素组成,如披萨的原料,顾客点的披萨:外壳:硬,原料:蘑菇,奶酪。
#字典中储存字典,运用于网站用户,用户名作为键,用户信息储存于一个字典中,字典作为值。

用户输入和while循环

  • 函数input()让程序暂停运行,等待用户输入文本,获取用户输入后,Python将其储存在一个变量中,供你使用:
message = input("Tell me something,and I will repeat it back to you:")         #获取输入,input中是提示信息
print(message)
  • while循环不断执行,知道指定的条件不满足为止:
num = 1
while num <= 5:
    print(num)
    num++
    if city == 'quit':      #可以设定条件主动退出循环
        break               #continue为退出本次循
  • 在for循环中不应修改列表,否则将导致Python难以跟踪其中的元素,可使用while进行修改:
unconfirmed_users = ['alice','brian','candace']
confirmed_users = []
while unconfirmed_users:                       #验证所有未验证用户,并从旧列表中删除,将已验证用户加入新列表中
    current_user = unconfirmed_users.pop()
    print(current_user.title())
    confirmed_users.append(current_users)
  • 删除包含特定值的所有列表元素
while 'cat' in pets:
    pets.remove('cat')

函数

函数是带名字的代码块,用于完成具体的工作。需要程序多次执行同一项任务是,无需反复编写代码,只需反复调用执行改任务的函数即可。

def greet_user():
    print('Hello!')
#向函数传递信息
def greet_user(username):                                          #一个参数
    print("Hello" + username.title())
def describe_pet(animal_type = ‘rabbit’,pet_name = 'snoby'):       #两个参数,可以给形参设定默认值,这样就可以不用传入实参
    print("I have a" + animal_type + '.The name is' + pet_name)    
describe_pet('hamster','harry')                                    #实际调用
describe_pet(animal_type = 'hamster',pet_name = 'harry')
#有返回值的函数
def git_fromatted_name(first_name,last_name):
    full_name = first_name + last_name
    return full_name.title()	                              #return {'first':first_name,'last':last_name}    可返回字典
#把列表传给函数后,函数可以直接对其进行修改,注意这在简单数据类型行不通。如果不想修改列表本身可以用list[:]副本传入
def print_models(unprinted_designs,completed_models):
    while unprinted_designs:
        current_designs = unprinted_designs.pop()
        print('Printing model:' + current_design)
        completed_models.append(current_design)
#传递任意数量的实参
def make_pizza(*toppings):                                        #这里toppings是一个元组,可以传入任意数量的元素
    print(toppings)
make_pizza('mushrooms','green peppers','extra cheese')            #调用函数,传入的参数自行组成元组
def make_pizza(size,*toppings):                                   #结合使用    
    print("make a " + size + "pizza with the following toppings:")
    for topping in toppings:
        print("- " + topping)
#使用任意数量的关键字实参(传入任意数量元素的字典)
def build_profile(first,last,**user_info):                         #传入信息和键值对,合并成一个字典
    profile = {}
    profile['first_name'] = first
    profile['last_name'] = last
    for key,value in user_info.items():
    	profile[key] = value
    return profile
bild_profile('albert','einstein',location = 'princeton',field = 'physics')      #调用
#将函数储存在模块中
#创建一个存储函数的独立文件,在其所在目录中的其他py文件都可以通过import+文件名导入模块
#pizza.py
def make_pizza(*toppings):                                        
    print(toppings)
#making_pizzas.py
import pizza                          #import pizza as p 可以给模块指定别名
pizza.make_pizza('mushroom','green peppers','extra cheese')
#导入特定函数
from pizza import make_pizza as mp           #可以给函数指定别名
#导入模块中所有函数
from pizza import *

面向对象编程是最有效的软件编写方法之一。在面向对象编程是,你编写表示现实世界中的事物和场景的类,基于这些类来创造对象,每个对象都具备类的通用行为,也可根据需要赋予每个对象独特的个性。

#创建一个类
#self是一个指向实类本身的引用,让实例能访问类中的属性和方法。
#Python调用_init_()方法来创建实例时会自动传入实参self,我们自己不需要传递它。
class Dog():
    def __init__(self,name,age):   #注意init两边的横线的两条杠            
        self.name = name
        self.age = age
        self.model = 0             #可以在此处定义另外没传入的变量
    def sit(self):
        print(self.name.title() + ‘is now sitting.’)
    def roll_over(self):
        print(self.name.title() + 'rolled over!')
    def update_model(self,model):
        self.model = model
#创建实例
my_dog = Dog('willie',6)
my_dog.name                        #访问实例中的变量
my_dog.age                         #访问实例中的变量
my_dog.sit()                       #调用类中的方法
#修改属性的值
my_dog.model = 10                  #大多数情况为了封装完整,应该使用方法对属性进行修改
my_dog.update_model(10)            #这样修改

类的继承

编写类是,并非总是从空白开始。一个类继承另一个类时,自动获得另一个类所有的属性和方法,原有的类称为父类,新类称为子类,子类也可定义属于自己的属性和方法。

#假设前面已有一个Car类
class ElectricCar(Car):
    def __init__(self,make,model,year):
        super()._init_(make,model,year) #super是一个特殊函数,将父类和子类联系起来,代表了父类,此处父类和子类的_init_函数保持一致
        #此处编写ElecticCar类的独特属性
#子类可以重写父类的方法
#可以将实例用作属性
class Battery():
    --skip
class ElectricCar(car):
    def _init_(self,make,model,year):
        super()._init_(make,model,year)
        self.battery = Battery();              #实例作属性

导入类

Python允许你将类储存在模块中,然后在主程序中导入所需的模块:

#car.py
Class Car():
    --skip
Class Smartphone():
    --skip
#my_car.py
from car import Car          #导入整个模块用import car,运用是要在类前加mudule_name.。导入模块中所有类用from car import *
my_new_car = Car('audi','a4',2016)

文件和异常

自此,你已掌握了编写组织有序易于使用的程序所需的基本技巧,为了让程序用途更广,本章将学习处理文件,让程序快速分析大量数据;处理异常,用于管理程序运行时出现的错误,还将学习模块json,保存用户数据,以免程序停止运行后丢失。本章的学习可提高程序的实用性,可用性,稳定性。

从文件中读取数据

假设已创建了一个文件pi_digits.txt

#读取文件,显示内容,不需要主动调用close
with open('pi_digits.txt') as file_object:   #文件命名为file_object。相对路径行不通可以用绝对路径
    contents = file_object.read()            #读取文件内容,传入字符串中 
    print(contents.rstrip())                 #rstrip()方法删除字符串末尾的空白
#逐行读取
with open(filename) as file_object:
    lines = file_object.readlines()         #readlines()方法从文件中读取每一行,储存在一个列表中
    for line in file_object:
        print(line)        

写入文件

#整体写入
with open(filename,'w') as file_object:             #以写入模式打开文件,如果文件不存在会自动创建,已经存在会清空文件
    file_object.write("I love programming.")        #注意只能写入字符串,写入多行时可以加换行符
#附加内容
with open(filename,'a') as file_object:             #以附加模式打开文件,不会清空原有文件,写入的行添加到文件末尾
    file_object.write("I love programming.")        

异常

发生程序异常时未对异常进行处理,程序会停止。使用try-except代码块时,即使出现异常,也会继续运行,显示你编写的友好的错误信息而不是traceback。

try:
    print(5/0)
except ZeroDivisionError:
    print('You can\'t divide by zero!')
else:                                                              #正常运行
    print("answer")
#处理文件找不到的异常
try:
    with open(filename) as f_obj:
        contents = f_obj.read()
except FileNotFoundError:
    msg = "Sorry,the file" + filename + "does not exist."          #如果这里写pass,错误时可以不输出信息
    print(msg)

分析文本

str.split()            #以空格为分隔符将字符串拆分为多个部分,并储存到一个列表中
#计算文本单词数
def count_words(filename):
    try:
        with open(filename) as f_obj:
            contents = f_obj.read()
    except FileNotFoundError:
        print('Sorry,the file does not exist')
    else:
        word = contents.split()
        num_words = len(words)
        print("The file " + filename + "has about " + str(num_words) + " words.")

存储数据

程序需要把用户提供的数据存储在列表和字典等数据结构中,用户关闭程序时,需要保存他们提供的信息,一种简单的办法是用模块json来储存信息

json.dump()和json.load()

import json
numbers = [2,3,5,7,11,13]
filename = 'numbers.json'
with open(filename,'w') as f_obj:
    json.dump(numbers,f_obj)         #传入要储存的数据和储存数据的文件对象
#文件中数据存储格式和Python中一样[2,3,5,7,11,13]
#再将列表读取到内存中
with open(filename) as f_obj:
    numbers = json.load(f_obj)       #列表被读取出来

测试代码

每个程序员都需要经常测试其代码,在用户发现问题前找到它,因此需要编写测试代码改进代码。

Python标准库中的模块unittest提供了测试工具,单元测试用于核实函数的某个方面没有问题。

import unittest
from name_function import get_formatted_name
#测试类,所有以test_开头的方法自动运行
class NamesTestCase(unittest.TestCase):                     #必须继承unittest.TestCase
    def test_first_last_name(self):                         #核实名和姓能否被正确格式化
        formatted_name = get_formatted_name('janis','joplin')
        self.assertEqual(formatted_name,'Janis Joplin')     #断言方法,核实得到的结果与期望是否一致
unittest.main()
#如果全部测试通过会输出OK,第一行的句点表明几个测试通过
#不通过第一行会有E,最后显示FALLED
#测试不通过时,意味新代码有错。不要修改测试,而应修改导致测试不通过的代码。
方法 用途
assertEqual(a,b) 核实a == b
assertNotEqual(a,b) 核实a != b
assertTrue(x) 核实x为True
assertFalse(x) 核实x为False
assertIn(item,list) 核实item在list中
assertNotIn(item,list) 核实item不在list中

类的测试和函数的测试类似:

#假设已经创建了一个匿名调查类AnonymousSurvey
from survey import AnonymousSurvey
import unittest
class TestAnonmyousSurvey(unittest.TestCase):
    def test_store_single_response(self):
        question = 'what language did you first learn to speak?'
        my_survey = AnonymousSurvey(question)                #问题
		my_survey.store_response('English')                  #答案,会添加到答案列表中  
        my_survey.store_response('Chinese')
        self.assertIn('English',my_survey.responses)
unittest.main()

方法setUp(),TestCase类包含方法,对测试函数初始化

from survey import AnonymousSurvey
import unittest
class TestAnonmyousSurvey(unittest.TestCase):
    def setUp(self):
        question = 'what language did you first learn to speak?'
        my_survey = AnonymousSurvey(question)                        #输入问题
        self.responses = ['English','Spanish','Mandarin']
    def test_store_single_response(self):                            #测试一个答案 
		my_survey.store_response(self.responses[0])                 
        self.assertIn(self.responses[0],my_survey.responses)
    def test_store_three_responses(self):                            #测试三个答案
        for response in self.responses:
            self.my_survey.store_reponse(response)
        for response in self.responses:
            self.assertIn(response,self.my_survey.responses)
unittest.main()

项目

数据可视化

数据可视化是通过可视化来表示探索数据,与数据挖掘紧密相关,在基因研究,天气研究,政治经济分析等众多领域,大家都使用Python完成数据密集型工作,数据科学家编写了一系列令人印象深刻的可视化和分析工具,最流行的是matplotlib,它是一个数学绘图库,可以制作简单的图表。我们还将使用Pygal包,它专注于生成适合在数字设备上显示的图表。

pycharm安装matplotlib十分方便快捷,直接在File→setting→Project interpreter中一键导入即可,不再赘述。

绘制简单曲线图

#最简版
import matplotlib.pyplot as plt
squares = [1,4,9,16,25]
plt.plot(squres)
plt.show()
#改善可读性
import matplotlib.pyplot as plt

input_values = [1,2,3,4,5]                           #x轴的值
squares = [1,4,9,16,25]                              #y轴的值
plt.plot(input_values,squares,linewidth =  5)       #线段宽度
plt.title("Square Numbers",fontsize = 24)            #标题及其大小
plt.xlabel("Value",fontsize = 14)                    #x轴标签及其大小
plt.ylabel("Square of Value",fontsize = 14)          #y轴标签及其大小
plt.tick_params(axis = 'both',labelsize = 14)        #设置刻度标记的大小
plt.show()

绘制散点图

import matplotlib.pyplot as plt
x_values = [1,2,3,4,5]
y_values = [1,4,9,16,25]
#scatter方法绘制散点图,可删除数据点黑色轮廓,可设置颜色,默认为蓝色。cmap = plt.cm.Blues为设置渐变蓝色
plt.scatter(x_values,y_values,s = 100,edgecolor = 'none',c = 'red')   

plt.title("Square Numbers",fontsize = 24)                   #配置四连,不再赘述
plt.xlabel("Value",fontsize = 14)                   
plt.ylabel("Square of Value",fontsize = 14)         
plt.tick_params(axis = 'both',labelsize = 14)        
plt.show()

保存图表

#第一个实参是文件名,第二个指定将图表多余的空白裁剪掉
plt.savefig('squares_plot.png',bbox_inches = 'tight')

随机漫步

from random import choice
import matplotlib.pyplot as plt

class RandomWalk():
    def __init__(self,num_points = 5000):
        self.num_points = num_points
        self.x_values = [0]
        self.y_values = [0]
    def fill_walk(self):
        while len(self.x_values) < self.num_points:
            x_direction = choice([1,-1])                        #这里采用choice方法选取随机数
            x_distance = choice([0,1,2,3,4])
            x_step = x_direction * x_distance
            y_direction = choice([1, -1])
            y_distance = choice([0, 1, 2, 3, 4])
            y_step = y_direction * y_distance
            if x_step == 0 and y_step == 0:
                continue
            next_x = self.x_values[-1] + x_step
            next_y = self.y_values[-1] + y_step

            self.x_values.append(next_x)
            self.y_values.append(next_y)

rw = RandomWalk()
rw.fill_walk()
point_numbers = list(range(rw.num_points))
plt.scatter(0,0,s = 100,c = 'green',edgecolor = 'none')                             #绘制起点            
plt.scatter(rw.x_values[-1],rw.y_values[-1],s = 100,c = 'red',edgecolor = 'none')   #绘制终点
plt.scatter(rw.x_values,rw.y_values,s = 15,c = point_numbers,                       #根据绘制的先后决定点颜色的深浅   
            cmap=plt.cm.Blues,edgecolor = 'none')
#plt.figure(dpi=128,figsize=(10,6))      调整绘图窗口尺寸
plt.show()

使用Pygal模拟掷骰子(柱状图)

from random import randint
import pygal

class Die():
    def __init__(self,numsides = 6):
        self.num_sides = numsides
    def roll(self):
        return  randint(1,self.num_sides)
#投掷1000次,结果储存在列表中
die = Die()
results = []
for roll_num in range(10000):
    result = die.roll()
    results.append(result)
#处理数据
frequencies = []
for value in range(1,die.num_sides+1):
    frequency = results.count(value)             #count函数,计算列表中一个值出现的次数
    frequencies.append(frequency)
print(frequencies)
#构建柱状图
hist = pygal.Bar()
hist.title = 'Results of rolling one D6 1000 times.'
hist.x_labels = ['1','2','3','4','5','6']
hist.x_title = "Result"
hist.y_title = "Frequency of Result"
hist.add('D6',frequencies)
hist.render_to_file('die_visual.svg')             #保存文件

从网上下载数据并处理和绘图

本章中,你将从网上下载数据,并对数据进行可视化。我们将使用Python模块csv来处理以CSV(逗号分隔的值)格式存储的天气数据,使用模块json来访问以JSON存储的人口数据。

绘制天气情况图表

import csv
from matplotlib import pyplot as plt
from datetime import datetime

filename = 'death_valley_2014.csv'
with open(filename) as f:
    reader = csv.reader(f)              #创建一个与该文件向关联的阅读器
    header_row = next(reader)           #返回文件中的下一行,这里是第一行,返回一个列表,以逗号分隔开的内容为一个元素
#    for index,conlumn_header in enumerate(header_row):  #enumerate()方法获取列表每个元素的索引和值
#        print(index,conlumn_header)
    dates,highs,lows = [],[],[]
    # 遍历文件余下的各行,阅读器对象从其停留的地方继续往下读取csv文件,每次返回下一行,第一行已经读取,这里从第二行开始
    # 返回的都是第二列每一天最高温度的值
    for row in reader:
        try:                                                    #对缺失的数据进行检查
            current_date = datetime.strptime(row[0],"%Y-%m-%d") #这里获取日期,第二个参数指定如何解读日期
            high = int(row[1])                                  #转化为整形,matplotlib才能读取它们
            low = int(row[3])
        except ValueError:
            print(current_date,'missing data')
        else:
            dates.append(current_date)
            highs.append(high)
            lows.append(low)
#绘图代码
fig = plt.figure(dpi = 128,figsize=(10,6))                      #调节图的大小,第一个参数是窗口分辨率,第二个是长和宽
plt.plot(dates,highs,c='red',alpha=0.5)                         #画折线图,alpha为透明度
plt.plot(dates,lows,c='blue',alpha=0.5)
plt.fill_between(dates,highs,lows,facecolor = 'blue',alpha=0.1) #中间填充颜色
fig.autofmt_xdate()                                             #绘制倾斜的x轴标签
plt.title('Daily high and low temperatures - 2014',fontsize=24)
plt.xlabel('',fontsize=16)
plt.ylabel('Temperature(F)',fontsize=16)
plt.tick_params(axis='both',which = 'major',labelsize = 16)
plt.show()

datetime.strptime方法第一个参数是传入的字符串,第二个参数规定字符串的格式,下表列出了一些这样的实参:

实参 含义
%A 星期的名称,如Monday
%B 月份名称,如January
%m 用数字表示的月份(01-12)
%d 用数字表示的月份中的一天(01-31)
%Y 四位的年份,如2021
%y 两位的年份,如21
%H 24小时的小时数(00-23)
%I 12小时的小时数(01-12)
%p am或pm
%M 分钟数(00-59)
%S 秒数(00-61)

制作世界人口地图

JSON文件其实是一个很长的列表,每个元素都是字典。

书本提供的链接已经不能下载数据,但感谢CSDN的兄弟,让我下载到了这份数据,链接附上:

https://pan.baidu.com/s/1FlwB2SQzn_z06SR3eM9mJg,提取码:q6vy

注意:原来的pygal.i18n的包已经弃用,改为pygal.maps.world,请自行下载pygal_maps_world模块。

代码如下:

import json
from pygal.maps.world import COUNTRIES      #注意原来的包已经弃用,COUNTIES是一个字典,两位国别码是键,国家名是值
import pygal
from pygal.style import RotateStyle as RS,LightColorizedStyle as LCS

def get_country_code(country_name):         #获取两位国别码的函数
    for code,name in COUNTRIES.items():
        if name == country_name:
            return code
    return None

filename = "population_data.json"
with open(filename) as f:
    pop_data = json.load(f)           #生成一个列表,每个元素是字典

cc_populations = {}
for pop_dict in pop_data:
    if pop_dict['Year'] == '2010':                #只用2010年的信息
        country_name = pop_dict['Country Name']
        population = int(float(pop_dict['Value']))
        code = get_country_code(country_name)     #获取两位国别码
        if code:                                  #过滤掉不是国家的信息
            cc_populations[code] = population
cc_pops_1,cc_pops_2,cc_pops_3 = {},{},{}          #分成三类可以用不同颜色的深浅表示,区分度更明显
for cc,pop in cc_populations.items():
    if pop < 10000000:
        cc_pops_1[cc] = pop
    elif pop < 1000000000:
        cc_pops_2[cc] = pop
    else:
        cc_pops_3[cc] = pop

wm_style = RS('#336699',base_style=LCS)           #让地图颜色更一致,更明亮,更容易区分不同的编组
wm = pygal.maps.world.World(style = wm_style)
wm.title = ('World Population in 2010, by Country')
wm.add('0-10m',cc_pops_1)
wm.add('10m-1bn',cc_pops_2)
wm.add('>1bn',cc_pops_3)

wm.render_to_file('americas.svg')

{% asset_img 7.png %}


使用API

在本章,程序将使用Web应用编程接口(API)自动请求网站的特定信息而不是整个网站,再对信息进行可视化,这样信息是最新的。

使用API调用请求数据,在浏览器地址栏输入:

https://api.github.com/search/repositories?q=language:python&sort=stars

这个调用返回GitHub当前托管了多少个项目,还有最受欢迎的Python仓库的信息。

import requests

url = 'https://api.github.com/search/repositories?q=language:python&sort=stars'
r = requests.get(url)
print('Status code:',r.status_code)       #状态码为200则响应成功

response_dict = r.json()                 #这个API返回的是JSON格式信息,因此转换为字典
print(response_dict.keys())
print("Total repositories:",response_dict['total_count'])   #看仓库总数
#看获取了多少仓库的信息
repo_dicts = response_dict['items']
print("Repositories returned:",len(repo_dicts))
#查看获取到的每个仓库的信息
print("\nSelected information about each repository:")
for repo_dict in repo_dicts:
    print("\nName:",repo_dict['name'])
    print("Owner:", repo_dict['owner']['login'])
    print('Stars:',repo_dict['stargazers_count'])
    print('Repository:',repo_dict['html_url'])
    print('Description:',repo_dict['description'])

监视API速率限制,在浏览器输入:https://api.github.com/rate_limit,内容如下:

{"resources":
	{"core":
		{"limit":60,"remaining":60,"reset":1610899050,"used":0},
		"graphql":{"limit":0,"remaining":0,"reset":1610899050,"used":0},
		"integration_manifest":{"limit":5000,"remaining":5000,"reset":1610899050,"used":0},
		"search":{"limit":10,"remaining":10,"reset":1610895510,"used":0}
	},
	"rate":{"limit":60,"remaining":60,"reset":1610899050,"used":0}
}

我们关心的是搜索API的速率限制,可知极限为每分钟10个请求,用完配额后会受到一条简单的响应,必须等待配额重置。很多API都要求你注册获得API秘钥后才能执行API调用。

使用API可视化仓库:

import requests
import pygal
from pygal.style import LightColorizedStyle as LCS,LightenStyle as LS

url = 'https://api.github.com/search/repositories?q=language:python&sort=stars'
r = requests.get(url)
print('Status code:',r.status_code)       #状态码为200则响应成功

response_dict = r.json()                 #这个API返回的是JSON格式信息,因此转换为字典
print(response_dict.keys())
print("Total repositories:",response_dict['total_count'])   #看仓库总数
repo_dicts = response_dict['items']

#收集每个仓库的名字和星数信息
names,plot_dicts,stars = [],[],[]
for repo_dict in repo_dicts:
    names.append(repo_dict['name'])
    plot_dict = {
        'value': repo_dict['stargazers_count'],          #数据
        'label': str(repo_dict['description']),          #描述
        'xlink': repo_dict['html_url'],                  #链接
        }
    plot_dicts.append(plot_dict)
    stars.append(repo_dict['stargazers_count'])

my_style = LS('#333366',base_style=LCS)   #定义样式,基色为深蓝色
my_config = pygal.Config()                #创建设定,下面是一系列设定
my_config.x_label_rotation = 45
my_config.show_legend = False
my_config.title_font_size = 24
my_config.label_font_size = 14
my_config.major_label_font_size = 18      #主要标签大小
my_config.truncate_label = 15             #较长的项目名缩短为15个字符
my_config.show_y_guides = False           #隐藏图表中的水平线
my_config.width = 1000                    #图表宽度
chart = pygal.Bar(my_config,style = my_style)

chart.title = "Most_Starred Python Projects on Github"
chart.x_labels = names                    #x轴标签
chart.add('',plot_dicts)                  #导入数据
chart.render_to_file('python_repos.svg')

探索如何使用其他网站的API调用,可以自己去查API接口,这部分内容我会在以后更新放在其他博文里。


Django入门

建立项目

Python提供了一组开发Web开发的卓越工具,这时一个Web框架,一套用于帮助开发交互式网站的工具。

建立虚拟环境,新建一个目录,将终端切换到这个目录,使用如下命令建立虚拟环境:

python -m venv 11_env

建立虚拟环境后,需要使用如下命令激活它:

source 11_env/Scripts/activate

环境处于激活状态时环境名包含在括号里,这时可以在环境中安装包,使用已安装的包,在11_env中安装的包在环境处于活动状态时才能使用。

如果要停止虚拟环境,执行命令:

deactivate

安装Django:

pip3 install Django

在Django中创建项目:

django-admin.py startproject learning_log

这时在根目录下就会出现learning_log文件夹,内含几个.py文件。manage.py接受命令交给Django的相关部分执行,管理诸如使用数据库和运行服务器等任务。文件settings.py指定Django如何与你的系统交互,如何管理项目。urls.py告诉Django应该创建哪些网页来响应浏览器的请求。文件wsgi.py帮助Django提供它创建的文件。

创建数据库:

Django将大部分与项目相关的信息都储存在数据库中,因此我们需要创建一个供Django使用的数据库,活跃状态下进入manage.py的目录执行如下命令:

python manage.py migrate

我们将修改数据库成为迁移数据库,在使用SQLite的新项目中首次执行这个命令时,Django将新建一个数据库。在这里Django创建必要的数据库表,用于储存我们将在这个项目中使用的信息,确保数据库结构与当前代码匹配。

核实Django是否正确创建了项目:

python manage.py runserver

Django启动服务器,让你能够查看系统中的项目,当你在浏览器中输入URL请求网页时,Django服务器将进行响应。打开浏览器输入:http://localhost:8000/即可查看网页,控制台按Ctrl+C可以关闭服务器。

创建应用程序

在激活状态下,切换到manage.py的目录下执行命令:

python manage.py startapp learning_logs

Django创建程序应用learning_logs,项目文件新增一个文件夹learning_logs,里面有一些.py文件,其中models.py定义我们要在应用程序中管理的数据。

models.py:

from django.db import models

class Topic(models.Model):          #Model是Django中定义了模型基本功能的类,只有text和date_added两个属性
    text = models.CharField(max_length=200)  #CharField储存少量文本,在数据库中预留200个字符的位置
    date_added = models.DateTimeField(auto_now_add=True)  #记录日期和时间,当用户创建新主题,自动设置成当前日期和时间

    def __str__(self):             #显示模型的简单表示
        return self.text

然后打开learning_log目录下的setting.py,把修改一段代码:

INSTALLED_APPS = [                              #安装在项目中的应用程序
    'django.contrib.admin',
    'django.contrib.auth',
    'django.contrib.contenttypes',
    'django.contrib.sessions',
    'django.contrib.messages',
    'django.contrib.staticfiles',
    
    #我的应用程序
    'learning_logs',
]

接下来需要Django修改数据库,使其能够储存于模型Topic相关的信息,在终端执行:

python manage.py makemigrations learning_logs

命令makemigrations让Django确定如何修改数据库,使其储存与我们定义的新模型相关的数据,Django创建了一个名为0001——initial.py的迁移文件,这个文件在数据库中为模型Topic创建一个表。

应用这种迁移:

python manage.py migrate

当需要修改数据时,都需要采取如下三个步骤:修改models.py,对learning_logs调用makemigrations,让Django迁移项目。

Django管理网站:

为应用程序定义模型时,Django提供的管理网站(admin site)让你能轻松处理模型。

Django允许创建具备所有权限的用户——超级用户,命令如下(要在cmd执行否则不成功):

python manage.py createsuperuser

向管理网站注册模型:

Django自动在网站中添加了一些模型,如User和Group,但对于我们自己创建的模型,必须进行手工注册。

models.py所在的目录中有admin.py文件,为向管理网站注册Topic,输入下面代码:

from django.contrib import admin
from learning_logs.models import Topic            #导入要注册的模型

admin.site.register(Topic)                       #可以通过管理网站管理模型了

访问http://localhost:8000/admin/,输入用户名和密码,可以进入超级用户账户访问管理网站,可以让你添加修改用户和用户组,管理刚才定义的模型Topic相关的数据。

添加主题

单击Topics进入主题网页,单击Add,看到一个用于添加新主题的菜单,输入Chess单击Save。在Add一个Rock Climbing。这样就有两个主题了。

定义模型Entry:

要记录学到的国际象棋和攀岩知识,需要为用户可在学习笔记中添加的条目定义模型,每个条目都与特定主题相关联,这种关系被称为多对一关系,即多个条目可关联到同一个主题:

models.py

from django.db import models

class Topic(models.Model):          #Model是Django中定义了模型基本功能的类,只有text和date_added两个属性
    text = models.CharField(max_length=200)  #CharField储存少量文本,在数据库中预留200个字符的位置
    date_added = models.DateTimeField(auto_now_add=True)  #记录日期和时间,当用户创建新主题,自动设置成当前日期和时间

    def __str__(self):             #显示模型的简单表示
        return self.text

class Entry(models.Model):
    #外键是数据库术语,引用数据库中另一条记录将条目关联到特定的主题
    #在django2.0后,定义外键和一对一关系的时候需要加on_delete选项,此参数为了避免两个表里的数据不一致问题,不然会报错
    topic = models.ForeignKey(Topic,on_delete=models.CASCADE) 
    text = models.TextField()                             #不需要长度限制的字段
    date_added = models.DateTimeField(auto_now_add=True)  #按创建顺序呈现条目,条目旁边放置时间戳
    class Meta:                                           #储存用于管理模型的额外信息,可以使用entries表示多个条目
        verbose_name_plural = 'entries'
    def __str__(self):
        return self.text[:50] + "..."                     #呈现条目时只显示前50个字符

修改完models.py后记得迁移模型:

python manage.py makemigrations learning_logs
python manage.py migrate

这时会生成一个新的迁移文件0002——entry.py,使数据库能储存于模型Entry相关的信息。

然后向管理网站注册Entry:

admin.py:

from django.contrib import admin
from .models import Topic,Entry

admin.site.register(Topic)
admin.site.register(Entry)

这时登录超级用户管理网站,就可以在主题下添加条目了:

输入一些数据后,就可以通过交互式终端会话以编程方式查看这些数据了,这种交互式环境被称为Django Shell,是测试项目和排除故障的理想之地。下面是交互式Shell示例:

$ python manage.py shell
>>> from learning_logs.models import Topic
>>> Topic.objects.all()
<QuerySet [<Topic: Chess>, <Topic: Rock Climbing>]>
>>> topics = Topic.objects.all()
>>> for topic in topics:
...     print(topic.id,topic)
...
1 Chess
2 Rock Climbing

知道对象的ID后,就可获取该对象并查看其任何属性,其中内容展示由于中文原因造成乱码:

>>> t = Topic.objects.get(id=1)
>>> t.text
'Chess'
>>> t.date_added
datetime.datetime(2021, 1, 18, 15, 29, 47, 62586, tzinfo=<UTC>)
>>> t.entry_set.all()
<QuerySet [<Entry: ▒й▒▒▒▒▒▒▒▒▒▒▒▒˫▒▒▒▒ִ16▒ӣ▒˫▒▒▒▒˫▒▒˫▒ڣ▒˫▒▒˫ʿ▒▒▒▒▒䣬һ▒▒▒▒▒▒▒▒▒▒▒▒90▒▒λ▒ÿɹ▒▒▒▒▒...>]>

退出shell会话输入exit()按回车即可。

创建网页:学习笔记主页

使用Django创建网页的过程分为三个阶段:定义URL、编写视图、编写模板。

首先必须定义URL模式,URL模式描述了URL是如何设计的,让Django知道如何将浏览器请求与网站URL匹配,以确定返回哪个网页。

每个URL都被映射到特定的视图——视图函数获取并处理网页所需的信息。视图函数通常调用一个模板,后者生成浏览器能够理解的网页。

映射URL:

用户通过在浏览器中点击链接或输入URL来请求网页,因此我们需要确定项目需要哪些URL。主页的URL最重要,是用户访问项目的基础URLhttp://localhost:8000/。我们将这个基础URL映射到“学习笔记”的主页。

在这里书本上的代码是Django1.0时期的,而后来新出的Django2.0使代码有了很大变化,此处是书本的大坑。此处请参照下面代码:

urls.py:

#注意2.0的代码的变化
from django.contrib import admin
from django.urls import path, include

urlpatterns = [
    path('admin/', admin.site.urls),
    path('', include('learning_logs.urls', namespace='learning_logs')),
]

此时我们需要在文件夹learning_logs中创建另一个urls.py文件,代码修改如下:

urls.py:

from django.urls import path
from . import views                #从此文件所在的目录中导入wiews

app_name='learning_logs'           #巨坑,书中没有,不写runserver时会报错
#URL模式是一个对函数path的调用,第一个参数是正则表达式,第二个参数指定了调用的视图函数第三个参数指定名称
#当需要提供到这个主页的链接时,都将使用这个名称而不是编写URL
urlpatterns = [                    #包含可在learning_logs中请求的网页
    path('', views.index, name='index'),                             # 主页
   ]

视图函数接受请求中的信息,准备好生成网页所需要的数据,再将这些数据发送给浏览器,这通常定义了网页是什么样的模板实现的。

views.py:

from django.shortcuts import render

def index(request):
    return render(request,'learning_logs/index.html') #参数是原始请求对象以及一个可用于创建网页的模板

URL请求与我们刚才定义的模式匹配时,Django将在views.py文件中查找函数,再将请求对象传递给这个视图函数,在这里我们不需要处理任何数据。

编写模板:

模板定义了网页的结构,指定了网页是什么样的,每当网页被请求时,Django将填入相关的数据,模板能让你访问视图提供的任何数据,我们的主页视图没有提供任何的数据,因此相应的模板非常简单。

在文件夹learning_logs中新建一个文件夹,命名templates,在里面新建一个文件夹名为learning_logs,在learning_logs中新建文件命名index.html。

index.html:

<p>learning Log</p>

<p>Learning Log helps you keep track of your learning.for any topic you're learning about.</p>

<p></p> 标识段落。这里定义两个段落,第一个充当标题,第二个阐述内容。

此时运行服务器,进入首页,看到的不是默认的Django网页,而是调用函数view.index()后使用index.html模板来渲染的网页。

创建网页的过程看起来很复杂,但将URL、视图、模板分离的效果实际上很好,这让我们能够分别考虑到项目的不同方面,在项目很大时,每个参与者可专注于其擅长的方面。数据库专家专注于模型,程序员专注于视图代码,Web设计人员专注于模板。

创建其他网页

我们将创建两个显示数据的网页,一个列出所有主题,一个显示特定主题下的所有条目,每个网页都将指定URL模式,编写一个视图函数,并编写一个模板,但这样做之前,要先创建一个父模板,项目中的其他模板都继承它。

模板继承:

在创建网站时,有一些所有网页都包含的元素,可编写一个包含通用元素的模板,让所有网页继承这个模板。

首先创建一个名为base.html的模板,储存在index.html所在的目录中,所有页面都包含顶端的标题,这个标题设置为到主页的链接:

base.html:

<p>
    <a href="{% url 'learning_logs:index' %}">learning Log</a>
</p>

{% block content %}{% endblock content %}

第一部分创建了包含项目名的段落,该段落也是一个到主页的链接。为创建链接,我们使用了一个模板标签,它生成了一个URL,该URL与learning_logs/urls.py中定义的名为index的URL模式匹配。此时,learning_logs是一个命名空间,index是该命名空间中一个名称独特的URL模式,在简单的HTML页面中,链接是使用锚标签定义的:

<a href="link_url">link text</a>

使用模板标签生成URL可让链接保持最新容易得多,修改项目中的URL只需修改urls.py的URL模式。

第二部分插入了一对块标签,名为content,是一个占位符,其中的内容由子模板指定。在父模板中,可使用任意多的块来预留空间,而子模板可根据需要定义相应数量的块。

重新编写index.html:

{% extends "learning_logs/base.html" %}

{% block content %}
    <p>Learning Log helps you keep track of your learning.for any topic you're learning about.</p>
{% endblock content %}

与原来的代码相比,我们将标题替换为了从父模板继承的代码。在子模板中只需包含当前网页特有的内容,简化了模板,让网站修改容易得多。

显示所有主题的页面:

首先定义显示所有主题页面的URL,我们将使用单词topics,因此http://localhost:8000/topics/ 将返回这个页面,learning_logs/urls.py修改如下:

from django.urls import path
from . import views                #从此文件所在的目录中导入wiews

app_name='learning_logs'           #巨坑,书中没有,不写runserver时会报错
#URL模式是一个对函数path的调用,第一个参数是正则表达式,第二个参数指定了调用的视图函数第三个参数指定名称
#当需要提供到这个主页的链接时,都将使用这个名称而不是编写URL
urlpatterns = [                    #包含可在learning_logs中请求的网页
    path('', views.index, name='index'),                             # 主页
    path('topics/', views.topics, name='topics'),                    # 显示所有的主题
   ]

views.py:

from django.shortcuts import render
from .models import Topic                  #导入所需数据相关的模型

def index(request):
    return render(request,'learning_logs/index.html')

def topics(request):                       #形参是Django从服务器收到的request对象          
    #显示所有主题
    topics = Topic.objects.order_by('date_added')  #查询数据库,请求提供Topic对象,按属性date_added进行排序
    context = {'topics':topics}                    #定义一个要发送给模板的上下文字典,键是模板中访问数据的名称,值是我们发送给模板的数据
    return render(request,'learning_logs/topics.html',context)#这里多传递一个数据

同样在index.html所在目录中,创建topics.html:

{% extends "learning_logs/base.html" %}
{% block content %}
    <p>Topics</p>
    <ul>
        {% for topic in topics %}
        <li>{{ topic }}</li>
        {% empty %}
        <li>No topics have been added yet.</li>
        {% endfor %}
    </ul>
{% endblock content %}

在content块中,首先显示一个主题名称的文本。然后使用了一个for循环的模板标签,遍历字典context中的列表topics,这里for循环必须使用 endfor 标签来显式指出其结束位置。在循环中,需要将变量名用双花括号括起来,告诉Django使用了一个模板变量,这样每次循环都会被替换成topic当前值。在标签对<ul></ul>的内部,<li>与</li>之间的内容都是一个项目列表项。 empty 标签告诉Django列表topics为空怎么办。

此时修改父模板,将其包含到显示所有主题的页面的链接:

base.html:

<p>
    <a href="{% url 'learning_logs:index' %}">learning Log</a> -
    <a href="{% url 'learning_logs:topics' %}">Topics</a>
</p>

{% block content %}{% endblock content %}

主页链接后面添加了一个连字符。后面新加一行让Django生成一个链接,与learning_logs/urls.py中名为topics的URL模式匹配。

现在可以在浏览器看到效果了:

显示特定主题的页面

显示特定主题的URL模式与前面所有的URL模式有所不同,它将使用主题的id属性来指出请求的是哪个主题,例如用户查看主题Chess(id为1)的详细页面,URL将为http://localhost:8000/topics/1/。

urls.py:

from django.urls import path,re_path
from . import views                #从此文件所在的目录中导入wiews

app_name='learning_logs'           #巨坑,书中没有,不写runserver时会报错
#URL模式是一个对函数path的调用,第一个参数是正则表达式,第二个参数指定了调用的视图函数第三个参数指定名称
#当需要提供到这个主页的链接时,都将使用这个名称而不是编写URL
urlpatterns = [                    #包含可在learning_logs中请求的网页
    path('', views.index, name='index'),                             # 主页
    path('topics/', views.topics, name='topics'),                    # 显示所有的主题
    re_path('topics/(?P<topic_id>\d+)/', views.topic, name='topic'),    # 特定主题的详细页面
    #使用正则表达式要用re_path否则控制台报警告
   ]

这里采用了正则表达式。这里捕获URL中的值?P和topic_id意思是将捕获到的值储存到topic_id中,\d+指与任何整数匹配,不管多少位。

views.py:

from django.shortcuts import render
from .models import Topic                  #导入所需数据相关的模型

def index(request):
    return render(request,'learning_logs/index.html')

def topics(request):                       #形参是Django从服务器收到的request对象
    #显示所有主题
    topics = Topic.objects.order_by('date_added')  #查询数据库,请求提供Topic对象,按属性date_added进行排序
    # 定义一个要发送给模板的上下文字典,键是模板中访问数据的名称,值是我们发送给模板的数据
    context = {'topics':topics}
    return render(request,'learning_logs/topics.html',context)#这里多传递一个数据

def topic(request,topic_id):
    #显示单个主题及其所有条目
    topic = Topic.objects.get(id=topic_id)                 #获取指定的主题
    entries = topic.entry_set.order_by('-date_added')      #获取与该主题相关联的条目,按date_added降序排序
    context = {"topic":topic,'entries':entries}            #数据储存在字典中发送给模板topic.html
    return render(request,'learning_logs/topic.html',context)

topic.html:

{% extends "learning_logs/base.html" %}

{% block content %}
<p>Topic:{{ topic }}</p>
<ul>
    {% for entry in entries %}
        <li>
            <p>{{ entry.date_added|date:'M d, Y H:i }}</p>
            <p>{{ entry.text|linebreaks }}</p>
        </li>
    {% empty %}
        <li>
         There are no entries for this topic yet.
        </li>
    {% endfor %}
</ul>
{% endblock content %}

首先使用了包含在字典context里的topic作为主题。然后遍历entries条目,显示出属性date_added的值,竖线|表示模板过滤器,对模板变量的值修改的函数,过滤器date:’M d, Y H:i以这样的格式显示时间戳:January 1,2021 23:00。接下来的一行享受text的完整值,过滤器linebreaks将包含换行符的长条目转换为浏览器能够理解的格式,以免显示一个不间断的文本块。

将所有主题页面中的每个主题都设置为链接:

topics.html:

{% extends "learning_logs/base.html" %}
{% block content %}
<p>Topics</p>
<ul>
    {% for topic in topics %}
    <li>
        <a href="{% url 'learning_logs:topic' topic.id %}">{{ topic }}</a>
    </li>
    {% empty %}
    <li>No topics have been added yet.</li>
    {% endfor %}
</ul>
{% endblock content %}

再次访问网页发现功能已经实现:


用户账户

Web应用程序的核心是让任何用户都能够注册账户并能够使用它。本章中,你将创建一些表单让用户能够添加主题和条目,以及编辑现有条目。你还将学习Django如何防范对基于表单的网页发起的常见攻击。然后,我们将实现一个用户身份验证系统,创建一个注册页面,供用户注册,并让有些页面只让已登录的用户访问。接下来,修改一些视图参数,使用户只能看到自己的数据。

让用户能输入数据

首先让用户能添加新主题,与前面的方法几乎一样,主要差别是需要导入包含表单的模块form.py。

用户输入并提交的信息都是表单,我们需要验证提供的信息是正确的数据类型且不是恶意信息,再对有效的信息进行处理。创建表单最简单的方式是使用ModelForm,它根据我们之前定义的模型中的信息自动创建表单。接下来在models.py的目录下创建forms.py文件:

form.py:

from django import forms
from .models import Topic

class TopicForm(forms.ModelForm):
    class Meta:
        model = Topic               #根据模型Topic创建一个表单
        fields = ['text']           #该表单只包含字段text
        labels = {'text':''}        #不用为text生成标签

URL模式new_topic:

当用户要添加新主题时,我们将切换到http://localhost:8000/new_topic/。

learning_logs/urls.py:

--snip--
urlpatterns = [                    #包含可在learning_logs中请求的网页
	--snip--
    path('new_topic/', views.new_topic, name='new_topic'),           # 同于添加新主题的网页
   ]

views.py:

from django.shortcuts import render
from .models import Topic                  #导入所需数据相关的模型
from django.http import HttpResponseRedirect
from django.urls import reverse            #注意这里在django2.0的包名有所变化
from .forms import TopicForm

--snip--
def new_topic(request):
    if request.method != 'POST':           #判断请求是GET还是POST,如未提交数据(点击进入页面链接),创建一个新表单
        form = TopicForm()
    else:                                  #POST请求,对提交的表单数据进行处理(点击了提交按钮)
        form = TopicForm(request.POST)     #使用用户输入的数据创建一个TopicForm实例,储存在form中
        if form.is_valid():                #检查填写了所有必不可少的字段(默认所有),且输入的数据与要求的类型一致
            form.save()                    #表单中的数据写入数据库
            return HttpResponseRedirect(reverse('learning_logs:topics'))#获取topics的URL,重新定位到页面topics
    context = {'form': form}
    return render(request,'learning_logs/new_topic.html',context)

创建Web应用程序时,用到的两种主要请求类型是GET请求和POST请求。只是从服务器读取数据的页面使用GET请求,用户需要通过表单提交信息使用POST请求,处理所有表单时,我们都将使用POST方法。

新建new_topic.html:

{% extends "learning_logs/base.html" %}

{% block content %}
<p>Add a new topic:</p>

<form action="{% url 'learning_logs:new_topic' %}" method="post">
    {% csrf_token %}
    {{ form.as_p }}
    <button name="submit">add topic</button>
</form>
{% endblock content %}

从form标签哪一行开始,定义了一个HTML表单,实参action告诉服务器提交的表单数据发送到视图函数new_topic(),以POST请求的方式。 csrf_token 标签防止攻击者利用表单获得对服务器未经授权的访问。显示表单,自动创建表单需要的全部字段,修饰符as_p以段落格式渲染所有表单元素。下边设置一个提交按钮。

topics.html添加一个到new_topic的链接:

{% extends "learning_logs/base.html" %}
{% block content %}
<p>Topics</p>
<ul>
   --snip--
</ul>

<a href="{% url 'learning_logs:new_topic' %}">Add a new topic:</a>

{% endblock content %}

现在打开服务器,打开网页,可以看到效果了:

添加新条目:

forms.py:

from django import forms
from .models import Topic,Entry

--snip--
class EntryForm(forms.ModelForm):
    class Meta:
        model = Entry
        field = {'text'}
        labels = {'text':''}
        #设置widgets可覆盖Django选择的默认小部件。forms.Textarea定制字段'text'的输入小部件,并设置文本宽度80列
        widgets = {'text': forms.Textarea(attrs={'cols':80})}

learning_logs/urls.py:

urlpatterns = [                    #包含可在learning_logs中请求的网页
	--snip--
    re_path('new_entry/(?P<topic_id>\d+)/', views.new_entry, name='new_entry'), # 添加新条目的网页
   ]

views.py:

--snip--
from .forms import TopicForm,EntryForm
--snip--
def new_entry(request,topic_id):
    topic = Topic.objects.get(id=topic_id)
    if request.method != 'POST':
        form = EntryForm()
    else:
        form = EntryForm(data=request.POST)          #填充数据
        if form.is_valid():
            new_entry = form.save(commit=False)      #保存到new_entry中,但不保存到数据库中
            new_entry.topic = topic                  #获取主题
            new_entry.save()                         #保存到数据库,与主题关联
            #列表args包含在URL中的所有实参,在这里只有一个
            return HttpResponseRedirect(reverse('learning_logs:topic',args=[topic_id]))
    context = {'topic': topic,'form': form}
    return  render(request,'learning_logs/new_entry.html',context)

new_entry.html:

{% extends "learning_logs/base.html" %}

{% block content %}
<p><a href="{% url 'learning_logs:topic' topic.id %}">{{ topic }}</a></p>

<p>Add a new entry:</p>
<form action="{% url 'learning_logs:new_entry' topic.id %}" method="post">
    {% csrf_token %}
    {{ form.as_p}}
    <button name="submit">add entry</button>
</form>
{% endblock content %}

页面的顶端是主题名,同时也是一个链接。表单的实参action包含topic_id值,让视图函数能将新条目关联到正确的主题。

更改topic.html,添加链接:

{% extends "learning_logs/base.html" %}
{% block content %}
<p>Topic:{{ topic }}</p>
<p>Entries</p>
<p>
    <a href="{% url 'learning_logs:new_entry' topic.id %}">add new entry</a>
</p>
<ul>
--snip--
</ul>
{% endblock content %}

这时网页就可以看到效果了:

编辑条目

创建一个页面,让用户能编辑现有的条目。

urls.py:

--snip--
urlpatterns = [                    #包含可在learning_logs中请求的网页
	--snip--
    re_path('edit_entry/(?P<topic_id>\d+)/', views.edit_entry, name='edit_entry'), # 编辑条目页面
   ]

views.py:

from django.shortcuts import render
from .models import Topic,Entry             #导入所需数据相关的模型
from django.http import HttpResponseRedirect
from django.urls import reverse            #注意这里在django2.0的包名有所变化
from .forms import TopicForm,EntryForm

--snip--
def edit_entry(request,entry_id):
    entry = Entry.objects.get(id-entry_id)
    topic = entry.topic
    if request.method != 'POST':
        form = EntryForm(instance=entry)                    #根据现有条目创建表单
    else:
        form = EntryForm(instance=entry,data=request.POST)  #根据现有条目创建表单,根据POST内容对其进行修改
        if form.is_valid():
            form.save()
            return HttpResponseRedirect(reverse('learning_logs:topic',args=[topic.id]))
    context = {'entry':entry,'topic':topic,'form':form}
    return render(request,'learning_logs/edit_entry.html',context)

新建edit_entry.html:

{% extends "learning_logs/base.html" %}

{% block content %}
<p><a href="{% url 'learning_logs:topic' topic.id %}">{{ topic }}</a></p>

<p>Edit entry:</p>
<form action="{% url 'learning_logs:edit_entry' entry.id %}" method="post">
    {% csrf_token %}
    {{ form.as_p}}
    <button name="submit">save changes</button>
</form>
{% endblock content %}

修改topic.html:

--snip--
        <li>
            <p>{{ entry.date_added|date:'M d, Y H:i' }}</p>
            <p>{{ entry.text|linebreaks }}</p>
            <a href="{% url 'learning_logs:edit_entry' entry.id %}">edit entry</a>
        </li>
--snip--

这时打开网页可以看到效果:

创建用户账户

下面将建立用户注册和身份验证系统,我们将创建一个新的应用程序,包含与处理用户账户相关的所有功能。

首先使用命令创建名为users的应用程序,结构与learning_logs相同:

python manage.py startapp users

settings.py

--snip--
INSTALLED_APPS = [                              #安装在项目中的应用程序
	--snip--
    #我的应用程序
    'learning_logs',
    'users',
]

learning_log/urls.py:

from django.contrib import admin
from django.urls import path,include

urlpatterns = [
    path('admin/', admin.site.urls),
    path('', include('learning_logs.urls', namespace='learning_logs')),
    path('users/', include('users.urls', namespace='users')),
]

登录页面:

在users文件夹中新建urls.py:

from django.urls import path,re_path
from . import views
from django.contrib.auth.views import LoginView         #导入默认视图LoginView

app_name='users'
#URL模式是一个对函数path的调用,第一个参数是正则表达式,第二个参数指定了调用的视图函数第三个参数指定名称
#当需要提供到这个主页的链接时,都将使用这个名称而不是编写URL
urlpatterns = [
    #鉴于没有编写视图函数而是使用默认的LoginView,后面的as_view告诉Django去哪里寻找模板
    path('login/', LoginView.as_view(template_name='users/login.html'), name='login')
   ]

接下来在users中创建一个templates目录,在templates中创建一个users目录,又在这个users目录中创建login.html:

{% extends "learning_logs/base.html" %}

{% block content %}
  {% if form.errors %}
    <p>>Your username and password didn't match.Please try again.</p>
   {% endif %}
   <form method="post" action=" {% 'users:login' %}">
       {% csrf_token %}
       {{ form.as_p }}
       <button name="submit">log in</button>
       <input type="hidden" name="next" value="{% url 'learning_logs:index' %}" />
   </form>
{% endblock content %}

首先如果表单的errors属性被设置,就显示错误信息,表示用户名密码不匹配。要让登录视图处理表单,因此将action设置为登录页面的URL,登录视图发送一个表单给模板,模板中显示这个表单并添加一个提交按钮。最后包含了一个隐藏表单元素next,其中的参数value告诉Django在用户登录成功后定位到主页。

下面在base.html添加到登录页面的链接:

<p>
    <a href="{% url 'learning_logs:index' %}">learning Log</a> -
    <a href="{% url 'learning_logs:topics' %}">Topics</a> -
    {% if user.is_authenticated %}
      Hello,{{ user.username }}.
    {% else %}
      <a href="{% url 'users:login' %}">log in</a>
    {% endif %}
</p>

{% block content %}{% endblock content %}

在Django身份验证系统中,每个模板都可使用变量user,这个变量的is_authenticated属性在登录时为True,否则为False。

此时在admin页面退出管理员账户,就可以在网页中看到log in按钮,点击它:

注销:

让用户点击一个按钮就可注销并返回主页:

users/urls.py:

--snip--
urlpatterns = [
    path('login/', LoginView.as_view(template_name='users/login.html'), name='login')
    path('logout/', views.logout_view, name='logout')
   ]

users/views.py(注意新版的包名有所变化):

from django.urls import reverse
from django.contrib.auth import logout
from django.http import HttpResponseRedirect

def logout_view(request):
    logout(request)
    return HttpResponseRedirect(reverse('learning_logs:index'))

建立链接,修改base.html,给登录后的用户名旁边加上log out按钮:

<p>
    <a href="{% url 'learning_logs:index' %}">learning Log</a> -
    <a href="{% url 'learning_logs:topics' %}">Topics</a> -
    {% if user.is_authenticated %}
      Hello,{{ user.username }}.
      <a href="{% url 'users:logout' %}">log out</a>
    {% else %}
      <a href="{% url 'users:login' %}">log in</a>
    {% endif %}
</p>
{% block content %}{% endblock content %}

注册页面:

users/urls.py:

--snip--
urlpatterns = [
    path('login/', LoginView.as_view(template_name='users/login.html'), name='login'),
    path('logout/', views.logout_view, name='logout'),
    path('register/', views.register, name='register'),
   ]

users/views.py:

from django.urls import reverse
from django.contrib.auth import logout,login,authenticate
from django.http import HttpResponseRedirect
from django.shortcuts import render
from django.contrib.auth.forms import UserCreationForm

def logout_view(request):
    logout(request)
    return HttpResponseRedirect(reverse('learning_logs:index'))

def register(request):
    if request.method != 'POST':                 #点击注册,创建一个表单
        form = UserCreationForm()
    else:                                        #点击提交,录入信息,切换登录状态,转到主页
        form = UserCreationForm(data=request.POST)
        if form.is_valid():
            new_user = form.save()
            authenticated_user = authenticate(username=new_user.username,password=request.POST['password1'])
            login(request,authenticated_user)
            return HttpResponseRedirect(reverse('learning_logs:index'))
    context = {'form':form}
    return render(request,'users/register.html',context)

保存用户信息后,我们调用authenticate(),将实参用户名和密码传递给它,返回一个通过了 身份验证的用户对象,然后调用login登录。用户注册时被要求输入密码两次,输入两次相同表单才可能有效。

在login.html目录下新建register.html:

{% extends "learning_logs/base.html" %}

{% block content %}

<form action="{% url 'users:register' %}" method="post">
    {% csrf_token %}
    {{ form.as_p}}
    <button name="submit">register</button>
    <input type="hidden" name="next" value="{% url 'learning_logs:index' %}" />
</form>
{% endblock content %}

在base.html添加用户在没有登录时显示到注册页面的链接:

--snip--
    {% else %}
      <a href="{% url 'users:register' %}">register</a> -
      <a href="{% url 'users:login' %}">log in</a>
    {% endif %}
--snip--

现在可以看到效果了:

让用户拥有自己的数据

这里将创建一个系统,确定各项数据所属用户,再限制对页面的访问,让用户只能使用自己的数据。

Django提供了修饰器@login_required,可以实现某些页面只允许已登录的用户访问。

修改learning_logs/views.py:

--snip--
from django.contrib.auth.decorators import login_required

--snip--
@login_required
def topics(request):                      
    --snip--
@login_required
def topic(request,topic_id):

@login_required
def new_topic(request):
	--snip--
@login_required
def new_entry(request,topic_id):
	--snip--
@login_required
def edit_entry(request,entry_id):
	--snip--

login_required()检查用户是否已登录,当用户已登录时,才运行topics的代码,否则重定向到登录页面。

修改settings.py,让Django知道到哪里查找登录页面,在末尾添加:

LOGIN_URL = '/users/login/'

现在未登录状态下点击topics或输入编辑添加主题条目的链接,就会跳转到登录页面。

将数据关联到用户:

我们只需将最高层的数据关联到用户,这样更底层的数据自动管理到用户。下面修改模型Topic,添加一个关联到用户的外键,完成后必须对数据库进行迁移。最后必须对一些视图进行修改,使其只显示与当前登录的用户相关联的数据。

models.py

from django.db import models
from django.contrib.auth.models import User

class Topic(models.Model):          
    text = models.CharField(max_length=200) 
    date_added = models.DateTimeField(auto_now_add=True)  
    owner = models.ForeignKey(User,on_delete=models.CASCADE)
    def __str__(self):             #显示模型的简单表示
        return self.text
--snip--

确定当前数据库有哪些用户,输入命令启动Django shell会话:

$ python manage.py shell
Python 3.9.1 (tags/v3.9.1:1e5d33e, Dec  7 2020, 17:08:21) [MSC v.1927 64 bit (AMD64)] on win32
Type "help", "copyright", "credits" or "license" for more information.
(InteractiveConsole)
>>> from django.contrib.auth.models import User
>>> User.objects.all()
......................................
>>> for user in User.objects.all():
...     print(user.username,user.id)
...
tianjue 1
testAccount 2
>>>

迁移数据库:

$ python manage.py makemigrations learning_logs
You are trying to add a non-nullable field 'owner' to topic without a default; we can't do that (the database needs something to populate existing rows).
Please select a fix:
 1) Provide a one-off default now (will be set on all existing rows with a null value for this column)
 2) Quit, and let me add a default in models.py
Select an option: 1
Please enter the default value now, as valid Python
The datetime and django.utils.timezone modules are available, so you can do e.g. timezone.now
Type 'exit' to exit this prompt
>>> 1
Migrations for 'learning_logs':
  learning_logs\migrations\0003_topic_owner.py
    - Add field owner to topic

执行迁移:

$ python manage.py migrate

现在可以在shell中验证是否符合预期:

>>> from learning_logs.models import Topic
>>> for topic in Topic.objects.all():
...     print(topic,topic.owner)
...
Chess tianjue
Rock Climbing tianjue
Learning tianjue

可以看到每个主题都属于用户tianjue了。

只允许访问自己的主题:

learning_logs/views.py:

--snip--
from django.http import HttpResponseRedirect,Http404
--snip--
@login_required
def topics(request):                     
    # 查询数据库,请求提供Topic对象,只让所有者访问,按属性date_added进行排序
    topics = Topic.objects.filter(owner=request.user).order_by('date_added')  
    context = {'topics':topics}
    return render(request,'learning_logs/topics.html',context)

@login_required
def topic(request,topic_id):
    topic = Topic.objects.get(id=topic_id)                
    if topic.owner != request.user:                               #如果主题不归用户所有,返回404响应
        raise  Http404
    entries = topic.entry_set.order_by('-date_added')     
    context = {'topic':topic,'entries':entries}          
    return render(request,'learning_logs/topic.html',context)
@login_required
def new_topic(request):
    if request.method != 'POST':           
        form = TopicForm()
    else:                                  
        form = TopicForm(request.POST)     
        if form.is_valid():               
            new_topic = form.save(commit=False)
            new_topic.owner = request.user                         #把新建立的主题和用户关联上
            new_topic.save()                    
            return HttpResponseRedirect(reverse('learning_logs:topics'))
    context = {'form': form}
    return render(request,'learning_logs/new_topic.html',context)
@login_required
def new_entry(request,topic_id):
    topic = Topic.objects.get(id=topic_id)
    if topic.owner != request.user:                              #如果主题不归用户所有,返回404响应
        raise Http404
	--snip--
@login_required
def edit_entry(request,entry_id):
    entry = Entry.objects.get(id=entry_id)
    topic = entry.topic
    if topic.owner != request.user:                               #如果主题不归用户所有,返回404响应
        raise Http404
    --snip--

现在这个项目运行任何用户注册,每个用户可以随意添加主题和条目,并且只能访问自己的数据。


设置应用程序的样式并对其进行部署

设置项目的样式

当前,学习笔记的功能基本完成,但未设置样式。我们将使用Bootstrap库,这是一种工具,用于为Web应用程序设置样式,最后把这个项目部署到服务器端。

执行命令,安装django-bootstrap3:

$ pip3 install django-bootstrap3

在settings.py中添加代码:

--snip--
INSTALLED_APPS = [                              
    --snip--
    #第三方应用程序
    'bootstrap3',
    #我的应用程序
    'learning_logs',
    'users',
]
--snip--
BOOTSTRAP3 ={
    'include_jquery': True,
}

需要让django-bootstrap3包含jQuery,这是一个JavaScript库,能够让你使用Bootstrap模板的一些交互性元素,这样无需手工下载jQuery。

Bootstarp是一个大型样式设置工具集,提供了大量的模板,具体可访问https://getbootstrap.com/。

首先需要修改base.html,在这个文件定义HTML头部,添加一些在模板中使用Bootstrap所需的信息,删除base.html全部代码,改为:

首先,HTML分为两个主要部分:头部(head)和主体(body),头部不包含任何内容,只是将正确显示页面所需的信息告诉浏览器。

主体包含用户在页面上看到的内容<nav>元素表示页面的导航链接部分,这个元素内的所有内容,都根据(selector)navbar、navbar-default、navbar-static-top定义的Bootstrap样式规则来设置样式,选择器决定了特定的样式规则应用于页面的哪些元素。

在class=”navbar-header”的地方,定义了一个按钮<button>,将浏览器窗口太窄,无法水平显示的整个导航栏显示出来,如果用户点击这个按钮,会出现一个下拉列表,包含所有的导航元素,在用户缩小浏览器窗口或在屏幕较小的移动设备上显示网页,collapse会使导航栏折叠起来。 在class=”navbar-brand”的地方表示在导航栏的左边显示项目名,设置为主页链接。

div id=”navbar” class=”navbar-collapse collapse”定义了一组让用户能够在网站中导航的链接,导航栏是一个以ul 开头的列表,其中的每个链接都是列表项 li,要添加更多的链接,可插入下述结构的行:

<li><a href="{% url 'learning_logs:title' %}">Title</a></li>

后面的部分是一个容器,包含一个名为header的块和content块,header块决定页面包含哪些消息以及用户可在页面上执行哪些操作,其属性page-header将一系列的样式应用于这个块。

现在打开浏览器可以看到网页发生了很大改变:

设置登录页面的样式:

login.html:

{% extends "learning_logs/base.html" %}
{% load bootstrap3 %}

{% block header %}
  <h2>Log in to your account.</h2>
{% endblock header %}

{% block content %}
   <form method="post" action="{% url 'users:login' %}" class="form">
       {% csrf_token %}
       {% bootstrap_form form %}
       {% buttons %}
         <button name="submit" class="btn btn-primary">log in</button>
       {% endbuttons %}
       <input type="hidden" name="next" value="{% url 'learning_logs:index' %}" />
   </form>
{% endblock content %}

首先是加载bootstrap3模板,然后定义header块,原来的if form.error代码块删除了,因为bootstrap3会自动管理表单错误。

后面用bootstrap_form form显示表单,替换了原来的form.as_p。后面的按钮用了bootstrap3模板标签。

现在 访问login页面,样式已经改变:

设置new_topic页面的样式:

修改new_topic.html:

{% extends "learning_logs/base.html" %}
{% load bootstrap3 %}

{% block header %}
  <h2>Add a new topic:</h2>
{% endblock header %}

{% block content %}
<form action="{% url 'learning_logs:new_topic' %}" method="post" class="form">
    {% csrf_token %}
    {% bootstrap_form form %}
    {% buttons %}
    <button name="submit" class="btn btn-primary">add topic</button>
    {% endbuttons %}
</form>
{% endblock content %}

可以看到效果:

设置topics页面的样式:

topics.html:

{% extends "learning_logs/base.html" %}

{% block header %}
<h1>Add a new topic:</h1>
{% endblock header %}

{% block content %}
<ul>
    {% for topic in topics %}
    <li>
        <h3>
            <a href="{% url 'learning_logs:topic' topic.id %}">{{ topic }}</a>
        </h3>
    </li>
    {% empty %}
    <li>No topics have been added yet.</li>
    {% endfor %}
</ul>
<h3><a href="{% url 'learning_logs:new_topic' %}">Add a new topic:</a></h3>
{% endblock content %}

这里没有使用bootstrap3自定义标签,只是加了header块,改了字体大小。

设置topic页面中的条目样式:

topic.html:

{% extends "learning_logs/base.html" %}

{% block header %}
<h2>{{ topic }}</h2>
{% endblock header %}

{% block content %}
<p>
    <a href="{% url 'learning_logs:new_entry' topic.id %}">add new entry</a>
</p>

{% for entry in entries %}
  <div class="panel panel-default">
      <div class="panel-heading">
          <h3>
              {{ entry.date_added|date:'M d, Y H:i' }}
              <small>
                  <a href="{% url 'learning_logs:edit_entry' entry.id %}">edit entry</a>
              </small>
          </h3>
      </div>
      <div class="panel-body">
          {{ entry.text|linebreaks }}
      </div>
  </div><!-- panel -->
{% empty %}
    There are no entries for this topic yet.
{% endfor %}
{% endblock content %}

删除了以前使用的无序列表结构,创建了面板式div元素,而不是将每一个条目作为一个列表项,其中有两个嵌套div:面板标题(panel-heading)div和面板主体(panel-body)div。面板标题div包含条目的创建日期以及用于编辑条目的链接,还使用了标签small使其比时间戳小一些。面板主体包含实际文本。

效果如下:

同样,给其他页面设置:

new_entry.html:

{% extends "learning_logs/base.html" %}
{% load bootstrap3 %}

{% block header %}
<h2><a href="{% url 'learning_logs:topic' topic.id %}">{{ topic }}</a></h2>
<h3>Add a new entry:</h3>
{% endblock header %}

{% block content %}
<form action="{% url 'learning_logs:new_entry' topic.id %}" method="post" class="form">
    {% csrf_token %}
    {% bootstrap_form form %}
    {% buttons %}
    <button name="submit" class="btn btn-primary">add entry</button>
    {% endbuttons %}
</form>
{% endblock content %}

edit_entry.html:

{% extends "learning_logs/base.html" %}
{% load bootstrap3 %}

{% block header %}
<h2><a href="{% url 'learning_logs:topic' topic.id %}">{{ topic }}</a></h2>
<h3>Edit entry:</h3>
{% endblock header %}

{% block content %}
<form action="{% url 'learning_logs:edit_entry' entry.id %}" method="post" class="form">
    {% csrf_token %}
    {% bootstrap_form form %}
    {% buttons %}
    <button name="submit" class="btn btn-primary">save changes</button>
    {% endbuttons %}
</form>
{% endblock content %}

部署学习笔记

由于在墙内是注册不了书本上要求的网站的,没办法还原书本的操作。我们需要按自己的方法部署。这部分需要话费的精力较多,还需要购买云服务器,需要一定的时间成本和金钱成本,具体可参考视频python3 django项目部署方案Nginx + uWsgi 部署 Django + Mezzanine 生产服务器。等我以后有需求时会写一篇新博文来完善此处内容。


最后的话

至此,pyhon的基本知识学习完毕了,我们掌握了python的基本语法,能够自己下载和处理一些数据了,还能用Django搭建自己的网站,现在已经具备了开发各种项目所需的python基本技能。在学习的过程中,肯定会遇到自己不能解决的问题,这时候,查阅资料的能力尤为关键,其中给我最大帮助的是CSDN,遇到的大多数问题都能在其中找到答案,最后郑重提醒:每学完一项技术时,一定要写博客,写博客,写博客,很重要!这对知识的巩固有至关重要的作用,博客的搭建可以参考我的另一篇博文《个人建立hexo博客Matery主题的过程心得》。

祝贺你在学习Python的道路上走出了坚实的一步,愿你在以后的学习中好运相伴!


文章作者: 微笑紫瞳星
版权声明: 本博客所有文章除特別声明外,均采用 CC BY 4.0 许可协议。转载请注明来源 微笑紫瞳星 !
  目录