





-
奥密克戎病毒的危害性,这需要若干年的科学数据累积才能判断到底有没有后遗症。怎么可能通过几个痊愈患者的采访,就断定没有问题,就定义为无害的大号流感?美国专家的研究报告都说病毒会入侵脑神经,从而造成可能的后遗症。 -
谁能保证新冠病毒的演化只会越来越弱?后续如果出现更厉害的病毒混在已有的病毒里一起传染,如何应对?就像两年前专家们都说疫苗出来了,疫情就结束了,但结果呢?










转自:https://mp.weixin.qq.com/s/u2GN_MVfsBl2skCYaD9O0A
分享个人经验,保留阅读记录,做时间的朋友
转自:https://mp.weixin.qq.com/s/u2GN_MVfsBl2skCYaD9O0A
转自:https://mp.weixin.qq.com/s/8_jqJYzajpwrh-T2gpvB_A
某市商务委员会刚刚发了个网站(阅读原文),一共列出了1072个保供企业,而且只能用手机看,这哪看得过来?作为IT男,我导出了所有保供企业的清单,并且逐一匹配了这些企业的工商信息,给大家做一个大数据分析,并且冒昧的给这些企业颁个奖。
在这1072家企业中,参加社保的总人数为48529人,当然这只能作为参考,因为大型集团公司的可用人力,会远高于社保参保人数。
在1072个保供企业中,我们发现了非常可喜的现象,某市并不是任由巨型企业垄断的,还有一部分小微企业上榜。其中注册资金小于100万,但实缴资本为0元的企业,一共有57家,其中餐饮企业25家,超市企业15家,电商平台3家(这点钱做不起来电商),副食品7家,药品2家,物流总部3家,外资企业1家。
注册资金、实缴资本均小于100万的企业一共92家,约占某市保供企业总数的9%,他们的注册资金总数为6754万元,实缴资本5724万元,参保人数4456人。
参保人数低于3人的企业,一共130家,占某市保供企业总数的12%,他们的注册资金总数390亿,实缴资本335亿,参保人数40人,也从侧面证明,某市的确是个资本的乐土,130家公司只缴纳了40人的社保,却缴纳了335亿的资本金。
蚂蚁是真正的大力士,而某市的保供企业也有这样的大力士,他们以不到10名参保人员、不到100万的实缴资本,成功列入全市1072家的保供企业中!他们的实力不容怀疑!
排名第一的是,某市XX贸易经营部!他们以3万元注册资本、0元实缴资本、0参保人数,成功入围某市保供企业,获得蚂蚁奖第一名!大家鼓掌!这家公司的法人代表赵总实力雄厚,一共持有三家公司,注册资金分别是10万、3万和1万人民币!
特别指出的,是赵总豪掷1万元,开设了某市宝山区XX烟杂店!有效缓解了烟民的焦虑,提出表扬!
排名第二的是,某市XX农副产品市场经营管理有限公司,他们以3万元注册资本、3万元实缴资本、0参保人数,获得蚂蚁奖第二名!致敬!
特别指出的是,这家公司的经营范围,是为本市场内农副产品经营者提供管理服务,本身并不具备副食品销售的资格;市场内的商户,能否挂靠在该公司名下提供保供服务,组委会无法了解;市场管理公司是否有充足人力核查供应商的健康情况,组委会也无法了解,但是我们相信某市评选出来的保供企业,一定全是有实力的企业!
排名第三的是,广东XX食品股份有限公司某市分公司,他们以0元资本,3参保人数,获得蚂蚁奖第三名!而这家公司先后被某市处罚了三次,包括虚假广告和虚假标注食品生产日期,但是他们能够成功入围,就说明他们改过自新了!
排名第四的是,某市XX物流有限公司,注册资金50万,实缴资本50万,参保人数1人。这家公司的老板名下有两家公司,一家成立于2021年10月19日,注册资金200万;而本次入围的公司,注册资金只有50万,1人参保,想必这个人是个多面手吧,要不然又如何能入围呢?
排名第五的是,某市XX贸易有限公司,这家公司是一人公司,注册资金20万,实缴资本0元,参保人数0人,属于小规模纳税人,能列入保供企业,想必有自己独到的优势。
接下来我们颁布的是最具潜力小新人奖,他们成立不到一年,就能以不到100万的注册资本、0名参保人数,成功在超市商贸、电商平台、餐饮企业、零售药品四大领域,进入某市保供企业名单!他们是:
但排名第一的是,某市XX农特产有限公司,这家公司成立于2022年1月4日,注册资金300万,实缴资金0元,参保人数0人,就能成功作为电商平台入围某市保供企业!曾给闵行捐菜,点赞
更令人震惊的是,这家公司并不是背后有大树的公司,而是两个自然人联合成立的小公司!这家公司开业不到4个月,就被澎湃新闻报道,大家请看报道详情《万人团购被指不发货不退款 涉及金额超过300万》,因选的物流没派送导致
所以,这家公司被组委会授予最佳新人奖,名至实归。
另外,某市XX商贸有限公司,成立于2021年12月27日,注册资金100万,实缴0元,参保0人,便能纳入闵行区的保供企业名单,公司年龄同样不到4个月。
我狠狠批评了一下小奶猫球球,看你3个月大,现在除了吃喝睡加卖萌,别的什么都不会;你看看这些新公司,不到4个月大,都上了某市保供企业的光荣榜了!但球球跟我喵了一声,他敢骂我荒喵!
某市XX净生物技术有限公司,成立于2021年11月22日,注册资金2000万,实缴资金0元,社保0人,但是这家生物技术公司成功入围了某市保供企业超市商贸组!与他同行的巨无霸相比,这家新生企业毫不逊色。
二等奖,是某市XX茶点餐饮管理有限公司,注册资金200万,实缴资本0元,社保0人,但是这家餐饮管理公司,也成功入围了某市某市保供企业超市商贸组!其实这家公司是一等奖的有力竞争对手,但是餐饮离超市更近,生物技术离超市更远,所以不务正业奖颁发给了某市XXX生物技术有限公司!
三等奖,是某市金山区的某市XX医疗科技中心,注册资金100万,实缴100万他们纳入了零售药品组,但是他们的经营范围,竟然还有食品经营,而且是某市金山市场监督管理局发证的,所以,他们荣获了第三名!
四等奖,是某市杨浦区XXX智慧助老服务中心,注册资本5000元(没错,0.5万),经营范围是「开展科技助老的咨询及调研,推进科技助老项目的实施及应用;承接政府相关委托项目」,但是也入围了餐饮企业组,完全脱离了经营范围,组委会不知道这个保供企业是如何确定的。但是有一说一,这家企业至少一直在招餐厅厨师,应该至少能满足500人的食品需求吧。
XX川咖啡(某市)有限公司,成立于2021年8月19日,注册资金1000万,实缴资本0元,社保0人,能够作为一个单品,杀入某市保供企业目录,成为六家咖啡保供企业中的一员,跟星巴克、拉比卡、皮氏咖啡并列,这是何等的城市特色!
在六家咖啡企业中,他是当之无愧的城市特色奖。当然,除了XX川咖啡,还有长沙XX半咖啡,都是注册资金不高、实缴资金为零,参保人数为零的企业,所以,长沙XX半咖啡,理所应当获得二等奖。
除了咖啡,还有鞋子。得物作为电商平台上榜,如果我没记错,得物是卖潮鞋的,我还在上面买过AJ,但是现在某市正在封控,万万没想到竟然还有买潮鞋的需求?如果不是穿给大白看,那就是穿给方舱的病友看吧?毕竟有人在方舱开始谈恋爱了。
一等奖,某市XXX绿色食品发展有限公司,这家公司在2019年4月26日,被最高人民法院公示为失信公司,而且有200条风险,这样的企业能够入围保供名单,说明某市有关部门本着治病救人,既往不咎的原则选择了保供企业。
二等奖,某市XX副食品有限公司,这家公司的实际控制人是公示的老赖,被限制高消费,而且目前尚未执行法院裁定。当然,魔都还是要既往不咎嘛。
XXX源(某市)企业发展有限公司,注册资金3000万,实缴资本1.85万,参保人员两名,入围保供超市商贸组。我注意到这家公司,是因为他在缴纳实缴资本的企业里面,实缴资本是最低,只有1.85万元,额……
某市XX坊餐饮有限公司,这家公司像是未卜先知一样,在2022年1月24日,完成了注册资本金变更,把原来的100万注册资本,变更到了3800万注册资本;当然实缴资本和参保人数还是0,恩。
组委会没什么好说的。
XX商贸(某市)有限公司,入围外资企业保供名单,但是这家公司注册资金用的是人民币,而且只有100万,还是一个自然人独资的有限责任公司,不知道某市有关部门是如何认定外资企业的,这种工作细致程度,佩服。
某市稷良食品有限公司,即是超市商超类企业,又是外资企业,也是整个1072家保供企业里面的独一份,可能是讨个社稷良心的好彩头吧?
这个奖,要颁布给组织保供入围工作的某市商务委员会,因为他们写错了40家入围企业的名字,错误率达到4%,让我花费了大量时间去矫正数据。
有错别字的,比如某市常林汽车运费有限公司,某市鑫欣源(缘)副食品有限公司,某市长宁唐宫海鲜坊(舫)有限公司,猪和萝卜(某市)餐饮管理~~公司~~有限公司,某市又一村(春)农贸市场经营管理有限公司;
有缺字的,比如「杭州可靠护理用品」应该是「杭州可靠护理用品股份有限公司」;有多字的,比如「某市云丰国际物流(某市)有限公司」应该是「云丰国际物流(某市)有限公司」;
还有名字全部搞错的,比如某市七宝商城农产品交易市场、某市江杨农产品批发市场这种,一共24个,全名都应该是XX市场经营管理有限公司。
你说如果真的执行了,发现企业公章跟保供企业名单上的名称不符,结果物流车辆/人员又被居委、社区拦住了,这不是耽误事儿么?
错误名单如下,还是友情建议某市商务委员会尽快修改一下,知道大家抢菜很累,但是核心大事不能出错啊!何况还是错了40个企业名字……
超市商贸 | 杭州娃哈哈集团有限公司某市分公司 |
---|---|
超市商贸 | 某市爱婴室商务服务有限公司 |
超市商贸 | 某市全球儿童用品有限公司 |
超市商贸 | 某市笑昕食品有限公司 |
超市商贸 | 某市依牛羊食品有限公司 |
超市商贸 | 杭州可靠护理用品 |
电商平台 | 某市悠乐汇企业管理咨询有限公司 |
零售药品 | 某市万仕诚国药(药业)制品有限公司 |
外资企业 | 爱德华(某市)医疗用品公司 |
物流总部 | 某市晶粮实业(集团)股份有限公司 |
物流总部 | 某市鑫欣源副食品有限公司 |
物流总部 | 某市常林汽车运费有限公司 |
物流总部 | 某市云丰国际物流(某市)有限公司 |
餐饮企业 | 某市鑫博海农副畜产品加工有限公司 |
餐饮企业 | 某市龙神食品(集团)有限公司 |
餐饮企业 | 领驰食品发展(某市)有限公司 |
餐饮企业 | 某市长宁唐宫海鲜坊有限公司 |
餐饮企业 | 猪和萝卜(某市)餐饮管理公司有限公司 |
餐饮企业 | 某市煜盐餐饮有限公司 |
副食品 | 中粮家佳康(江苏)公司某市分公司 |
副食品 | 某市江桥批发市场 |
副食品 | 某市西郊国际农产品交易中心 |
副食品 | 某市拾分味道食品(集团)有限公司 |
副食品 | 某市国京农贸综合市场经营管理有限公司 |
副食品 | 某市申象农贸市场管理有限公司 |
副食品 | 某市又一村农贸市场经营管理有限公司 |
副食品 | 某市农产品中心批发市场 |
副食品 | 某市江杨农产品批发市场 |
副食品 | 某市江阳水产品批发交易市场 |
副食品 | 某市七宝商城农产品交易市场 |
副食品 | 某市浦南农副产品批发市场 |
副食品 | 某市东方国际水产中心市场 |
副食品 | 某市龙上农副产品批发市场 |
副食品 | 某市辉展果蔬市场 |
副食品 | 某市胜辛集贸市场 |
副食品 | 某市陇南肉类批发市场 |
副食品 | 某市新安农副产品市场 |
副食品 | 某市三林副食品批发交易市场 |
副食品 | 某市军工路国太综合市场 |
副食品 | 某市江杨水产批发交易市场 |
副食品 | 某市青莲上选食品公司 |
某市商务委员会发布的《疫情防控生活物资保障企业查询库》,入口在这里,请感兴趣的同学查收。我的数据时间是4月21日14点,当时还是1072家,目前某市保障企业已经变成1124家了,不过大数据分析还是有借鉴意义的。
感觉我每天都在封号的边缘疯狂试探,但我保证,我写的每篇文章,都有国家机关公开信源的数据支持。只要某市不再出新的魔幻事件,我保证我马上就不写某市了。
转自:https://mp.weixin.qq.com/s/2QdJhPA7m8HE46NXfSARig
今天小明哥要分享的主题是:改变类定义的神器-metaclass
看到标题,你可能会想改变类的定义有什么用呢?什么时候才需要使用metaclass呢?
今天我将带大家设计一个简单的orm框架,并简单剖析一下YAML这个序列化工具的原理。
说到metaclass,我们首先必须清楚一个最基础的概念就是对象是类的实例,而类是type的实例,重复一遍:
在面向对象的编程模型中,类就相当于一个房子的设计图纸,而对象则是根据这个设计图纸建出来的房子。
下图中,玩具模型就可以代表一个类,而具体生产出来的玩具就可以代表一个对象:
总之,类就是创建对象的模板。
而type又是创建类的模板,那么我们就可以通过type创建自己想要的类。
比如定义一个 Hello 的 class:
class Hello(object):
def hello(self, name='world'):
print('Hello, %s.' % name)
当 Python 解释器载入 hello 模块时,就会依次执行该模块的所有语句,执行结果就是动态创建出一个 Hello 的 class对象。
type()函数既可以查看一个类型或变量的类型,也可以根据参数创建出新的类型,比如上面那段类的定义本质上就是:
def hello(self, name='world'):
print('Hello, %s.' % name)
Hello = type('Hello', (object,), dict(hello=hello))
type()函数创建class 对象,依次传入 3 个参数:
class 类的名称;
继承的父类集合,注意 Python 支持多重继承,如果只有一个父类,别忘了 tuple 的单元素写法;
class 的方法名称与函数绑定以及字段名称与对应的值,这里我们把函数 fn 绑定到方法名 hello 上。
通过 type() 函数创建的类和直接写 class 是完全一样的,因为 Python 解释器遇到 class 定义时,仅仅是扫描一下class 定义的语法,然后调用 type() 函数创建出 class。
正常情况下,我们肯定都是用 class Xxx… 来定义类,但是type() 函数允许我们动态创建出类来,这意味着Python这门动态语言支持运行期动态创建类。你可能感受不到这有多强大,要知道想在静态语言运行期创建类,必须构造源代码字符串再调用编译器,或者借助一些工具生成字节码实现,本质上都是动态编译,会非常复杂。
那type和metaclass有什么关系呢?metaclass到底是什么呢?
我认为metaclass 其实就是type或type的子类,通过继承type,重载__call__
运算符,便可以在class类对象创建时作出一些修改。
对于类 MyClass:
class MyClass():
pass
其实相当于:
class MyClass(metaclass = type):
pass
一旦我们把它的 metaclass 设置成 MyMeta:
class MyClass(metaclass = MyMeta):
pass
MyClass 就不再由原生的 type 创建,而是会调用 MyMeta 的__call__
运算符重载。
class = type(classname, superclasses, attributedict)
## 变为了
class = MyMeta(classname, superclasses, attributedict)
对于具有继承关系的类:
class Foo(Bar):
pass
Python做了如下的操作:
假想一个很傻的例子,你决定在你的模块里所有的类的属性都应该是大写形式。有好几种方法可以办到,但其中一种就是通过在模块级别设定__metaclass__:
class UpperAttrMetaClass(type):
## __new__ 是在__init__之前被调用的特殊方法
## __new__是用来创建对象并返回之的方法
## 而__init__只是用来将传入的参数初始化给对象
## 你很少用到__new__,除非你希望能够控制对象的创建
## 这里,创建的对象是类,我们希望能够自定义它,所以我们这里改写__new__
## 如果你希望的话,你也可以在__init__中做些事情
## 还有一些高级的用法会涉及到改写__call__特殊方法,但是我们这里不用
def __new__(cls, future_class_name, future_class_parents, future_class_attr):
##遍历属性字典,把不是__开头的属性名字变为大写
newAttr = {}
for name,value in future_class_attr.items():
if not name.startswith("__"):
newAttr[name.upper()] = value
## 方法1:通过'type'来做类对象的创建
## return type(future_class_name, future_class_parents, newAttr)
## 方法2:复用type.__new__方法,这就是基本的OOP编程
## return type.__new__(cls, future_class_name, future_class_parents, newAttr)
## 方法3:使用super方法
return super(UpperAttrMetaClass, cls).__new__(cls, future_class_name, future_class_parents, newAttr)
class Foo(object, metaclass = UpperAttrMetaClass):
bar = 'bip'
print(hasattr(Foo, 'bar'))
## 输出: False
print(hasattr(Foo, 'BAR'))
## 输出:True
f = Foo()
print(f.BAR)
## 输出:'bip'
ORM全称“Object Relational Mapping”,即对象-关系映射,就是把关系数据库的一行映射为一个对象,也就是一个类对应一个表,这样,写代码更简单,不用直接操作SQL语句。
现在设计一下ORM框架的调用接口,比如用户想通过User
类来操作对应的数据库表User
,我们期待他写出这样的代码:
class User(Model):
## 定义类的属性到列的映射:
id = IntegerField('id')
name = StringField('username')
email = StringField('email')
password = StringField('password')
## 创建一个实例:
u = User(id=12345, name='xiaoxiaoming', email='test@orm.org', password='my-pwd')
## 保存到数据库:
u.save()
上面的接口通过常规方法很难或几乎很难实现,但通过metaclass就会相对比较简单。核心思想就是通过metaclass修改类的定义,将类的所有Field类型的属性,用一个额外的字典去保存,然后从原定义中删除。对于User创建对象时传入的参数(id=12345, name=’xiaoxiaoming’等)可以模仿字典的实现或直接继承dict类保存起来。
其中,父类Model
和属性类型StringField
、IntegerField
是由ORM框架提供的,剩下的魔术方法比如save()
全部由metaclass自动完成。虽然metaclass的编写会比较复杂,但ORM的使用者用起来却异常简单。
首先定义Field类,它负责保存数据库表的字段名和字段类型:
class Field(object):
def __init__(self, name, column_type):
self.name = name
self.column_type = column_type
def __str__(self):
return '<%s:%s>' % (self.__class__.__name__, self.name)
在Field的基础上,进一步定义各种类型的Field,比如StringField,IntegerField等等:
class StringField(Field):
def __init__(self, name):
super(StringField, self).__init__(name, 'varchar(100)')
class IntegerField(Field):
def __init__(self, name):
super(IntegerField, self).__init__(name, 'bigint')
下一步,编写ModelMetaclass:
class ModelMetaclass(type):
def __new__(cls, name, bases, attrs):
if name == 'Model':
return type.__new__(cls, name, bases, attrs)
print('Found model: %s' % name)
mappings = dict()
for k, v in attrs.items():
if isinstance(v, Field):
print('Found mapping: %s ==> %s' % (k, v))
mappings[k] = v
for k in mappings.keys():
attrs.pop(k)
attrs['__mappings__'] = mappings ## 保存属性和列的映射关系
attrs.setdefault('__table__', name) ## 当未定义__table__属性时,表名直接使用类名
return type.__new__(cls, name, bases, attrs)
以及基类Model:
class Model(dict, metaclass=ModelMetaclass):
def __init__(self, **kw):
super(Model, self).__init__(**kw)
def __getattr__(self, key):
try:
return self[key]
except KeyError:
raise AttributeError(r"'Model' object has no attribute '%s'" % key)
def __setattr__(self, key, value):
self[key] = value
def save(self):
fields = []
params = []
args = []
for k, v in self.__mappings__.items():
fields.append(v.name)
params.append('?')
args.append(getattr(self, k, None))
sql = 'insert into %s (%s) values (%s)' % (self.__table__, ','.join(fields), ','.join(params))
print('SQL: %s' % sql)
print('ARGS: %s' % str(args))
在ModelMetaclass
中,一共做了几件事情:
User
)中查找定义的类的所有属性,如果找到一个Field属性,就把它保存到一个__mappings__
的dict中,同时从类属性中删除该Field属性(避免实例的属性遮盖类的同名属性);__table__
字段时,直接将类名保存到__table__
字段中作为表名。在Model
类中,就可以定义各种操作数据库的方法,比如save()
,delete()
,find()
,update
等等。
我们实现了save()
方法,把一个实例保存到数据库中。因为有表名,属性到字段的映射和属性值的集合,就可以构造出INSERT
语句。
测试:
u = User(id=12345, name='xiaoxiaoming', email='test@orm.org', password='my-pwd')
u.save()
输出如下:
Found model: User
Found mapping: id ==> <IntegerField:id>
Found mapping: name ==> <StringField:username>
Found mapping: email ==> <StringField:email>
Found mapping: password ==> <StringField:password>
SQL: insert into User (id,username,email,password) values (?,?,?,?)
ARGS: [12345, 'xiaoxiaoming', 'test@orm.org', 'my-pwd']
测试2:
class Blog(Model):
__table__ = 'blogs'
id = IntegerField('id')
user_id = StringField('user_id')
user_name = StringField('user_name')
name = StringField('user_name')
summary = StringField('summary')
content = StringField('content')
b = Blog(id=12345, user_id='user_id1', user_name='xxm', name='orm框架的基本运行机制', summary="简单讲述一下orm框架的基本运行机制",
content="此处省略一万字...")
b.save()
输出:
Found model: Blog
Found mapping: id ==> <IntegerField:id>
Found mapping: user_id ==> <StringField:user_id>
Found mapping: user_name ==> <StringField:user_name>
Found mapping: name ==> <StringField:user_name>
Found mapping: summary ==> <StringField:summary>
Found mapping: content ==> <StringField:content>
SQL: insert into blogs (id,user_id,user_name,user_name,summary,content) values (?,?,?,?,?,?)
ARGS: [12345, 'user_id1', 'xxm', 'orm框架的基本运行机制', '简单讲述一下orm框架的基本运行机制', '此处省略一万字...']
可以看到,save()
方法已经打印出了可执行的SQL语句,以及参数列表,只需要真正连接到数据库,执行该SQL语句,就可以完成真正的功能。
YAML是一个家喻户晓的 Python 工具,可以方便地序列化 / 逆序列化结构数据。
官方文档:https://pyyaml.org/wiki/PyYAMLDocumentation
安装:
pip install pyyaml
YAMLObject 的任意子类支持序列化和反序列化(serialization & deserialization)。比如说下面这段代码:
import yaml
class Monster(yaml.YAMLObject):
yaml_tag = '!Monster'
def __init__(self, name, hp, ac, attacks):
self.name = name
self.hp = hp
self.ac = ac
self.attacks = attacks
def __repr__(self):
return f"{self.__class__.__name__}(name={self.name}, hp={self.hp}, ac={self.ac}, attacks={self.attacks})"
monster1 = yaml.load("""
--- !Monster
name: Cave spider
hp: [2,6]
ac: 16
attacks: [BITE, HURT]
""")
print(monster1, type(monster1))
monster2 = Monster(name='Cave lizard', hp=[3, 6], ac=16, attacks=['BITE', 'HURT'])
print(yaml.dump(monster2))
运行结果:
Monster(name=Cave spider, hp=[2, 6], ac=16, attacks=['BITE', 'HURT']) <class '__main__.Monster'>
!Monster
ac: 16
attacks: [BITE, HURT]
hp: [3, 6]
name: Cave lizard
这里面调用统一的 yaml.load(),就能把任意一个 yaml 序列载入成一个 Python Object;而调用统一的 yaml.dump(),就能把一个 YAMLObject 子类序列化。
对于 load() 和 dump() 的使用者来说,他们完全不需要提前知道任何类型信息,这让超动态配置编程成了可能。比方说,在一个智能语音助手的大型项目中,我们有 1 万个语音对话场景,每一个场景都是不同团队开发的。作为智能语音助手的核心团队成员,我不可能去了解每个子场景的实现细节。
在动态配置实验不同场景时,经常是今天我要实验场景 A 和 B 的配置,明天实验 B 和 C 的配置,光配置文件就有几万行量级,工作量不可谓不小。而应用这样的动态配置理念,就可以让引擎根据配置文件,动态加载所需要的 Python 类。
对于 YAML 的使用者也很方便,只要简单地继承 yaml.YAMLObject,就能让你的 Python Object 具有序列化和逆序列化能力。
据说即使是在大厂 Google 的 Python 开发者,发现能深入解释 YAML 这种设计模式优点的人,大概只有 10%。而能知道类似 YAML 的这种动态序列化 / 逆序列化功能正是用 metaclass 实现的人,可能只有 1% 了。而能够将YAML 怎样用 metaclass 实现动态序列化 / 逆序列化功能讲出一二的可能只有 0.1%了。
对于YAMLObject 的 load和dump() 功能,简单来说,我们需要一个全局的注册器,让 YAML 知道,序列化文本中的!Monster
需要载入成 Monster 这个 Python 类型,Monster 这个 Python 类型需要被序列化为!Monster
标签开头的字符串。
一个很自然的想法就是,那我们建立一个全局变量叫 registry,把所有需要逆序列化的 YAMLObject,都注册进去。比如下面这样:
registry = {}
def add_constructor(target_class):
registry[target_class.yaml_tag] = target_class
然后,在 Monster 类定义后面加上下面这行代码:
add_constructor(Monster)
这样的缺点很明显,对于 YAML 的使用者来说,每一个 YAML 的可逆序列化的类 Foo 定义后,都需要加上一句话add_constructor(Foo)
。这无疑给开发者增加了麻烦,也更容易出错,毕竟开发者很容易忘了这一点。
更优雅的实现方式自然是通过metaclass 解决了这个问题,YAML 的源码正是这样实现的:
class YAMLObjectMetaclass(type):
def __init__(cls, name, bases, kwds):
super(YAMLObjectMetaclass, cls).__init__(name, bases, kwds)
if 'yaml_tag' in kwds and kwds['yaml_tag'] is not None:
cls.yaml_loader.add_constructor(cls.yaml_tag, cls.from_yaml)
cls.yaml_dumper.add_representer(cls, cls.to_yaml)
## 省略其余定义
class YAMLObject(metaclass=YAMLObjectMetaclass):
yaml_loader = Loader
yaml_dumper = Dumper
## 省略其余定义
可以看到,YAMLObject 把 metaclass 声明成了 YAMLObjectMetaclass,YAMLObjectMetaclass则会改变YAMLObject类和其子类的定义,就是下面这行代码将YAMLObject 的子类加入到了yaml的两个全局注册表中:
cls.yaml_loader.add_constructor(cls.yaml_tag, cls.from_yaml)
cls.yaml_dumper.add_representer(cls, cls.to_yaml)
YAML 应用 metaclass,拦截了所有 YAMLObject 子类的定义。也就是说,在你定义任何 YAMLObject 子类时,Python 会强行插入运行上面这段代码,把我们之前想要的add_constructor(Foo)
和add_representer(Foo)
给自动加上。所以 YAML 的使用者,无需自己去手写add_constructor(Foo)
和add_representer(Foo)
。
这次分享主要是简单的浅析了 metaclass 的实现机制。通过实现一个orm框架并解读 YAML 的源码,相信你已经对metaclass 有了不错的理解。
metaclass 是 Python 黑魔法级别的语言特性,它可以改变类创建时的行为,这种强大的功能使用起来务必小心。
看完本文,你觉得装饰器和 metaclass 有什么区别呢?欢迎下方留言和我讨论。记得一键三连呦,笔芯!
转自:https://mp.weixin.qq.com/s/Ls2UP7Zh4f4dIrf38IVDQA
来源/无相财经(ID:wuxiangcj)