python3包、模块、函数与变量作用域(7)

7.1 while循环与使用场景

常见的循环:while循环和for循环
while容易造成死循环

CONDITION = True

while CONDITION :
print('I am While')

死循环输出:I am While

因为CONDITION永远为True,没有被改变过
ctrl + c 结束死循环

counter = 0

while counter :
counter += 1
print(counter)

没有打印出任何结果,因为counter = 0,0就是False,所以不会打印出结果
现在把counter改成1

counter = 1

while counter :
counter += 1
print(counter)

打印的结果会一直刷屏,ctrl + c 结束循环
无限循环是while错误的用法,改成正确的用法:

counter = 1

while counter <= 10 :
    counter += 1
    print(counter)

打印出从2到11,这才是while正确的使用方法

如何避免while出现死循环?
首先while后面的条件判断语句不能是常量,如果是常量,条件判断的结果是永远不会改变的,如果想让while运行的次数是有限的,那么在while内部的代码块里面就必须要有能够影响条件判断的语句,在上式中的 counter += 1就是影响条件判断结果的语句,这样就能避免出现死循环;
while除了可以单独使用之外,还可以和 if 一样,和else结合使用,在上面的事例中,如果不加 else,一旦 counter >10 ,整个程序就结束了,在 counter >10,整个程序结束的时候,打印出一个结束的标志EOF,代码如下:

counter = 1

while counter <= 10 :
    counter += 1
    print(counter)
else:
    print('EOF')               #while结束后会执行else

结果打印出从2到11,还有一个EOF,else的作用就是当 while 后面的条件语句的返回结果是 False 的时候,将执行 else 分支后面的代码块。
while 的使用场景是有一个目标,当达到目标的时候,while 就结束了。递归情况下比较适合用while ,其他情况下建议用 for 。

7.2 for与for-else循环

for主要是用来遍历/循环 序列或者集合、字典,凡是可以表示组概念的python类型都可以用 for 循环来遍历;
对应于其他语言中的for each
事例:

a = ['apple','orange','banana','grape']
for x in a:
    print(x)    #x遍历到列表中哪个元素就输出哪个元素

运行结果是依次打印出了apple  orange   banana    grape  

代码块中还可以嵌套代码块:

a = [['apple','orange','banana','grape'],(1,2,3)]
for x in a:
    for y in x:
       print(y)      #打印出列表中所有元素

运行结果是依次打印出了apple  orange   banana    grape  1   2   3 

理论上来说,for 循环可以无限极的嵌套,默认情况下, print 打印出的元素都是以列的形式展示出来的,如果想每个元素都在一行打印出来,可以在 print 后面加一个 end=’ ’

a = [['apple','orange','banana','grape'],(1,2,3)]
for x in a:
    for y in x:
       print(y,end ='')                #一横行打印

for 也可以和 else 搭配使用

a = [['apple','orange','banana','grape'],(1,2,3)]
for x in a:
    for y in x:
       print(y)
else:
    print('fruit is gone')

运行结果是遍历打印出列表中的所有元素后,最末尾打印出了 fruit is gone
当列表中的所有元素都被遍历完之后,else 就会被执行

利用break强行终止循环

a = [1,2,3]
for x in a:
    if x == 2:
        break
    print(x)

运行结果是1

利用continue跳过某些字符

a = [1,2,3]
for x in a:
    if x == 2:
        continue
    print(x)

运行结果是1  3

利用break强行终止的for循环不会继续执行后面的else语句,而continue会继续执行else
用break:

a = [1,2,3]
for x in a:
    if x == 2:
        break
    print(x)
else:
    print('EOF')        #此句不执行

运行结果是1

用continue:

a = [1,2,3]
for x in a:
    if x == 2:
       continue
    print(x)
else:
    print('EOF')        #此句执行

运行结果是1   3   EOF

循环二维数组的跳出

a = [['apple','orange','banana','grape'],(1,2,3)]
for x in a:
    for y in x:
        if y == 'orange':
            break         #只跳出了列表内部循环,元组的循环继续执行
       print(y)
else:
    print('fruit is gone')     #break跳出的是里面的for循环,和else配对的是外面的for循环

运行结果是apple    1   2   3  fruit is gone

7.3 for 与 range

利用range()打印0-9

for x in range(0,10):
    print(x)

range(首,跨度,步长)打印0,2,4,6,8

for x in range(0,10,2):       #递增输出
    print(x,end = ' | ')

运行结果是:0 | 2 | 4 | 6 | 8 |

递减的等差数列:

for x in range(10,0,-2):             #递减输出
    print(x,end = ' | ')

运行结果是:10 | 8 | 6 | 4 | 2 |

7.4 新篇章导言

打印出序列中相间隔的元素
a = [1,2,3,4,5,6,7,8]

for i in range(0,len(a),2):  #步长是2    
    print(a[i],end = ' | ')

运行结果是:1 | 3 | 5 | 7 | 

原理是利用range函数生成了一个等差数列,把数列作为a列表的下标,依次把a里面的每个间隔元素全部取出来
python中有比for循环更好的方式,那就是序列的切片;

a = [1,2,3,4,5,6,7,8]
b = a[0:len(a):2]
print(b)

运行结果是:[1,3,5,7] 

会写代码,非常容易,因为这只考验程序员的逻辑能力
但写出高性能、封装性(可复用)的代码就比较难,因为这不仅考验人的逻辑能力还考验抽象能力

7.5 Python工程的组织结构:包、模块儿、类

最顶级的组织结构:包(文件夹)
第二个层级:模块(文件)
第三个层级:类
第四个层级:函数、变量(不属于组织结构,是类本身的特性)

一个包下面可以包含很多模块
一个模块下面可以包含很多类
在python里面,模块的物理的表现就是我们实际看到的,就是一个.py文件;
模块下面并不是只能写类,还可以写函数、变量及业务逻辑,但是不推荐,最好是用类把函数、变量及业务逻辑组织起来。
包和模块我们可以简单的理解成文件夹和文件,但不代表包和模块就是文件夹和文件,只不过是在物理的表现上,像是一个文件夹和文件。

7.6 Python包与模块的名字

区分不同包的同名模块:包名.模块名
形成的模块的路径叫做命名空间

一个包下面可以有子包,模块可以和包平级

普通文件夹想要变成包必须要有__init__.py文件
init.py本身也是一个模块,可以不写内容只是标注包
特殊的,init.py模块的名字就是包名

7.7 import导入模块

对于重复的定义可以从其他模块里引用,不需要重复编写。
利用 import 模块名 导入

#test1.c1

a = 1

#test1.c2
import c1           #导入c1
print(c1.a)
运行结果:1

python是解释型语言,要先定义后使用

被导入模块在子包下面,要加上命名空间

#test1.t.c1
a = 1

#test1.c2
import t.c1
print(t.c1.a)

运行结果:1

import导入的总是模块,不能直接导入模块下面的变量,需要用模块名.变量名的方法引用

可以用as简化书写

#test1.c1
a = 1

#test1.c2 
import c1 as m
print(m.a)

运行结果:1

不用as简写的优点是可以一眼看出属于哪个模块

7.8 from import 导入变量

#test1.c2
from c1 import a
print(a)

运行结果:1

from import 与 import 的区别:
导入的类型不同 import 导入的是模块,from import导入的是具体的变量
因为from import导入的是具体的变量,所以就不需要在变量的前面再加命名空间了。

也可以from 包 import 模块
引用时用 模块.变量,与from区别就不大了
如:

#test1.c2
from t import c1
print(c1.a)

运行结果:1

引用大量变量时使用 * :

#test1.c1
a = 1
b = 2
c = 3
d = 4

#test1.c2 
from c1 import *
print(a)
print(b)
print(c)
print(d)

运行结果:1    2    3     4

不推荐过多的使用 * 因为这样会使引用非常的不明确,而且鼠标放到abcd上会提示abcd没有定义。
编译器会提示语法错误,不过会运行成功;
"*"会一次性的把模块下面所有的变量或者函数全部导入进来,但是在实际的编码过程中不太可能在引入的时候需要全部的变量,那么就需要我们控制 * 的行为,让 * 导入的时候只导入我们指定的变量。

#test1.c1
__all__ = ['a','c']    #用内置变量__all__来控制 * 的选择范围
a = 1
b = 2
c = 3
d = 4

#test1.c2 
from c1 import *
print(a)
print(c)
print(d)       #d没有被打印,报错

运行结果:1   3   

7.9 init.py 的用法

隐藏__pycache__文件夹
文件→首选项→设置→搜索files.exclu,添加**/pycache,保存

代码换行:在上一行末尾加上\或者加上括号利用括号的特性换行,推荐使用括号换行

from c6 import a,b,\
c
print(a)                       

from c6 import (a,b,
c)
print(a)

包下面都有__init__.py文件,若没有就不是包,而是一个普通的文件夹
init.py文件的作用:
当包被导入时,init.py会首先自动被执行

#t.__init__.py
a = 'This is a __init__.py file'
print(a)

#c11.py
import t               #导入包t,t里面的__init__.py文件就会被自动执行

运行结果:This is a __init__.py file

假如不导入包,而是导入包里面模块的某个变量,init文件依旧会自动运行

#t.c7.py
__all__ = ['a','c']
a = 2
c = 3
d = 4

#c12.py
from t.c7 import a    

运行结果:This is a __init__.py file

结论:无论是导入包还是包下面的某个模块的变量,都会自动运行__init__.py文件

init.py的应用场景:

#t.__init__.py
__all__ = ['c7']       #初始化*,标明哪些模块被导出

#t.c7.py
__all__ = ['a','c']
a = 2
c = 3
d = 4

#t.c8.py
e = 2
f = 3
g = 4

#c11.py
from t import *         #* 表示导入 t 包里的所有模块
print(c7.a)             #c7在init中被定义,可引用
print(c8.e)             #c8在init中未定义,不可引用

运行结果是2

init重要的作用:批量导入库

#t.__init__.py
import sys        #批量导入库,内置标准库
import datetime
import io

#c13.py
import t
print(t.sys.path)             #在这里插入代码片

7.10 包与模块的几个常见错误

包和模块是不会被重复的导入的,只会执行一次
避免循环导入,不要形成闭环(多个模块之间形成闭环)
导入模块的时候会执行模块里所有的代码

7.11 模块内置变量

#t.c14.py
a = 2
c = 3
infos = dir()        #打印当前模块内所有变量
print(infos)
运行结果:['__annotations__', '__builtins__', '__cached__', '__doc__', '__file__', '__loader__', '__name__', '__package__', '__spec__','a',']

  #t.c9.py
    '''
        This is a c9 doc                          #注释
    '''
    print('name:' + __name__)                    # + 把他们连接在一起,
    print('package:' + __package__)
    print('doc:' + __doc__)                      #doc指的是模块的注释
    print('file:' + __file__)

#c15.py
import t.c9
c15.py运行结果:
name:t.c9                             #c9模块的完整命名
package:t                             #c9模块的包
doc:
This is a c9 doc                      #c9模块的注释
file:D:\install\python\t\c9.py        #c9模块在系统中的物理路径

7.12 入口文件和普通模块内置变量的区别

#c15.py
import t.c9
print('package:' + __package__ )
print('name:' + __name__)
print('doc:' + __doc__)
print('file:' + __file__)

运行结果:
name:t.c9
package:t
doc:
    This is a c9 doc
file:D:\install\python\t\c9.py
c9运行成功,c15运行失败,因为c15不属于任何包

修改后:

#c15.py
import t.c9
print('package:' + (__package__ or '当前模块不属于任何包'))        #因为+的优先级高于or,所以加()
print('name:' + __name__)
print('doc:' + __doc__)
print('file:' + __file__)
运行结果:
name:t.c9
package:t
doc:
    This is a c9 doc
file:D:\install\python\t\c9.py
package:当前模块不属于任何包
name:__main__                          #入口文件中__name__会被强制修改为__main__ 

再次修改:

#c15.py
import t.c9
print('~~~~~~~~~~~~c15~~~~~~~~~~~~~)
print('package:' + (__package__ or '当前模块不属于任何包'))  
print('name:' + __name__)
print('doc:' +( __doc__or ‘当前模块没有文档注释'))
print('file:' + __file__)
运行结果:
name:t.c9
package:t
doc:
    This is a c9 doc
file:D:\install\python\t\c9.py
~~~~~~~~~~~~c15~~~~~~~~~~~~~
package:当前模块不属于任何包
name:__main__       
doc:当前模块没有文档注释
file:c15.py                   #入口文件中路径不同 ,和执行python命令所在目录是有关系的

差异:
导入的模块:有package name是模块名字 file是完整的系统路径
入口文件:没有package name是__main__ file是py文件的名字

7.13 __name__的经典应用

dir()若不传参数可以打印当前模块所有可见的变量,传参显示特定的变量

#c16.py

import sys
infos = dir(sys)
print(infos)

运行结果是打印出很多变量

#c17.py

if __name__ == '__main__':               #内置变量
    print('This is app')

    print('This is a module')
运行结果是:This is app
                     This is a module 

#c18.py

import c17
运行结果是:This is a module 

make a script both importable and executable:
翻译:让你的python脚本既可以作为一个普通的模块提供给其他的应用程序调用,
也可以让他自己成为一个可执行的文件

cmd中 python -m 命名空间.模块 可以把入口文件当作模块来运行
作为普通模块必须要有包,可执行文件没有包

7.14 相对导入和绝对导入 一

有一个主入口文件。
顶级包和可执行文件在同一级。
绝对导入:从顶级包开始往下导入
相对导入:. 同级目录 …上级目录 …上上级目录
在相对路径中应使用from import 不能使用 import
不能在入口文件中使用相对路径

7.15 相对导入和绝对导入 二

相对导入不能超过顶级包。
入口不能使用相对路径导入,相对路径根据__name__定位,由于入口文件被强制改成了__main__所以不能使用。
在入口文件中可以使用绝对路径导入。
若想在入口文件使用相对导入,可以把入口文件当做一个模块来使用,回到PYTHON的上一级,即

用python -m PYTHON.main来执行。

你可能感兴趣的:(测试,编程,Python)