Python3自定義json逐層解析器代碼
用python3對(duì)json內(nèi)容逐層進(jìn)行解析,拿中國(guó)天氣網(wǎng)的接口返回?cái)?shù)據(jù)測(cè)試,
代碼如下:
# -*- coding: utf-8 -*- import operator as op from collections import defaultdict class Json(object): def __init__(self, json: str): sth = eval(json) load = lambda sth: sth if op.eq(type(sth).__name__, dict.__name__) else None self.json_dict = load(sth) self.ret_j = defaultdict(dict) self.analyze(self.json_dict) def analyze(self, j_dict: dict, lvl=0) -> None: lvl += 1 for k in j_dict: v = j_dict[k] v_type = type(v) try: self.ret_j[lvl][str(j_dict)].append(f"{k}:{v}") except: self.ret_j[lvl][str(j_dict)] = [] self.ret_j[lvl][str(j_dict)].append(f"{k}:{v}") if op.eq(v_type.__name__, dict.__name__): self.analyze(v, lvl) elif op.eq(v_type.__name__, list.__name__): for each in v: if op.eq(type(each).__name__, dict.__name__): self.analyze(each, lvl) def get_analysis(self) -> None: print(f"這個(gè)json攏共分{len(self.ret_j)}層") print("------") for lvl in self.ret_j: print(f"第{lvl}層解析") for root in self.ret_j[lvl]: print(f"解析內(nèi)容:{root}") for each in self.ret_j[lvl][root]: print(each) print("------") if __name__ == '__main__': try: import requests except: exit(0) url = "http://forecast.weather.com.cn/napi/h5map/city/101/jQuery1533133004035?callback=jQuery1533133004035" r = requests.get(url) d_r = r.content.decode() json_4_test = d_r[d_r.index("(") + 1:d_r.index(")")] Json(json_4_test).get_analysis()
其中json_4_test是待解析的json字符串。
設(shè)計(jì)思路:
補(bǔ)充知識(shí):python之logging模塊:將不同的日志寫入到不同的文件
如下所示:
import logging.config from logging import LogRecord # 通常用于Linux系統(tǒng)下,使控制臺(tái)輸出的日志帶顏色 class ColorFormatter(logging.Formatter): log_colors = { 'CRITICAL': '\033[0;31m', 'ERROR': '\033[0;33m', 'WARNING': '\033[0;35m', 'INFO': '\033[0;32m', 'DEBUG': '\033[0;00m', } def format(self, record: LogRecord) -> str: s = super().format(record) level_name = record.levelname if level_name in self.log_colors: return self.log_colors[level_name] + s + '\033[0m' return s class MyFilter400(logging.Filter): def filter(self, record: LogRecord): if record.msg.startswith("4"): return True return False class MyFilter300(logging.Filter): def filter(self, record: LogRecord): if record.msg.startswith("3"): return True return False LOG_LEVEL = logging.INFO LOGGER = { 'version': 1, 'disable_existing_loggers': True, 'formatters': { 'color': { 'class': '__main__.ColorFormatter', # 如果你的模塊不是寫在啟動(dòng)程序中,請(qǐng)將__main__更換成你模塊的路徑,下同 'format': '%(asctime)s [%(name)s] %(levelname)s: %(message)s' }, 'default': { 'class': 'logging.Formatter', 'format': '%(message)s' } }, 'filters': { 'filter_400': { '()': '__main__.MyFilter400' }, 'filter_300': { '()': '__main__.MyFilter300' } }, 'handlers': { 'console': { 'level': LOG_LEVEL, 'class': 'logging.StreamHandler', 'formatter': 'color', }, 'file1': { 'level': LOG_LEVEL, 'class': 'logging.FileHandler', 'mode': 'w', 'formatter': 'default', 'filename': '400_log.txt', 'encoding': 'utf-8', 'filters': ['filter_400'] }, 'file2': { 'level': LOG_LEVEL, 'class': 'logging.FileHandler', 'mode': 'w', 'formatter': 'default', 'filename': '300_log.txt', 'encoding': 'utf-8', 'filters': ['filter_300'] }, }, 'loggers': { '__main__': { 'handlers': ['file1', 'file2', 'console'], 'level': LOG_LEVEL, }, } } logging.config.dictConfig(LOGGER) logger = logging.getLogger(__name__) logger.debug('200,this is a logger debug message') logger.info('302,this is a logger info message') logger.warning('301,this is a logger warning message') logger.error('404,this is a logger error message') logger.critical('500,this is a logger critical message') print("%s" % __name__)
運(yùn)行效果圖:
控制臺(tái):
文件:
3開頭的寫入到300_log.txt
4開頭的寫入到400_log.txt
特別注意,使用過(guò)濾器的一個(gè)問(wèn)題
class MyFilter400And500(logging.Filter): def filter(self, record: LogRecord): if record.msg.startswith("4") or record.msg.startswith("5"): return True return False # record.msg = "404, %s, %s" logger.info(f"{status_code}, %s, %s", website, link) # record.msg = "%s, %s, %s",這就導(dǎo)致過(guò)濾器返回False logger.info("%s, %s, %s", status_code, website, link)
因此,如果發(fā)現(xiàn)消息沒有寫入文件,可能是消息格式的問(wèn)題。
目前,官方推薦字符串格式化的方式就是第一種方式,%s和.format()的方式都不如這個(gè)好。
以上這篇Python3自定義json逐層解析器代碼就是小編分享給大家的全部?jī)?nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
- Python3 json模塊之編碼解碼方法講解
- Python3中對(duì)json格式數(shù)據(jù)的分析處理
- python3實(shí)現(xiàn)從kafka獲取數(shù)據(jù),并解析為json格式,寫入到mysql中
- python3 實(shí)現(xiàn)的對(duì)象與json相互轉(zhuǎn)換操作示例
- python3 json數(shù)據(jù)格式的轉(zhuǎn)換(dumps/loads的使用、dict to str/str to dict、json字符串/字典的相互轉(zhuǎn)換)
- Python3爬蟲爬取百姓網(wǎng)列表并保存為json功能示例【基于request、lxml和json模塊】
- Python3實(shí)現(xiàn)將本地JSON大數(shù)據(jù)文件寫入MySQL數(shù)據(jù)庫(kù)的方法
- Python3實(shí)現(xiàn)的字典、列表和json對(duì)象互轉(zhuǎn)功能示例
- 解決python3 json數(shù)據(jù)包含中文的讀寫問(wèn)題
- Python3內(nèi)置json模塊編碼解碼方法詳解
相關(guān)文章
13行python代碼實(shí)現(xiàn)對(duì)微信進(jìn)行推送消息的示例代碼
本文主要介紹了13行python代碼實(shí)現(xiàn)對(duì)微信進(jìn)行推送消息的示例代碼,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-08-08Python中摘要算法MD5,SHA1簡(jiǎn)介及應(yīng)用實(shí)例代碼
這篇文章主要介紹了Python中摘要算法MD5,SHA1簡(jiǎn)介及應(yīng)用實(shí)例代碼,具有一定借鑒價(jià)值,需要的朋友可以參考下2018-01-01pandas DataFrame.to_sql()用法小結(jié)
Pandas是基于NumPy的一種工具,該工具是為了解決數(shù)據(jù)分析任務(wù)而創(chuàng)建的,本文主要介紹了pandas DataFrame.to_sql()用法小結(jié),感興趣的可以了解一下2024-02-02pandas中的DataFrame數(shù)據(jù)遍歷解讀
這篇文章主要介紹了pandas中的DataFrame數(shù)據(jù)遍歷解讀,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-12-12如何用C代碼給Python寫擴(kuò)展庫(kù)(Cython)
這篇文章主要介紹了如何用C代碼給Python寫擴(kuò)展庫(kù)(Cython),小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2019-05-05Pandas中DataFrame.head()函數(shù)的具體使用
DataFrame.head()是Pandas庫(kù)中一個(gè)非常重要的函數(shù),用于返回DataFrame對(duì)象的前n行,本文主要介紹了Pandas中DataFrame.head()函數(shù)的具體使用,感興趣的可以了解一下2024-07-07Python數(shù)據(jù)可視化之使用matplotlib繪制簡(jiǎn)單圖表
這篇文章主要為大家詳細(xì)介紹了使用matplotlib繪制簡(jiǎn)單圖表的方法,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來(lái)幫助2022-03-03