python学习(二)
2.面向对象(中)
2.1析构方法
class Animal:
def __init__(self,name):
self.name=name
print('这是构造初始化方法')
pass
def __del__(self):
# 主要的应用就是来操作 对象的释放 一旦释放完毕 对象便不能在使用
print('当在某个作用域下面 没有被使用【引用】的情况下 解析器会自动的调用此函数 来释放内存空间')
print('这是析构方法')
print('%s 这个对象 被彻底清理了 内存空间也释放了'%self.name)
pass
cat=Animal('小花猫')
# del cat #手动的去清理删除对象 会指定__del__函数
print(cat.name)
input('程序等待中.....')
# print('*'*40)
# dog=Animal('柯基小狗')
2.2单继承
class Animal:
def eat(self):
'''
吃
:return:
'''
print('吃饭了')
pass
def drink(self):
'''
喝
:return:
'''
pass
class Dog(Animal): #继承了Animal 父类 此时dog就是子类
def wwj(self):
'''
子类独有的实现
:return:
'''
print('小狗汪汪叫')
pass
class Cat(Animal):
def mmj(self):
'''
子类独有的实现
:return:
'''
print('小猫喵喵叫')
pass
d1=Dog()
d1.eat() #具备了吃的行为 是继承了父类的行为
d1.wwj()
print('**************cat 的行为**********************')
c1=Cat()
c1.eat()
c1.mmj()
2.3多继承
注:# 问题是 当多个父类当中存在相同方法的时候 应该去调用哪一个呢?
答:广度优先/(print(A.mro) #可以显示类的依次继承关系)
class shenxian:
def fly(self):
print("神仙都会飞")
pass
class Monkey:
def chitao(self):
print('猴子喜欢吃桃')
pass
class Sunwukong(shenxian,Monkey): #即使神仙 同时也是猴子
pass
# swk=Sunwukong()
# swk.chitao()
# swk.fly()
####################################################
# 问题是 当多个父类当中存在相同方法的时候 应该去调用哪一个呢
class D(object):
def eat(self):
print('D.eat')
pass
pass
class C(D):
def eat(self):
print('C.eat')
pass
pass
class B(D):
pass
class A(B,C):
pass
a=A()
a.eat()
print(A.__mro__) #可以显示类的依次继承关系
#在执行eat的方法时 查找方法的顺序是
# 首先到A里面去查找 如果A中没有 则继续的去B类中去查找 如果B中没有
# 则去C中查找 如果C类中没有 则去D类中去查找,如果还是没有找到 就会报错
# A-B-C-D 也是继承的顺序
2.4间接继承
class GrandFather:
def eat(self):
print('吃的 方法')
pass
pass
class Father(GrandFather):
pass
class Son(Father):
pass
son=Son()
print(Son.__mro__)
son.eat() #此方法 是从GrandFather继承过来的
2.5重写和调用父类方法
- 所谓重写,就是子类中,有一个和父类相同名字的方法,在子类中的方法会覆盖掉父类中同名的方法
- 为什么要重写, 父类的方法已经不满足子类的需要,那么子类就可以重写父类或者完善父类的方法
class Dog:
def __init__(self,name,color):
self.name=name
self.color=color
def bark(self):
print('汪汪叫....')
pass
pass
class kejiquan(Dog):
def __init__(self,name,color):#属于重写父类的方法
# 针对这种需求 我们就需要去调用父类的函数了
# Dog.__init__(self,name,color) #手动调用 调用父类的方法了 执行完毕就可以具备name,color这两个实例属性了\
super().__init__(name,color) #super是自动找到父类 进而调用方法, 假设继承了多个父类,那么会按照顺序逐个去找 然后在调用
# 拓展其他的属性
self.height=90
self.weight=20
pass
def __str__(self):
return '{}的颜色会{} 它的身高是{}cm 体重是:{}'.format(self.name,self.color,self.height,self.weight
)
def bark(self): #属于重写类的方法
super().bark() #调用父类的方法
print('叫的跟神一样')
print(self.name)
pass
kj=kejiquan('柯基犬','红色')
kj.bark()
print(kj)
注: super是自动找到父类 进而调用方法
2.6 类属性和实例属性
注: 类属性 就是类对象所拥有的属性(类对象不同于实例对象,这里和java有点不同,这里所说的类对象应该就是java里面的元对象)
- 小结
- 类属性是可以 被类对象和实例对象共同访问使用的
- 实例属性只能由实例对象所访问
class Student:
name='李明' #属于类属性 就是student类对象所拥有的
def __init__(self,age):
self.age=age #实例属性
pass
@staticmethod
def aaa(x,y):
print(x+y)
pass
Student.name='李易峰' #通过类对象去修改数据 可以修改的 因为name的所拥有的权利属于类对象
lm=Student(18)
print(lm.name) #通过实例对象去访问类属性
# lm.name='刘德华' #通过实例对象 对类属性进行修改 可以吗? 不可以的
print(lm.name)
print(lm.age)
Student.aaa(4,6)
print('---------xh的数据---------------')
xh=Student(28)
print(xh.name)
print(xh.age)
print('---------通过类对象 student 去访问name---------------')
# print(Student.name) 如 类名.属性名 形式去访问
# print(Student.age)
# 小结
# 类属性是可以 被类对象和实例对象共同访问使用的
# 实例属性只能由实例对象所访问
分析:
2.7类方法和静态方法
- 类方法是类所拥有的,可以访问类变量,可以通过类对象和实例对象访问。必须传参数。
- 静态方法和类方法差不多,但是不必须传参数。
- 为什么要使用静态方法呢?1. 由于静态方法主要来存放逻辑性的代码,本身和类以及实例对象没有交互;2.也就是说,在静态方法中,不会涉及到类中方法和属性的操作;3.数据资源能够得到有效的充分利用。
class People:
country='china'
#类方法 用 classmethod 来进行修饰
@classmethod
def get_country(cls):
return cls.country #访问类属性
pass
@classmethod
def change_country(cls,data):
cls.country=data #修改类属性的值 在类方法中
pass
@staticmethod
def getData():
return People.country #通过类对象去引用
pass
@staticmethod
def add(x,y):
return x+y
pass
print(People.add(10,56)) #带有参数的静态方法
# print(People.getData())
# print(People.get_country()) #通过类对象去引用
p=People()
print(p.getData()) #注意 一般情况下 我们不会通过实例对象去访问静态方法
# print('实例对象访问 %s'%p.get_country())
# print('-----------------修改之后---------------------------')
# People.change_country('英国')
# print(People.get_country()) #通过类对象去引用
# 为什么要使用静态方法呢
# 由于静态方法主要来存放逻辑性的代码,本身和类以及实例对象没有交互,
# 也就是说,在静态方法中,不会涉及到类中方法和属性的操作
# 数据资源能够得到有效的充分利用
# demo 返回当前的系统时间
import time # 引入第三方的时间模块
class TimeTest:
def __init__(self,hour,min,second):
self.hour=hour
self.min = min
self.second = second
@staticmethod
def showTime():
return time.strftime("%H:%M:%S",time.localtime())
pass
pass
print(TimeTest.showTime())
t=TimeTest(2,10,15)
print(t.showTime()) #没有必要通过这种方式去访问 静态方法
2.8三种方法对比
- 从方法定义的形式可以看出来
- 1.类方法的第一个参数是类对象 cls 进而去引用类对象的属性和方法 必须用装饰器 @classmethod来修饰
- 2.实例方法的第一个参数必须是self,通过这个self 可以去引用类属性或者实例属性,若存在相同名称实例属性和类属性的话,实例属性的优先级最高
- 3.静态方法不需要定义额外的参数,若是要引用属性的话 则可以通过类对象或者是实例对象去引用即可 必须用装饰器 @staticmethod来修饰
2.9多态
-
多态:顾名思义就是多种状态、形态,就是同一种行为 对于不同的子类【对象】有不同的行为表现
-
要想实现多态 必须的有两个前提需要遵守:
-
1、继承:多态必须发生在父类和子类之间
-
2、重写: 子类重写父类的方法
-
多态有什么用: 增加程序的灵活性;增加程序的拓展性。
# 案例演示
class Animal:
'''
父类【基类】
'''
def say_who(self):
print('我是一个动物....')
pass
pass
class Duck(Animal):
'''
鸭子类 【子类】 派生类
'''
def say_who(self):
'''
在这里重写父类的方法
:return:
'''
print('我是一只漂亮的鸭子')
pass
pass
class Dog(Animal):
'''
小狗类 【子类】 派生类
'''
def say_who(self):
print('我是一只哈巴狗')
pass
pass
class Cat(Animal):
'''
小猫类 【子类】 派生类
'''
def say_who(self):
print('我是一只小花猫 喵喵喵喵')
pass
pass
class Bird(Animal):
'''
新增鸟类 无需修改原来的代码
'''
def say_who(self):
print('我是一只黄鹂鸟')
pass
class People:
def say_who(self):
print('我是人类')
pass
class student(People):
def say_who(self):
print('我是一年级的学习 张明')
pass
def commonInvoke(obj):
'''
统一调用的方法
:param obj: 对象的实例
:return:
'''
obj.say_who()
# duck1=Duck()
# duck1.say_who()
# dog1=Dog()
# dog1.say_who()
# cat1=Cat()
# cat1.say_who()
listObj=[Duck(), Dog(),Cat(),Bird(),student()]
for item in listObj:
'''
循环去调用函数
'''
commonInvoke(item)
2.10 三大特征(面向对象)
-
在python中展现面向对象的三大特征:
-
封装、继承、多态
-
封装:指的是把内容封装到某个地方,便于后面的使用
他需要:
把内容封装到某个地方
从另外一个地方去调用被封装的内容
对于封装来说 其实就是使用初始化构造方法将内容封装到对象中,然后通过对象直接或者self来获取被封装的内容 -
继承: 和现实生活当中的继承是一样的:也就是 子可以继承父的内容【属性和行为】(爸爸有的儿子都有,相反 儿子有的爸爸不一定有)
所以对于面向对象的继承来说 其实就是将多个类共有的方法提取到父类中 子类仅需继承父类而不必一一去实现
这样就可以极大的提高效率 减少代码的重复编写,精简代码的层级结构 便于拓展class 类名(父类): ''' 子类就可以继承父类中公共的属性和方法 ''' pass
-
多态:顾名思义就是多种状态、形态,就是同一种行为 对于不同的子类【对象】有不同的行为表现
要想实现多态 必须的有两个前提需要遵守:
1、继承:多态必须发生在父类和子类之间
2、重写: 子类重写父类的方法
3.面向对象(下)
3.1私有化属性
- 使用私有属性的场景
- 1.把特定的一个属性隐藏起来 不想让类的外部进行直接调用
- 2.我想保护这个属性 不想让属性的值随意的改变
- 3.保护这个属性 不想让派生类【子类】去继承
class Person:
__hobby='跳舞' #私有的类属性
def __init__(self):
self.__name='李四' #加两个下划线 将此属性私有化之后 就不能再外部直接访问了,当然在类的内部是可以防蚊贴
self.age=30
pass
def __str__(self):
'''
私有化的属性在内部可以使用 self.__name
:return:
'''
return '{}的年龄是{} 爱好是{}'.format(self.__name,self.age,Person.__hobby)
def changeValue(self):
Person.__hobby='唱歌'
class Student(Person):
def printInfo(self):
# print(self.__name) #在此访问父类中的私有属性 可以吗? 不可以
print(self.age)
pass
stu=Student()
# print(stu.__name)
stu.printInfo()
stu.changeValue() #修改私有属性的值
print(stu)
# print(stu.__hobby) #实例对象访问类属性
# print(Person.__hobby) #实例对象访问类属性
# xl=Person()
# # print(xl.__name) #是通过类对象 在外部访问的 不能访问私有属性
# print(xl)
# 小结:
# 1 私有化的【实例】属性 不能再外部直接的访问 可以在类的内部随意的使用
# 2.子类不能继承父类的私有化属性【只能继承父类公共的属性和行为】
# 3. 在属性名的前面直接加‘ __’ 就可以变为私有化了
3.2私有化方法
class Animal:
def __eat(self):
print('吃东西')
pass
def run(self):
self.__eat() #在此调用私有化的方法
print('飞快的跑')
pass
class Bird(Animal):
pass
b1=Bird()
# print(b1.eat())
# b1.eat()
b1.run()
3.3Property属性函数(意义不大)
- 目的:不需要通过函数的形式,来访问私有属性
class Person(object):
def __init__(self):
self.__age = 18 # 定义一个私有化属性,属性名字前加连个 __ 下滑线
# def get_age(self): # 访问私有实例属性
# return self.__age
# def set_age(self,age): # 修改私有实例属性
# if age < 0:
# print('年龄不能小于0')
# else:
# self.__age=age
# pass
# pass
# 定义一个类属性 实现通过直接访问属性的形式去访问私有的属性
# age=property(get_age,set_age)
# 实现方式2 通过装饰器的方式去声明
@property #用装饰器修饰 添加属性标志 提供一个getter方法
def age(self):
return self.__age
@age.setter #提供一个setter方法
def age(self,parms):
if parms < 0:
print('年龄不能小于0')
else:
self.__age=parms
pass
pass
pass
p1=Person()
print(p1.age)
p1.age=30
print(p1.age)
# p1.get_age()
# p1.set_age()
3.4__new__方法
class Animal:
def __init__(self):
self.color='红色'
pass
#在python当中, 如果不重写 __new__ 默认结构如下
def __new__(cls, *args, **kwargs):
return super().__new__(cls,*args, **kwargs)
# return object.__new__(cls, *args, **kwargs)
pass
tigger=Animal() #实例化的过程会自动调用__new__去创建实例
# 在新式类中 __new__ 才是真正的实例化的方法,为类提供外壳制造出实例框架,然后调用该框架内的构造方法__init__ 进行丰满操作
# 比喻建房子 __new__ 方法负责开发地皮 打地基 并将原料存放在工地,而__init__ 负责从工地取材料建造出地皮开发图纸规定的大楼,
# 负责细节设计、建造 最终完成
3.5单例模式
- 单例模式 是一种常用的软件设计模式 目的:确保某一个类只有一个实例存在
- 如果希望在真个系统中 某个类只能出现一个实例的时候,那么这个单例对象就满足要求
- 创建一个单例对象 基于__new__去实现的【推荐的一种】
class DataBaseClass(object):
def __new__(cls, *args, **kwargs):
# cls._instance=cls.__new__(cls) 不能使用自身的new方法
# 容易造成一个深度递归,应该调用父类的new方法
if not hasattr(cls,'_instance'): #如果不存在就开始创建
cls._instance=super().__new__(cls,*args, **kwargs)
return cls._instance
pass
class DBoptSingle(DataBaseClass):
pass
db1=DBoptSingle()
print(id(db1))
db2=DBoptSingle()
print(id(db2))
db3=DBoptSingle()
print(id(db3))
------------------------------------异常处理--------------------------------------
1.异常处理(1)
- 语法格式
# import Exception
# except 在捕获错误异常的时候 只要根据具体的错误类型来捕获的
# 用一个块 可以捕获多个不同类型的异常
# Exception 可以捕获所有的异常 当对出现的问题或者错误不确定的情况下 可以使用Exception
# print(dir(Exception))
try:
# print(b) #捕获逻辑的代码
li=[1,2,34]
print(li[10]) #通过下标去访问列表
# a=10/0
pass
except NameError as msg:
# # 捕获到的错误 才会在这里执行
# print(msg)
pass
except IndexError as msg:
# print(msg)
pass
except ZeroDivisionError as msg:
# print(msg)
pass
except Exception as result:
# print(result)
# 在此尽量的去处理捕获到的错误
pass
# print('初次接触异常处理')
# print('HAHAHAHHAHA')
# try-except-else
# try:
# print('我是没有错误产生的')
# pass
# except SyntaxError as msg:
# print(msg)
# except Exception as msg:
# print('error',msg)
# else:
# print('当Try里面的代码 没有出现异常的情况下 我才会执行')
# pass
# try-except-finally
try:
int('34')
open('aaaa.txt')
except Exception as msg:
print(msg)
pass
finally:
print('释放文件的资源、数据库连接是资源等等')
print('不管有没有出错都执行的代码块')
2.异常处理(2)
- 常见的异常类型
异常 导致的错误 AssertionError 当assert语句失败时引发。 AttributeError 当属性分配或引用失败时引发。 EOFError 当input()函数达到文件结束条件时引发。 FloatingPointError 当浮点运算失败时引发。 GeneratorExit 调用生成器的close()方法时引发。 ImportError 找不到导入的模块时引发。 IndexError 当序列的索引超出范围时引发。 KeyError 在字典中找不到键时引发。 KeyboardInterrupt 当用户按下中断键(Ctrl+c或delete)时引发。 MemoryError 当操作耗尽内存时引发。 NameError 在局部或全局范围内找不到变量时引发。 NotImplementedError 由抽象方法提出。 OSError 当系统操作导致系统相关错误时引发。 OverflowError 当算术运算的结果太大而无法表示时引发。 ReferenceError 使用弱引用代理访问垃圾收集的引用时引发。 RuntimeError 当错误不属于任何其他类别时引发。 StopIteration 函数引发,以指示迭代器不再返回任何项。 SyntaxError 遇到语法错误时由解析器引发。 IndentationError 当缩进不正确时引发。 TabError 当缩进由不一致的制表符和空格组成时引发。 SystemError 当解释器检测到内部错误时引发。 SystemExit 由sys.exit()函数引发。 TypeError 将函数或操作应用于类型不正确的对象时引发。 UnboundLocalError 当在函数或方法中引用局部变量,但没有将值绑定到该变量时引发。 UnicodeError 当发生与unicode相关的编码或解码错误时引发。 UnicodeEncodeError 当编码过程中发生与unicode相关的错误时引发。 UnicodeDecodeError 当解码过程中出现与unicode相关的错误时引发。 UnicodeTranslateError 翻译过程中发生与unicode相关的错误时引发。 ValueError 当函数得到类型正确但值不正确的参数时引发。 ZeroDivisionError 当除法或模运算的第二个操作数为零时引发。
- 异常处理机制
def A(s): return 10/int(s) pass def B(s): return A(s)*2 def main(): # B(0) try: B('0') pass except Exception as msg: print(msg) pass pass # main() # 不需要在每个可能出错的地方去捕获,只要在合适的层次去捕获错误就可以了 这样的话 就大大减少我们写try---except的麻烦 # 异常的抛出机制 #如果在运行时发生异常 解释器会查找相应的异常捕获类型 #如果在当前函数里面没有找到的话 它会将异常传递给上层的调用函数,看能否处理 # 如果在最外层 没有找到的话,解释器就会退出 程序down哦
3.异常处理(3)
-
自定义异常:都要直接或者间接继承Error或者Exception类
class ToolongMyException(Exception): def __init__(self,leng): ''' :param leng: 长度 ''' self.len=leng pass def __str__(self): return '您输入姓名数据长度是'+str(self.len)+'超过长度了..' pass def name_Test(): name=input('请输入姓名.....') try: if len(name)>5: raise ToolongMyException(len(name)) else: print(name) pass pass except ToolongMyException as result: print(result) pass finally: print('执行完毕了....') name_Test()
4.动态添加属性和方法(了解)
-
代码
import types #添加方法的库 def dymicMethod(self): print('{}的体重是:{}kg 在 {} 读大学'.format(self.name,self.weight,Student.shcool)) pass @classmethod def classTest(cls): print('这是一个类方法') pass @staticmethod def staticMethodTest(): print('这是一个静态方法') pass class Student: def __init__(self,name,age): self.name=name self.age=age pass def __str__(self): return '{}今天{}岁了'.format(self.name,self.age) pass print('绑定类方法') Student.TestMethod=classTest # Student.TestMethod() Student.staticMethodTest=staticMethodTest Student.staticMethodTest() print('---------------绑定类方法执行结束----------------------') zyh=Student('张艳华',20) zyh.weight=50 #动态添加 zyh.printInfo=types.MethodType(dymicMethod,zyh) #动态的绑定方法 zyh.TestMethod() print('-------------实例对象调用 动态绑定类方法-----------------------') # print(zyh) # print(zyh.weight) print('-------------另外一个实例对象 张明--------------------------') zm=Student('张名',20) zm.weight=80 #动态添加 print(zm) zm.printInfo=types.MethodType(dymicMethod,zm) #动态的绑定方法 # print(zm.weight) print('-------------给类对象添加属性--------------------------') Student.shcool='北京邮电大学' #动态添加类属性 print(zm.shcool) print('-------------执行动态调用实例方法--------------------------') zyh.printInfo() #调用动态绑定的方法 zm.printInfo()
5. __slots __属性
- 作用: 限制要添加的实例属性;节约内存空间
class Student(object): __slots__ = ('name','age') def __str__(self): return '{}....{}'.format(self.name,self.age) pass xw=Student() xw.name='小王' xw.age=20 xw.score=96 # 没有在范围内 所以报错 # print(xw.__dict__) #所有可以用的属性都在这里存储 不足的地方就是占用的内存空间大 # 可以看到 在定义了 slots变量之后 student类的实例已经不能随意创建不在 __slots__定义的属性了 # 同时还可以看到实例当中也不在有__dict__ # print(xw) # 在继承关系当中的使用 __slots__ # 子类未声明 __slots__时,那么是不会继承父类的__slots__,此时子类是可以随意的属性赋值的 # 子类声明 了__slots__时,继承父类的__slots__,也就是子类__slots__的范围是为: # 其自身+父类的__slots__ class subStudent(Student): __slots__ = ('gender','pro') pass ln=subStudent() ln.gender='男' ln.pro='计算机信息管理' print(ln.gender,ln.pro)
6.文件操作
6.1文件读写
-
写
# 文件的操作 # 打开文件 open # 默认的编码是gbk 这个是中文编码,最好的习惯呢就是我们再打开一个文件的时候 # 给它指定一个编码类型 fobj = open('./Test.txt','w',encoding='utf-8') # # 开始操作 读/写操作 fobj.write('在苍茫的大海上') fobj.write('狂风卷积着乌云') fobj.write('在乌云和大海之间\r\n') fobj.write('海燕像黑色的闪电\r\n') fobj.close() # 以二进制的形式去写数据 fobj=open('Test_1.txt','wb') #str-->bytes fobj.write('在乌云和大海之间'.encode('utf-8')) fobj.close() #用于追加数据 fobj=open('Test.txt','a') fobj.write('在乌云和大海之间\r\n') fobj.write('海燕像黑色的闪电\r\n') fobj.close() fobj=open('Test.txt','a') # a 用于追加数据 fobj.write('在苍茫的大海上') fobj.write('狂风卷积着乌云') fobj.write('在乌云和大海之间\n') fobj.write('海燕像黑色的闪电\n') fobj.write('今天我诗兴大发\n') fobj.write('发感觉咋样呢\n') fobj.close()
-
读
# 读数据操作 f=open('Test.txt','rb') data=f.read() #读取所有的数据 print(data) print(data.decode('gbk')) # print(f.read(12)) # # print(f.read()) # print(f.readline()) #读一行数据 # print(f.readlines(1)) # f.close() #文件对象关闭掉 # with上下文管理对象 # 优点 自动释放打开关联的对象 with open('Test.txt','a') as f: # print(f.read()) f.write('我觉得python非常的好学\n') # 小结 # 文件读写的几种操作方式 # read r r+ rb rb+ # r r+ 只读 使用普通读取场景 # rb rb+ 适用于 文件 图片 视频 音频 这样文件读取 # write w w+ wb+ wb a ab # w wb+ w+ 每次都会去创建文件 # 二进制读写的时候 要注意编码问题 默认情况下 我们写入文件的编码是gbk # a ab a+ 在原有的文件的基础之后去【文件指针的末尾】去追加, # 并不会每次的都去创建一个新的文件 def ctime(sec): pass
注:模式
6.2备份文件
# import fileOpt
# 文件的备份
def copyFile():
# 接收用户输入的文件名
old_file=input('请输入要备份的文件名:')
file_list=old_file.split('.')
# 构造新的文件名.加上备份的后缀
new_file=file_list[0]+'_备份.'+file_list[1]
old_f=open(old_file,'r') #打开需要备份的文件
new_f=open(new_file,'w') #以写的模式去打开新文件,不存在则创建
content=old_f.read() #将文件内容读取出来
new_f.write(content) #将读取的内容写入到备份文件
old_f.close()
new_f.close()
pass
copyFile()
def copyBigFile():
# 接收用户输入的文件名
old_file=input('请输入要备份的文件名:')
file_list=old_file.split('.')
# 构造新的文件名.加上备份的后缀
new_file=file_list[0]+'_备份.'+file_list[1]
try:
#监视啊哟处理的逻辑
with open(old_file,'r') as old_f,open(new_file,'w')as new_f:
while True:
content=old_f.read(1024) #一次读取1024字符
new_f.write(content)
if len(content)<1024:
break
except Exception as msg:
print(msg)
pass
6.3文件定位(了解)
# tell 返回指针当前所在的位置
# 对于中文讲 每次读取到的一个汉字 实际上占用了 2个字符
with open('Test.txt','r+') as f:
print(f.read(3))
print(f.tell())
print(f.read(2))
print(f.tell())
pass
truncate 可以对源文件进行截取操作
fobjB=open('Test.txt','r')
print(fobjB.read())
fobjB.close()
print('截取之后的数据........')
fobjA=open('Test.txt','r+')
fobjA.truncate(15) #保留前15个字符
print(fobjA.read())
fobjA.close()
# seek 可以控制光标所在位置
with open('Test_备份.txt','rb') as f:
f.seek(4,0) #光标是从0的位置开始 像前【右】移动 4个字符
data=f.read(2)
print(data.decode('gbk'))
f.seek(-2,1) #相当于光标有设置到了0的位置
print(f.read(4).decode('gbk'))
f.seek(-6,2) #2 表示光标在末尾处 往回移动了6个字符
print(f.read(4).decode('gbk'))
pass
# 对于这种情况 用'r'这种模式打开文件,在文本文件中,没有使用二进制的选项打开文件,
# 只允许从文件的开头计算相对位置,从文件尾部计算或者当前计算的话 就会引发异常
6.4模块导入
-
1:import 模块
注: import 到首次导入模块的时候 会发生如下3步操作:
1 打开模块文件
2 执行模块对应的文件,将执行过程中产生的名字都丢到模块的名称空间
3 在程序中会有一个模块【可以取别名的】的名称指向模块的名称空间去import time as myTime #加了一个别名 print(myTime.ctime()) # print(time.ctime()) #调用模块中的函数了 import fileOpt
-
2:路径搜索过程
-
3:from 模块 import 方法
注:from …import 首次导入发生了3个步骤
1 以模块为准创造一个模块的名称空间
2 执行模块对应的文件了,将执行过程中产生的名字都丢到模块的名称空间
3 在当前执行文件的名称空间中拿到一个名字,该名字直接指向模块中的某一个名字,意味着可以不用加任何的前缀而直接使用from time import ctime,time #部分导入 # from time import * print(ctime()) # 优点:不用加前缀 代码更加简洁了 # 缺点: 容易与当前执行文件中名称空间中的名字冲突
-
4.比较
# 相同点和不同点 # 相同 # 都会执行模块对应的文件 并且呢 都会产生模块课的名称空间 # 两者在调用功能时呢,都需要转到定义时寻找作用域关系 # 不同点 # import 需要加前缀 from...import 不需要加前缀
7.常见模块
-
os模块
-
time、datatime模块
8.模块的制作、发布、安装
8.1模块的制作
- 模块的制作
- 一个随便的py文件就是一个模块
- 有了__name__,这样别的模块调用的时候,不会运行
注:__all__的作用
8.2模块的发布
- 1.写好包放在一个目录下
- 2.创建setup.py文件
- 3.创建模块
- 4.生成压缩包
- 5.tree看结构
8.3模块的安装
- 1.压缩包解压
- 2.安装
- 3.模块引入
------------------------------------内存管理---------------------------------
1.引用计数法
注:类似于java,这个系统自动回收垃圾对象,明显有循环引用的弊端。
- 代码例子
import sys import psutil import os import gc print(gc.get_threshold()) def showMemSize(tag): pid=os.getpid() p=psutil.Process(pid) info=p.memory_full_info() memory=info.uss/1024/1024 print('{} memory used:{} MB'.format(tag,memory)) pass # 验证循环引用的情况 def func(): showMemSize('初始化') a=[i for i in range(10000000)] b=[i for i in range(10000000)] a.append(b) b.append(a) showMemSize('创建列表对象 a b 之后') # print(sys.getrefcount(a)) # print(sys.getrefcount(b)) # del a # del b pass func() gc.collect() #手动是释放的 showMemSize('完成时候的') # sys.getrefcount() # a=[] # print(sys.getrefcount(a)) #两次 # b=a # print(sys.getrefcount(a)) #三次 # c=b # d=b # e=c # f=e # g=d # print(sys.getrefcount(a)) #八次
2.标记清除、分代收集算法
- 标记清楚:先标记对象,然后清除垃圾(java的可达性分析)
3.小整数和大整数对象池(内存优化)
- 类似于java的字符串常量池
- 大整数池和小整数池的区别是:
1 从结果来看是一样的
2 大整数池是没有提前的创建好对象,是个空池子,需要我们自己去创建,创建好之后,会把整数对象保存到池子里面,后面都不需要再创建了 直接拿来使用;小整数池是提前将【-5,256】的数据都提前创建好; 字符串的驻留共享机制intern机制a=140 b=140 print(id(a)) print(id(a)) # del a # del b # c=140 # print(id(c)) # biga=100000 # bigb=100000 # print(id(biga)) # print(id(bigb)) # del biga # del bigb # bigc=100000 # print(id(bigc)) # sa='ab_c' # sb='ab_c' # sc='ab_c' # sd='ab_c'
4.规范
- 缩进:4个空格的缩进
- 添加注释
- 命名规范:类–大驼峰法,方法–小驼峰法
5.命令行参数_sys
-
sys模块
import sys import argparse print('参数个数为:', len(sys.argv), '个参数.') print('参数列表:', str(sys.argv[1:]))
-
执行结果
-
弊端:
6.命令行参数_argparse
-
argparse为首选
# 创建一个解析器对象 parse=argparse.ArgumentParser(prog='my - 我自己的程序', usage='%(prog)s [options] usage', description = 'my-编写自定义命令行的文件',epilog = 'my - epilog') # 添加位置参数【必选参数】 parse.add_argument('name',type=str, help='你自己的名字') parse.add_argument('age',type=str,help='你的年龄') # 添加可选参数 # parse.add_argument('-s','--sex', action='append',type=str,help='你的性别') # 限定一个范围 parse.add_argument('-s','--sex',default='男', choices=['男','femal','女','male'],type=str,help='你的性别') # print(parse.print_help()) result=parse.parse_args() #开始解析参数 print(result.name,result.age,result.sex)
-
构造对象的参数说明
-
参数的属性
参考链接:
https://www.bilibili.com/video/BV1vA411b7Rn?p=24
- 点赞
- 收藏
- 关注作者
评论(0)