亚洲免费在线-亚洲免费在线播放-亚洲免费在线观看-亚洲免费在线观看视频-亚洲免费在线看-亚洲免费在线视频

python 猴子補(bǔ)丁(monkey patch)

系統(tǒng) 2765 0

寫了一段時(shí)間java切回寫python偶爾會出現(xiàn)一些小麻煩,比如:在java中自定義對象變成json串很簡單,調(diào)用一個(gè)方法就行,但同樣的轉(zhuǎn)換在python中卻不太容易實(shí)現(xiàn)。在尋找python自定義對象轉(zhuǎn)json串的過程中,接觸到了猴子補(bǔ)丁這個(gè)東西,感覺還有點(diǎn)意思;本文先實(shí)現(xiàn)python自定義對象轉(zhuǎn)json串,再簡單談一下猴子補(bǔ)丁。

python自定義對象轉(zhuǎn)json串

python自帶的json包不支持自定義對象轉(zhuǎn)json串,在python中用json.dumps轉(zhuǎn)自定義對象時(shí)會報(bào)異常class is not JSON serializable,通過增加一段代碼補(bǔ)丁(稱作猴子補(bǔ)丁)便可實(shí)現(xiàn)自定義轉(zhuǎn)換,補(bǔ)丁代碼如下:

            
from json import JSONEncoder
  def _default(self, obj):
    return getattr(obj.__class__, "to_json", _default.default)(obj)
  _default.default = JSONEncoder().default
  default.JSONEncoder.default = _default

          

同時(shí)在自定義對象里面實(shí)現(xiàn)to_json方法。

            
class Tmp:
  def __init__(self, id, name):
    self.id = id
    self.name = name

  def to_json():
    # 返回自定義對象json串
    pass


          

最后保證補(bǔ)丁代碼在自定義對象轉(zhuǎn)json之前執(zhí)行過一次即可。

通過補(bǔ)丁代碼我們可以看到,代碼替換了json包的默認(rèn)轉(zhuǎn)json的方法,運(yùn)行了補(bǔ)丁代碼后,轉(zhuǎn)json的過程變成了先找對象的to_json屬性,在沒有to_json屬性的情況下才使用默認(rèn)的JSONEncoder.default的方法,也就是通過這么一個(gè)patch,增加了json包原來沒有的功能。

猴子補(bǔ)丁

關(guān)于猴子補(bǔ)丁為啥叫猴子補(bǔ)丁,據(jù)說是這樣子的:

這個(gè)叫法起源于Zope框架,大家在修正Zope的Bug的時(shí)候經(jīng)常在程序后面追加更新部分,這些被稱作是“雜牌軍補(bǔ)丁(guerilla patch)”,后來guerilla就漸漸的寫成了gorllia((猩猩),再后來就寫了monkey(猴子),所以猴子補(bǔ)丁的叫法是這么莫名其妙的得來的。

猴子補(bǔ)丁主要有以下幾個(gè)用處:

  • 在運(yùn)行時(shí)替換方法、屬性等
  • 在不修改第三方代碼的情況下增加原來不支持的功能
  • 在運(yùn)行時(shí)為內(nèi)存中的對象增加patch而不是在磁盤的源代碼中增加

例如:上面自定義對象轉(zhuǎn)json,在原有json包不滿足的條件下,只需要將以上的一個(gè)patch寫在一個(gè)文件里自己再import一次,便可實(shí)現(xiàn)自己想要的功能,這是非常方便的。

可以知道猴子補(bǔ)丁的主要功能便是在不去改變源碼的情況下而對功能進(jìn)行追加和變更;對于編程過程中使用一些第三方不滿足需求的情況下,使用猴子補(bǔ)丁是非常方便的。

猴子補(bǔ)丁,算是編程中的一個(gè)技巧了。

拓展

json包默認(rèn)轉(zhuǎn)json的過程

可以看一下json包里面轉(zhuǎn)json串的過程:

            
 def _iterencode(o, _current_indent_level):
    if isinstance(o, basestring):
      yield _encoder(o)
    elif o is None:
      yield 'null'
    elif o is True:
      yield 'true'
    elif o is False:
      yield 'false'
    elif isinstance(o, (int, long)):
      yield str(o)
    elif isinstance(o, float):
      yield _floatstr(o)
    elif isinstance(o, (list, tuple)):
      for chunk in _iterencode_list(o, _current_indent_level):
        yield chunk
    elif isinstance(o, dict):
      for chunk in _iterencode_dict(o, _current_indent_level):
        yield chunk
    else:
      if markers is not None:
        markerid = id(o)
        if markerid in markers:
          raise ValueError("Circular reference detected")
        markers[markerid] = o
      o = _default(o)
      for chunk in _iterencode(o, _current_indent_level):
        yield chunk
      if markers is not None:
        del markers[markerid]

          

其實(shí)就是一連串的if-elif-else,將所有的自建對象都匹配一遍,最后匹配不到的就報(bào)錯(cuò)了,所以自定義對象轉(zhuǎn)json自然會有問題。

其他實(shí)現(xiàn)自定義對象轉(zhuǎn)json的方法

其實(shí)json包的源碼文檔里面也有很詳細(xì)的別的自定義對象轉(zhuǎn)json的方法。

            
r'''
Specializing JSON object decoding::

  >>> import json
  >>> def as_complex(dct):
  ...   if '__complex__' in dct:
  ...     return complex(dct['real'], dct['imag'])
  ...   return dct
  ...
  >>> json.loads('{"__complex__": true, "real": 1, "imag": 2}',
  ...   object_hook=as_complex)
  (1+2j)
  >>> from decimal import Decimal
  >>> json.loads('1.1', parse_float=Decimal) == Decimal('1.1')
  True

Specializing JSON object encoding::

  >>> import json
  >>> def encode_complex(obj):
  ...   if isinstance(obj, complex):
  ...     return [obj.real, obj.imag]
  ...   raise TypeError(repr(o) + " is not JSON serializable")
  ...
  >>> json.dumps(2 + 1j, default=encode_complex)
  '[2.0, 1.0]'
  >>> json.JSONEncoder(default=encode_complex).encode(2 + 1j)
  '[2.0, 1.0]'
  >>> ''.join(json.JSONEncoder(default=encode_complex).iterencode(2 + 1j))
  '[2.0, 1.0]'
'''


          

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。


更多文章、技術(shù)交流、商務(wù)合作、聯(lián)系博主

微信掃碼或搜索:z360901061

微信掃一掃加我為好友

QQ號聯(lián)系: 360901061

您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點(diǎn)擊下面給點(diǎn)支持吧,站長非常感激您!手機(jī)微信長按不能支付解決辦法:請將微信支付二維碼保存到相冊,切換到微信,然后點(diǎn)擊微信右上角掃一掃功能,選擇支付二維碼完成支付。

【本文對您有幫助就好】

您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描上面二維碼支持博主2元、5元、10元、自定義金額等您想捐的金額吧,站長會非常 感謝您的哦!!!

發(fā)表我的評論
最新評論 總共0條評論
主站蜘蛛池模板: 国产免费自拍视频 | 欧美一级视频免费看 | 国产欧美久久久另类精品 | 国产精品视频第一区二区 | 久久99亚洲精品久久久久网站 | 中文字幕精品一区二区三区视频 | 天天插日日射 | 中文字幕一区日韩在线视频 | 日韩亚射 | 成人午夜爽爽爽免费视频 | 亚洲欧洲一区二区三区在线 | 欧美爱爱片 | 日韩精品一区二区三区四区 | 亚洲美女视频网 | 国产91成人精品亚洲精品 | 亚洲成人网在线观看 | 国产成人高清精品免费观看 | 婷婷成人基地 | 久久青青草视频 | 一本大道香蕉高清久久 | 免费 高清 日本1在线观看 | 欧做爰xxxⅹ性欧美大片孕妇 | 国产亚洲精品中文带字幕21页 | 久久精品综合网 | 高清国产天干天干天干不卡顿 | 国产一区二区在线免费观看 | 国产第六页 | 久久草国产 | 手机看片国产精品 | 日本草草视频 | 免费国产视频 | 亚洲一区在线视频观看 | 一区二区福利视频 | 成人美女免费网站视频 | 波多野结衣一区二区三区在线观看 | 亚洲成a人一区二区三区 | 中文字幕亚洲综久久2021 | 国产日本三级 | 久久精品在线免费观看 | 午夜看一级特黄a大片 | www.五月天激情 |