一.第一类对象, 函数名的使用
函数名就是变量名, 函数名存储的是函数的内存地址
- 变量的命名规范:
- 由数字, 字母, 下划线组成
- 不能是数字开头, 更不能是纯数字
- 不能用关键字
- 不要太长
- 要有意义
- 不要用中文
- 区分大小写
- 驼峰或者下滑线
2. 闭包
闭包: 在内层函数中引入外层函数的变量
作用:
- 保护变量不受侵害(javascript)
- 让一个变量常驻内存
*闭包的作用:保护我们的变量,必须是局部变量 a = 10 # 全局的东西都是不安全的 def func1(): print(a) def func2(): print(a) func1() # 你的同事搞事情 def func3(): global a a = 20 func3() func2() def func(): a = 10 # 安全的 def func2(): print(a) def func3(): print(a) def func4(): nonlocal a a = 20 闭包: 内层函数对外层函数的变量的使用 作用: 1. 保护我们的变量不受侵害 2. 可以让一个变量常驻内存. def outer(): # a = 10 def inner(): print("哈哈") print(inner.__closure__) # (<cell at 0x000001C079677588: int object at 0x0000000054776D30>,) return inner ret = outer() # 1800行代码... ret() # 由于inner函数执行的时机是不确定的. 必须要保证innter可以正常执行. 必须把a保留到最后 # 爬虫 (low版本) from urllib.request import urlopen # 打开一个连接用的模块 # 外层函数 def but(): # 打开连接. 读取源代码 content = urlopen("http://www.cctv.com/").read() # 永久驻留在内存 # 内层函数 def get_content(): # 返回content return content return get_content # 内层函数 fn = but() # 这里会很慢. 需要网络请求 print(fn()) # 不会再进行网络请求了 print(fn()) 关联的小点 def func(a, b): ''' 文档注释 这个函数用来计算两个数的和并返回 :param a: 第一个数 :param b: 第二个数 :return: 第一个数和第二个数的和 author:sylar date:2018-10-31 ''' print("我是func") return a + b print(func.__doc__) # document 文档注释 print(func.__name__) # name 名字 获取函数名
二. 迭代器
dir() 查看变量能够执行的方法(函数)
Iterator: 迭代器, __iter__(), __next__()
Iterable: 可迭代的, __iter__()
for循环的流程: it = lst.__iter__() while 1: try: el = it.__next__() for循环的循环体 except StopIteration: break
从迭代器中获取数据的唯一方法: __next__()
迭代器的三个特征:
- 省内存
- 惰性机制
- 只能往前. 不能后退
print(dir(str)) # 查看str能够执行的操作. 内部的方法 __iter__ 字符串可以被迭代. 发现了__iter__ print(dir(list)) print(dir(open("x",mode="w"))) # int中没有__iter__ 简单的下一个结论. 主要这个数据类型可以执行__iter__ 可以被迭代的数据类型 lst = ["汉高祖", "清高祖", "明高祖", "哈哈", "娃哈哈", "爽歪歪"] it = lst.__iter__() # <list_iterator object at 0x000001ED54B17128> iterator 迭代器 print(it) print(dir(it)) # 迭代器本身是可迭代的 可以使用__next__获取数据 print(it.__next__()) # 汉高祖 print(it.__next__()) # 清高祖 print(it.__next__()) # 明高祖 print(it.__next__()) # 明高祖 print(it.__next__()) # 明高祖 print(it.__next__()) # StopIteration 迭代器中没有元素了. for循环内部的代码 it = lst.__iter__() # 获取新的迭代器 while 1: try: el = it.__next__() print(el) except StopIteration: print("结束了") break list(内部有for循环) for内部 迭代器 lst = [1,55,5,55,5,5,5,555,55,555] ll = list(set(lst)) print(ll) list(1) # 'int' object is not iterable 如何判断一个数据是否是可迭代对象 1. dir() -> __iter__ 可迭代的 dir() -> __next__ 迭代器 lst = ["秦始皇", "汉武帝", "孝文帝", "隋炀帝", "李世民"] print("__iter__" in dir(lst)) # True 可迭代的 print("__next__" in dir(lst)) # False 不是迭代器 print("__iter__" in dir(int)) print("__next__" in dir(int)) it = lst.__iter__() # 迭代器 print("__iter__" in dir(it)) # True 迭代器本身就是可迭代的 print("__next__" in dir(it)) # True lst = ["秦始皇", "汉武帝", "孝文帝", "隋炀帝", "李世民"] # collections 关于集合类的相关操作 # Iterable : 可迭代的 # Iterator : 迭代器 from collections import Iterable, Iterator print(isinstance(lst, Iterable)) # True print(isinstance(lst, Iterator)) # False print(isinstance({1,2,3}, Iterable)) # True, 可以使用for循环
三..练习
1.写函数,接收一个参数(此参数类型必须是可迭代对象),将可迭代对象的每个元素以’_’相连接,形成新的字符串,并返回. 例如 传入的可迭代对象为[1,'老男孩','武sir']返回的结果为’1_老男孩_武sir’ def func(can): if "__iter__" in dir(can): # 判断是否是可迭代的 result = "" for item in can: # 传递的参数中的每一个元素 "周杰伦" result += str(item)+"_" return result.strip("_") else: # 不是可迭代的 return None # 回去 , 正常来讲这里应该抛出异常. print(func((1,'老男孩','武sir'))) 2.打印图形 * * * * * * * * * * * * * * * * n = int(input("请输入你要打印多少行")) for i in range(1, n+1): # 方案一 for k in range(n-i): print(" ", end="") for j in range(2 * i - 1): print("*", end="") print() # 换行 #方案二 for i in range(1,n+1) print(" " * (n - i) + "*" * (2 * i - 1)) 3.求1-100内所有的质数的和(升级题) def func(n): # 只能被1和自身整除的数 if n == 1: # 特殊处理的 return False # n / 2, 3, 4, 5, 6, 7, 8, 9....n-1 for i in range(2, n): if n % i == 0: # 不是质数 return False else: return True print(func(2)) sum = 0 for i in range(1, 101): if func(i): # 如果是质数. 帮我累加 sum += i print(sum)