Python—面向对象01
1、如何使用类
# 先定义类
class LuffyStudent():
school = "luffycity" # 数据属性
def learn(self): # 函数属性
print("is learning...")
def sleep(self): # 函数属性
print("is sleeping...")
# 查看类的名称空间
print(LuffyStudent.__dict__)
# {'__module__': '__main__', 'school': 'luffycity', 'learn': <function LuffyStudent.learn at 0x00000000025B8B70>, 'sleep': <function LuffyStudent.sleep at 0x00000000025B8BF8>, '__dict__': <attribute '__dict__' of 'LuffyStudent' objects>, '__weakref__': <attribute '__weakref__' of 'LuffyStudent' objects>, '__doc__': None}
print(LuffyStudent.__dict__["school"]) # luffycity
print(LuffyStudent.__dict__["learn"]) # <function LuffyStudent.learn at 0x00000000025B8B70>
# 查
print(LuffyStudent.school) # luffycity
print(LuffyStudent.learn) # <function LuffyStudent.learn at 0x0000000002158AE8>
# 增
LuffyStudent.country = "China"
print(LuffyStudent.country) # China
# 删
del LuffyStudent.country
print(LuffyStudent.__dict__)
# {'__module__': '__main__', 'school': 'luffycity', 'learn': <function LuffyStudent.learn at 0x0000000002158AE8>, 'sleep': <function LuffyStudent.sleep at 0x0000000002158B70>, '__dict__': <attribute '__dict__' of 'LuffyStudent' objects>, '__weakref__': <attribute '__weakref__' of 'LuffyStudent' objects>, '__doc__': None}
# 改
LuffyStudent.school = "foguang University"
print(LuffyStudent.__dict__)
# {'__module__': '__main__', 'school': 'foguang University', 'learn': <function LuffyStudent.learn at 0x0000000002158AE8>, 'sleep': <function LuffyStudent.sleep at 0x0000000002158B70>, '__dict__': <attribute '__dict__' of 'LuffyStudent' objects>, '__weakref__': <attribute '__weakref__' of 'LuffyStudent' objects>, '__doc__': None}
2. ____init____方法
# __init__ 方法用来为对象定制对象自己独有的特征
class LuffyStudent:
school = "luffycity"
def __init__(self, name, sex, age):
self.Name = name
self.Sex = sex
self.Age = age
def learn(self):
print("is learning...")
def sleep(self):
print("is sleeping...")
stu1 = LuffyStudent("gudon", "男", 22) # LuffyStudent.__init__(stu1, "gudon", "男", 22)
# 加上__init__方法后,实例化的步骤
# 1、先产生一个空对象stu1
# 2、LuffyStudent.__init__(stu1, "gudon", "男", 22)
# 查
print(LuffyStudent.__dict__)
# {'__module__': '__main__', 'school': 'luffycity', '__init__': <function LuffyStudent.__init__ at 0x0000000002598AE8>, 'learn': <function LuffyStudent.learn at 0x0000000002598B70>, 'sleep': <function LuffyStudent.sleep at 0x0000000002598BF8>, '__dict__': <attribute '__dict__' of 'LuffyStudent' objects>, '__weakref__': <attribute '__weakref__' of 'LuffyStudent' objects>, '__doc__': None}
print(stu1.Name) # gudon
print(stu1.Sex) # 男
print(stu1.Age) # 22
# 改
stu1.Name = "Astro"
print(stu1.Name) # Astro
# 删除
del stu1.Name
print(stu1.__dict__) # {'Sex': '男', 'Age': 22}
# 增
stu1.color = "cyan"
print(stu1.__dict__) # {'Sex': '男', 'Age': 22, 'color': 'cyan'}
3.属性查找
# *_*coding:utf-8 *_*
class LuffyStudent:
school = "luffycity"
def __init__(self, name, sex, age):
self.Name = name
self.Sex = sex
self.Age = age
def learn(self):
print("%s is learning..." % (self.Name))
def sleep(self, x):
print("%s is sleeping... %s" % (self.Name, x))
stu1=LuffyStudent('王二丫', '女', 18)
stu2=LuffyStudent('李三炮', '男', 38)
stu3=LuffyStudent('张铁蛋', '男', 48)
print(stu1.__dict__)
print(stu2.__dict__)
print(stu3.__dict__)
# {'Name': '王二丫', 'Sex': '女', 'Age': 18}
# {'Name': '李三炮', 'Sex': '男', 'Age': 38}
# {'Name': '张铁蛋', 'Sex': '男', 'Age': 48}
# 对象:特征与技能的结合体
# 类:类是一系列对象相似的特征与相似的功能的结合体
# 类中的数据属性,是所有对象共有的,我们看到id() 后得到的地址是一样的
print(LuffyStudent.school,id(LuffyStudent.school)) # luffycity 41639344
print(stu1.school,id(stu1.school)) # luffycity 41639344
print(stu2.school,id(stu2.school)) # luffycity 41639344
print(stu3.school,id(stu3.school)) # luffycity 41639344
# 类中的函数属性,是绑定给对象使用的,绑定到不同的对象是不同的绑定方法,对象调用绑定方式时,会把对象本身当做第一个参数传入,传给self
print(LuffyStudent.learn)
print(stu1.learn)
print(stu2.learn)
print(stu3.learn)
# <function LuffyStudent.learn at 0x00000000027C8B70> 这其实是一个函数
# <bound method LuffyStudent.learn of <__main__.LuffyStudent object at 0x00000000027C7B00>>
# <bound method LuffyStudent.learn of <__main__.LuffyStudent object at 0x00000000027C7B70>>
# <bound method LuffyStudent.learn of <__main__.LuffyStudent object at 0x00000000027C7BA8>>
# 以上打印结果说明,函数是绑定给对象使用的,不同的对象绑定各自的方法(虽然名称一样)
# <function LuffyStudent.learn at 0x00000000027C8B70> 这其实是一个函数,既然是一个函数,那么就可以调用,如下:
LuffyStudent.learn(stu1) # 王二丫 is learning...
# 我们再看 绑定方法 ,以print(stu1.learn) 为例
# 打印结果为:<bound method LuffyStudent.learn of <__main__.LuffyStudent object at 0x00000000027C7B00>>
# bound method 绑定方法 ,我们用 stu1.learn() 调用试试
stu1.learn() #王二丫 is learning... 此时,会把stu1 作为参数传入 self
# -----------------------------------------------
# 还可以传其他参数进去
stu1.sleep(1) # 王二丫 is sleeping... 1
stu2.sleep(2) # 李三炮 is sleeping... 2
stu3.sleep(3) # 张铁蛋 is sleeping... 3
# ----------------------------------------------
stu1.x = "11111"
LuffyStudent.x = "22222"
print(stu1.__dict__) # {'Name': '王二丫', 'Sex': '女', 'Age': 18, 'x': '11111'}
print(stu1.x) # 11111
# ------------------------------
LuffyStudent.y = "yyyyyyy"
print(stu2.__dict__) # {'Name': '李三炮', 'Sex': '男', 'Age': 38} 注意,查看stu2 的命名空间,是没有 y 的 ^_^
print(stu2.y) # yyyyyyy
# 综上,对象会先去自己的命名空间去找这个属性,如果没有,再去类里面找
4. python 里一切皆对象
# Python 一切皆对象,python3里,统一了 类与类型的概念
class Student:
def learn(self):
print("is learning...")
def sleep(self):
print("is sleeping...")
print(list) # <class 'list'>
print(Student) # <class '__main__.Student'>
l = [1, 2, 3] # 相当于 l = list([1, 2, 3] )
# list 和 Student 都是 class,即都是类
5. 两个小练习
'''
练习1:编写一个学生类,产生一堆学生对象,
要求:
有一个计数器(属性),统计总共实例了多少个对象
'''
class Student:
count = 0
def __init__(self, name, sex, age):
self.Name = name
self.Sex = sex
self.Age = age
Student.count += 1 # 此处用Student.count 做累加
def learn(self):
print("%s is learning" % self.Name)
stu1 = Student("nurato" , "男", 18)
stu2 = Student("Astro", "男", 15)
stu3 = Student("spider man", "男" ,26)
print(Student.count)
print(stu1.count)
print(stu2.count)
print(stu3.count)
# 3
# 3
# 3
# 3
'''
练习2:模仿LoL定义两个英雄类,
要求:
英雄需要有昵称、攻击力、生命值等属性;
实例化出两个英雄对象;
英雄之间可以互殴,被殴打的一方掉血,血量小于0则判定为死亡。
'''
class Garen:
def __init__(self, nickname, life_value, attack_value):
self.Nickname = nickname
self.Life_value = life_value
self.Attack_value = attack_value
def attack(self, enemy):
enemy.Life_value -= self.Attack_value
class Riven:
def __init__(self, nickname, life_value, attack_value):
self.Nickname = nickname
self.Life_value = life_value
self.Attack_value = attack_value
def attack(self, enemy):
enemy.Life_value -= self.Attack_value
g1 = Garen("盖文", 100, 30)
r1 = Riven("瑞文", 80, 50)
g1.attack(r1)
print("瑞文的生命值为:", r1.Life_value)
# 瑞文的生命值为: 50
6. 继承 和 重用
继承指的是类与类之间的关系,是一种什么“是”什么的关系,继承的功能之一就是用来解决代码重用问题
继承是一种创建新类的方式,在python中,新建的类可以继承一个或多个父类,父类又可以成为基类或超类,新建的类称为派生类或子类
class ParentClass1:
pass
class ParentClasse2:
pass
class SubClass1(ParentClass1):
pass
class SubClass2(ParentClass1,ParentClasse2):
pass
# 打印 基类 __bases__ 可以查看所有继承的父类
print(SubClass1.__bases__)
print(SubClass2.__bases__)
# (<class '__main__.ParentClass1'>,)
# (<class '__main__.ParentClass1'>, <class '__main__.ParentClasse2'>)
在开发程序的过程中,如果我们定义了一个类A,然后又想新建立另外一个类B,但是类B的大部分内容与类A的相同时
我们不可能从头开始写一个类B,这就用到了类的继承的概念。
通过继承的方式新建类B,让B继承A,B会‘遗传’A的所有属性(数据属性和函数属性),实现代码重用
关于属性查找:
class Hero:
x=3
def __init__(self,nickname,life_value,aggresivity):
self.nickname=nickname
self.life_value=life_value
self.aggresivity=aggresivity
def attack(self,enemy):
enemy.life_value-=self.aggresivity
class Garen(Hero):
pass
g1 = Garen("Nurato", 100, 30)
print(g1.nickname, g1.life_value, g1.aggresivity) # Nurato 100 30
print(g1.x) # 3 子类没有,就去父类找,父类没有,再往上一层父类找
# 属性查找举例
class Foo:
def f1(self):
print("from Foo.f1")
def f2(self):
print("from Foo.f2")
self.f1() # b.f1() 先从本身类里找,没有才从父类找,此处需注意
class Bar(Foo):
def f1(self):
print("from Bar.f1")
b = Bar()
b.f2() # 先调用父类的 f2(),再调用 b.f1()
# from Foo.f2
# from Bar.f1
7.派生
子类可以添加自己的新属性或者重新定义一些属性(不会影响到父类)。
如果自己定义的属性与父类重名,在调用的时候以自己定义的为准
class Hero:
def __init__(self,nickname,life_value,aggresivity):
self.nickname=nickname
self.life_value=life_value
self.aggresivity=aggresivity
def attack(self,enemy):
enemy.life_value-=self.aggresivity
class Garen(Hero):
camp='Demacia'
def attack(self,enemy):
print('from Garen Class')
class Riven(Hero):
camp='Noxus'
g = Garen("盖文", 100, 30)
r = Riven("瑞文", 80, 50)
print(g.camp) # Demacia
g.attack(r) # from Garen Class Garen重写了attack() ,调用自己的,不用再去父类找了
print(r.life_value) # 80 因为调用 g.attack(r),只是打印了 “from Garen Class”,并不是走的父类的方法,所以还是 80
另外:
在子类中,新建的重名的函数属性,在编辑函数内功能的时候,有可能需要重用父类中重名的那个函数功能,应该是用调用普通函数的方式,即:类名.func(),此时就与调用普通函数无异了
8.python 继承的实现原理
python到底是如何实现继承的,对于你定义的每一个类,python会计算出一个方法解析顺序(MRO)列表,这个MRO列表就是一个简单的所有基类的线性顺序列表,例如
>>> F.mro() #等同于F.__mro__
[<class '__main__.F'>, <class '__main__.D'>, <class '__main__.B'>,
<class '__main__.E'>, <class '__main__.C'>, <class '__main__.A'>, <class 'object'>]
为了实现继承,python会在MRO列表上从左到右开始查找基类,直到找到第一个匹配这个属性的类为止。而这个MRO列表的构造是通过一个C3线性化算法来实现的。 它实际上就是合并所有父类的MRO列表并遵循如下三条准则:
- 子类会先于父类被检查
- 多个父类会根据它们在列表中的顺序被检查
- 如果对下一个类存在两个合法的选择,选择第一个父类
在Java和C#中子类只能继承一个父类,而Python中子类可以同时继承多个父类,如果继承了多个父类,那么属性的查找方式有两种,分别是:深度优先和广度优先
# 1、新式类
# 2、经典类
# 在python2中,经典类:没有继承object的类,以及他的子类
class Foo:
pass
class Bar:
pass
# 在python2中,新式类:继承 object 的类,以及他的子类都称为新式类
class Foo2(object):
pass
class Bar2(Foo2):
pass
# 在python3 中,全都是新式类,python3中,默认继承 object 类
class Foo3:
pass
print(Foo3.__bases__) # (<class 'object'>,)
以下两张图,展示了经典类和新式类,多继承情况下的属性查找方式:
来段代码示例:
class A(object):
def test(self):
print('from A')
class B(A):
def test(self):
print('from B')
class C(A):
def test(self):
print('from C')
class D(B):
def test(self):
print('from D')
class E(C):
def test(self):
print('from E')
class F(D,E):
# def test(self):
# print('from F')
pass
f1=F()
f1.test()
print(F.__mro__) #只有新式才有这个属性可以查看线性列表,经典类没有这个属性
#新式类继承顺序:F->D->B->E->C->A
#经典类继承顺序:F->D->B->A->E->C
#python3中统一都是新式类 广度优先
#pyhon2中才分新式类与经典类 深度优先
针对以上代码,有如下图可参考:
9.在子类中重用父类的属性
在子类派生出的新的方法中重用父类的方法,有两种实现方式
方式一:指名道姓(不依赖继承)
# 在子类派生出的新的方法中重用父类的方法,有两种实现方式
# 方式一:指名道姓(不依赖继承)
class Hero:
def __init__(self,nickname,life_value,aggresivity):
self.nickname=nickname
self.life_value=life_value
self.aggresivity=aggresivity
def attack(self,enemy):
enemy.life_value-=self.aggresivity
class Garen(Hero):
camp='Demacia'
def attack(self,enemy):
Hero.attack(self,enemy) # 指名道姓 并未依赖继承
print('from Garen Class')
class Riven(Hero):
camp='Noxus'
g=Garen('草丛伦',100,30)
r=Riven('锐雯雯',80,50)
print(r.life_value) # 80
g.attack(r) # from Garen Class
print(r.life_value) # 50
class Hero:
def __init__(self,nickname,life_value,aggresivity):
self.nickname=nickname
self.life_value=life_value
self.aggresivity=aggresivity
def attack(self,enemy):
enemy.life_value-=self.aggresivity
class Garen(Hero):
camp='Demacia'
def __init__(self,nickname,life_value,aggresivity,weapon):
# self.nickname=nickname
# self.life_value=life_value
# self.aggresivity=aggresivity
Hero.__init__(self, nickname, life_value, aggresivity) # 相当于上面三条语句
self.weapon=weapon
def attack(self,enemy):
Hero.attack(self,enemy) #指名道姓
print('from Garen Class')
g=Garen('草丛伦',100,30,'金箍棒')
print(g.__dict__)
# {'nickname': '草丛伦', 'life_value': 100, 'aggresivity': 30, 'weapon': '金箍棒'}
方式二:super() (依赖继承)
# 方式二:super() (依赖继承)
class Hero:
def __init__(self,nickname,life_value,aggresivity):
self.nickname=nickname
self.life_value=life_value
self.aggresivity=aggresivity
def attack(self,enemy):
enemy.life_value-=self.aggresivity
class Garen(Hero):
camp='Demacia'
def attack(self,enemy):
super(Garen,self).attack(enemy) # 依赖继承
# super(Garen,self) 表示得到父类,为固定写法,这是python2的写法,在python3中,也可以直接写为 super().attack(enemy)
print('from Garen Class')
class Riven(Hero):
camp='Noxus'
g=Garen('草丛伦',100,30)
r=Riven('锐雯雯',80,50)
print(r.life_value) # 80
g.attack(r) # from Garen Class
print(r.life_value) # 50
刚才上面代码中,指名道姓的方式,改为 继承的方式:
class Garen(Hero):
camp='Demacia'
def __init__(self,nickname,life_value,aggresivity,weapon):
# self.nickname=nickname
# self.life_value=life_value
# self.aggresivity=aggresivity
# super(Garen,self).__init__(nickname,life_value,aggresivity) #python2的写法,python3可以直接写为 super().__init__(nickname,life_value,aggresivity) ,如下
super().__init__(nickname,life_value,aggresivity)
self.weapon=weapon
def attack(self,enemy):
Hero.attack(self,enemy) #指名道姓
print('from Garen Class')
class A:
def f1(self):
print("from A")
super().f1()
class B:
def f1(self):
print("from B")
class C(A, B):
pass
print(C.mro())
# [<class '__main__.C'>, <class '__main__.A'>, <class '__main__.B'>, <class 'object'>]
c = C()
c.f1()
# 打印结果:
# from A
# from B
# 分析一下:c对象里没有f1(),去 class C 这个类里面找,也没有,去A里面找,找到后,执行 print("from A"),然后执行 super().f1(),此时,mro 中,已经找到 <class '__main__.A'> ,接下来会去找 <class '__main__.B'> ,所以,会执行 class B 里的 f1()
虽然A没有继承B,B 也确实不是 A 的父类,因为现在是基于 c 的 MRO 来找的,程序走到哪一步,调用super(),就会基于目前的查找位置往后继续查找
10. 组合
软件重用的重要方式除了继承之外还有另外一种方式,即:组合
组合指的是,在一个类中以另外一个类的对象作为数据属性,称为类的组合
class People:
school = "木叶学院"
def __init__(self, name, age, sex):
self.name = name
self.age = age
self.sex = sex
class Teacher(People):
def __init__(self, name, age, sex, level, salary):
super().__init__(name, age, sex)
self.level = level
self.salary = salary
def teach(self):
print("%s is teaching..." % self.name)
class Student(People):
def __init__(self,name, age, sex, class_time):
super().__init__(name,age,sex)
self.class_time = class_time
def learn(self):
print("%s is learning..." % self.name)
class Course:
def __init__(self, course_name, course_price):
self.course_name = course_name
self.course_price = course_price
def tell_info(self):
print("课程名称:<%s> 课程价格:<%s>" % (self.course_name, self.course_price))
teacher1 = Teacher("卡卡西", 30, "male", "上忍", 10000)
teacher2 = Teacher("大和", 28, "male", "影", 9000)
fly_skill = Course("飞行术", 500) # 创建课程对象
teacher1.course = fly_skill # 把课程对象作为一个属性给对象teacher1,然后就可以通过 teacher1.course.xx 来访问Course 的属性了
print(teacher1.course.course_name) # 飞行术
teacher1.course.tell_info() # 课程名称:<飞行术> 课程价格:<500>
stu1 = Student("鸣人", 15, "male", "白天")
sword_in_hand = Course("手里剑", 300)
stu1.course = sword_in_hand
print(stu1.course.course_name) # 手里剑
stu1.course.tell_info() # 课程名称:<手里剑> 课程价格:<300>
组合和继承,都是有效利用已有类的资源的重要方式。但是二者的概念和使用场景皆不同。
1.继承的方式
通过继承建立了派生类与基类之间的关系,它是一种'是'的关系,比如白马是马,人是动物。
当类之间有很多相同的功能,提取这些共同的功能做成基类,用继承比较好,比如老师是人,学生是人
2.组合的方式
用组合的方式建立了类与组合的类之间的关系,它是一种‘有’的关系,比如老师有生日,老师有课程
所以,当类之间有显著不同,并且较小的类是较大的类所需要的组件时,用组合比较好
11. 抽象类与归一化
这篇写的太长了,另起一篇继续写.......
Python—面向对象01的更多相关文章
- Python面向对象01 /面向对象初识、面向对象结构、类、self、实例化对象
Python面向对象01 /面向对象初识.面向对象结构.类.self.实例化对象 目录 Python面向对象01 /面向对象初识.面向对象结构.类.self.实例化对象 1. 面向对象初识 2. 面向 ...
- python 面向对象专题(一):面向对象初识、面向对象结构、类、self、实例化对象
https://www.cnblogs.com/liubing8/p/11301344.html 目录 Python面向对象01 /面向对象初识.面向对象结构.类.self.实例化对象 1. 面向对象 ...
- python进阶01 面向对象、类、实例、属性封装、实例方法
python进阶01 面向对象.类.实例.属性封装.实例方法 一.面向对象 1.什么是对象 #一切皆对象,可以简单地将“对象”理解为“某个东西” #“对象”之所以称之为对象,是因为它具有属于它自己的“ ...
- Python学习--01入门
Python学习--01入门 Python是一种解释型.面向对象.动态数据类型的高级程序设计语言.和PHP一样,它是后端开发语言. 如果有C语言.PHP语言.JAVA语言等其中一种语言的基础,学习Py ...
- Python开发【第七篇】:面向对象 和 python面向对象进阶篇(下)
Python开发[第七篇]:面向对象 详见:<Python之路[第五篇]:面向对象及相关> python 面向对象(进阶篇) 上一篇<Python 面向对象(初级篇)> ...
- Python面向对象之反射,双下方法
一. 反射 反射的概念是由Smith在1982年首次提出的,主要是指程序可以访问.检测和修改它本身状态或行为的一种能力(自省).这一概念的提出很快引发了计算机科学领域关于应用反射性的研究.它首先被程序 ...
- python 面向对象初级篇
Python 面向对象(初级篇) 概述 面向过程:根据业务逻辑从上到下写垒代码 函数式:将某功能代码封装到函数中,日后便无需重复编写,仅调用函数即可 面向对象:对函数进行分类和封装,让开发" ...
- Python 面向对象 基础
编程范式概述:面向过程 和 面向对象 以及函数式编程 面向过程:(Procedure Oriented)是一种以事件为中心的编程思想. 就是分析出解决问题所需要的步骤,然后用函数把这些步骤一步一步实现 ...
- python面向对象进阶(八)
上一篇<Python 面向对象初级(七)>文章介绍了面向对象基本知识: 面向对象是一种编程方式,此编程方式的实现是基于对 类 和 对象 的使用 类 是一个模板,模板中包装了多个“函数”供使 ...
随机推荐
- xss攻击汇总--转
(1)普通的XSS JavaScript注入<SCRIPT SRC=http://3w.org/XSS/xss.js></SCRIPT>(2)IMG标签XSS使用JavaScr ...
- CityBuilder导入OSGB,生成3DML填坑
工具:CityBuilder 问题1:导入osgb时,中文路径报错 当导入osgb layer时,Input folder和output foder 中最好不要出现中文 否则可能报出如下错误: 日志: ...
- mysql用户操作
一, 创建用户: 命令:CREATE USER 'username'@'host' IDENTIFIED BY 'password'; 说明:username - 你将创建的用户名, host - 指 ...
- C++程序设计基础(5)sizeof的使用
1.知识点 (1)sizeof是一个单目运算发,而不是一个函数,其用于获取操作数所占内存空间的字节数. (2)sizeof的操作数可以使类型名,也可以是表达式,如果是类型名则直接获得该类型所占字节数, ...
- 【学习笔记】关于DOM4J:使用DOM4J解析XML文档
一.概述 DOM4J是一个易用的.开源的库,用于XML.XPath和XSLT中.采用了Java集合框架并完全支持DOM.SAX.和JAXP. DOM4J最大的特色是使用大量的接口,主要接口都在org. ...
- 图片的URL上传至阿里云OSS操作(微信小程序二维码返回的二进制上传到OSS)
当我们从网络中获取一个URL的图片我们要存储到本地或者是私有的云时,我们可以这样操作 把url中的图片文件下载到本地(或者上传到私有云中) public String uploadUrlToOss ...
- js之可迭代对象
遍历Array可以采用下标循环,遍历Map和Set就无法使用下标.为了统一集合类型,ES6标准引入了新的iterable类型,Array.Map和Set都属于iterable类型. 具有iterabl ...
- GitHub网页端和客户端操作
参见GitHub上的repository中的moreLove.tata.tata2 moreLove 在网页版GitHub上创建的空项目然后填充的tata 在windows客户端创建的空项目然后填充的 ...
- Shader之性能优化
1.像素>>顶点数>>物体个数:shader中的计算应首先考虑放在script,其次vert,最后frag中 2.尽量用精度小的类型替换精度大的类型(特别是在frag中,要尽可 ...
- Android 自定义View实现SegmentControlView(自定义多样式tablayout)
偷懒一下,不做过多阐述 参考资源: Android 自定义View实现SegmentControlView : https://blog.csdn.net/a512337862/article/det ...