学习使用turtle在屏幕上绘制图形。
说明:turtle是Python内置的一个非常有趣的模块,特别适合对计算机程序设计进行初体验的小伙伴,它最早是Logo语言的一部分,Logo语言是Wally Feurzig和Seymour Papert在1966发明的编程语言。
1 import turtle
2
3 turtle.pensize(4)
4 turtle.pencolor('red')
5
6 turtle.forward(100)
7 turtle.right(90)
8 turtle.forward(100)
9 turtle.right(90)
10
11 turtle.mainloop()
变量和类型
变量名由字母(广义的Unicode字符,不包括特殊字符)、数字和下划线构成,数字不能开头。 用type()检查变量的类型。
- 整型:Python中可以处理任意大小的整数(Python 2.x中有
int
和long
两种类型的整数,因此在Python 3.x中整数只有int这一种),而且支持二进制(如0b100
,换算成十进制是4)、八进制(如0o100
,换算成十进制是64)、十进制(100
)和十六进制(0x100
,换算成十进制是256)的表示法。 - 浮点型:浮点数也就是小数,浮点数除了数学写法(如
123.456
)之外还支持科学计数法(如1.23456e2
)。 - 字符串型:字符串是以单引号或双引号括起来的任意文本,比如
'hello'
和"hello"
,字符串还有原始字符串表示法、字节字符串表示法、Unicode字符串表示法。 - 布尔型:布尔值只有
True
、False
两种值。在Python中可以直接用True
、False
表示布尔值(请注意大小写),也可以通过布尔运算计算出来(例如3 < 5
会产生布尔值True
,而2 == 1
会产生布尔值False
)。 - 复数型:形如
3+5j
,跟数学上的复数表示一样,唯一不同的是虚部的i
换成了j
。了解一下就可以了
可以使用Python中内置的函数对变量类型进行转换。
-
int()
:将一个数值或字符串转换成整数,可以指定进制。 -
float()
:将一个字符串转换成浮点数。 -
str()
:将指定的对象转换成字符串形式,可以指定编码。 -
chr()
:将整数转换成该编码对应的字符串(一个字符)。 -
ord()
:将字符串(一个字符)转换成对应的编码(整数)。
a = int(input('a = ')) # 用input()函数获取键盘输入字符串
b = int(input('b = '))
print('%d %% %d = %d' % (a, b, a % b)) # %%表示百分号
运算符
短路处理: and
运算符左边为False
的情况下,右边的表达式根本不会执行; or 左边的布尔值为True
的情况下,右边的表达式根本不会执行
运算符 | 描述 |
| 下标,切片 |
| 指数 |
| 加,减, 乘,除,模,整除 |
| 右移,左移 |
| 按位与,按位异或,按位或, 按位取反 |
| 小于等于,小于,大于,大于等于,等于,不等于 |
| 身份运算符 |
| 成员运算符 |
| 逻辑运算符 |
| =^=>>=<<=` |
for x in range(101):
break
只能终止它所在的那个循环; continue
用来放弃本次循环后续的代码直接让循环进入下一轮。
-
range(101)
:产生0到100范围的整数,需要注意的是取不到101。 -
range(1, 101)
:产生1到100范围的整数,相当于前闭后开区间。 -
range(1, 101, 2)
:产生1到100的奇数,其中2是步长,即每次数值递增的值。 -
range(100, 0, -2)
:产生100到1的偶数,其中-2是步长,即每次数字递减的值。
函数
在Python中,函数的参数可以有默认值,也支持使用可变参数,所以Python并不需要像其他语言一样支持函数的重载,
def add(*args): # 可变参数
total = 0
for val in args:
total += val
return total
Python查找一个变量时会按照“局部作用域”、“嵌套作用域”、“全局作用域”和“内置作用域”的顺序进行搜索,所谓的“内置作用域”就是Python内置的那些标识符,我们之前用过的input
、print
、int
等都属于内置作用域。
'''用global关键字来指示foo函数中的变量a来自于全局作用域,如果全局作用域中没有a,下面一行代码就会定义变量a并将其置于全局作用域。同理,如果希望函数内部的函数能够修改嵌套作用域中的变量,可以使用nonlocal关键字来指示变量来自于嵌套作用域'''
def foo():
global a
a = 200
print(a) # 200
# 只有被Python解释器直接执行的模块的名字才是__main__
if __name__ == '__main__':
a = 100
foo()
print(a) # 200
- 将函数视为“一等公民”
- 函数可以赋值给变量
- 函数可以作为函数的参数
- 函数可以作为函数的返回值
- 高阶函数的用法(
filter
、map
以及它们的替代品)
items1 = list(map(lambda x: x ** 2, filter(lambda x: x % 2, range(1, 10)))) # [1, 9, 25, 49, 81]
items2 = [x ** 2 for x in range(1, 10) if x % 2]
- 位置参数、可变参数、关键字参数、命名关键字参数
- 参数的元信息(代码可读性问题)
- 匿名函数和内联函数的用法(
lambda
函数) - 闭包和作用域问题
- Python搜索变量的LEGB顺序(Local >>> Embedded >>> Global >>> Built-in)
global
和nonlocal
关键字的作用global
:声明或定义全局变量(要么直接使用现有的全局作用域的变量,要么定义一个变量放到全局作用域)。nonlocal
:声明使用嵌套作用域的变量(嵌套作用域必须存在该变量,否则报错)。
自定义装饰器:
from functools import wraps
def singleton(cls):
instances = {}
@wraps(cls)
def wrapper(*args, **kwargs):
if cls not in instances:
instances[cls] = cls(*args, **kwargs)
return instances[cls]
return wrapper
@singleton
class President:
"""总统(单例类)"""
pass
用装饰器来实现单例模式
from functools import wraps
from threading import RLock
def singleton(cls):
instances = {}
locker = RLock()
@wraps(cls)
def wrapper(*args, **kwargs):
if cls not in instances:
with locker:
if cls not in instances:
instances[cls] = cls(*args, **kwargs)
return instances[cls]
return wrapper
类和对象
PEP 8要求: 受保护实例属性用单个下划线开头; 私有实例属性用两个下划线开头
封装、继承和多态
class Test:
NUM = 11 # 静态变量
def __init__(self, foo):
self.__foo = foo
def __bar(self):
print(self.__foo)
def main():
test = Test('hello')
Test.NUM
test.__bar() # AttributeError: 'Test' object has no attribute '__bar'
test._Test__bar() # Test._Test__bar(test) 也可通过类调用对象方法但要传入对象作为参数。
print(test._Test__foo)
if __name__ == "__main__":
main()
@property装饰器 | @staticmethod | @classmethod | __slots__: 只对当前类的对象生效,对子类并不起任何作用 | @abstractmethod |
@property | @staticmethod def is_valid(a, b, c) | @classmethod def now(cls): # 通过cls可获取和类相关信息并创建出类对象 cls("foo") | # 限定Person对象只能动态绑定_name, _age和_gender属性 __slots__ = ('_name', '_age', '_gender') | 抽象方法 --> 抽象类 |
- 实例方法:可获取类属性、构造函数定义的变量,属于 method 类型。只能通过实例化调用。
- 静态方法:不能获取类属性、构造函数定义的变量,属于 function 类型。两种调用方式:类.方法名 ,实例化调用。
- 类方法 :可获取类属性,不能获取构造函数定义的变量,属于 method 类型。两种调用方式:类.方法名 ,实例化调用。
类和类之间的关系有三种:is-a、has-a和use-a关系。
- is-a: 继承或泛化
- has-a: 关联,比如部门和员工的关系,汽车和引擎的关系都属于关联关系;关联关系如果是整体和部分的关联,那么我们称之为聚合关系;如果整体进一步负责了部分的生命周期(整体和部分是不可分割的,同时同在也同时消亡),那么这种就是最强的关联关系,我们称之为合成关系。
- use-a: 依赖,比如司机有一个驾驶的行为(方法),其中(的参数)使用到了汽车,那么司机和汽车的关系就是依赖关系。
继承: 方法重写
def __init__(self, name, age, title):
super().__init__(name, age)
self._title = title
对象的复制(深复制/深拷贝/深度克隆和浅复制/浅拷贝/影子克隆)
垃圾回收、循环引用和弱引用
Python使用了自动化内存管理,这种管理机制以引用计数为基础,同时也引入了标记-清除和分代收集两种机制为辅的策略。
- 引用计数+1: 对象被创建、被引用、作为函数参数、作为元素存在容器中;
- 引用计数-1: 对象别名被显示销毁、被赋予新对象、离开作用域、所在容器被销毁或从容器中删除对象。
- 垃圾回收:调用
gc.collect()、gc模块的计数器达到阀值、程序退出。
引用计数可能会导致循环引用问题,而循环引用会导致内存泄露——解决:引入“标记-清除”和“分代收集”。创建对象时,对象被放在第一代中,如果在第一代的垃圾检查中对象存活了下来,该对象就会被放到第二代中,同理在第二代的垃圾检查中对象存活下来,该对象就会被放到第三代中。