Python Tutorial이 칼럼에서는 JSON을 중첩하는 방법을 소개합니다
권장(무료): Python Tutorial
API
를 호출하면 문서 데이터베이스가 중첩된 상태로 반환됩니다. JSON
개체에서 Python
을 사용하여 중첩 구조의 키를 열로 변환하려고 하면 pandas
에 로드된 데이터는 종종 다음 결과: API
和文档数据库会返回嵌套的JSON
对象,当我们使用Python
尝试将嵌套结构中的键转换为列时,数据加载到pandas
中往往会得到如下结果:
df = pd.DataFrame.from_records(results [“ issues”],columns = [“ key”,“ fields”])
说明:这里results是一个大的字典,issues是results其中的一个键,issues的值为一个嵌套JSON对象字典的列表,后面会看到JSON嵌套结构。
问题在于API返回了嵌套的JSON
结构,而我们关心的键在对象中确处于不同级别。
嵌套的JSON
结构张成这样的。
而我们想要的是下面这样的。
下面以一个API返回的数据为例,API通常包含有关字段的元数据。假设下面这些是我们想要的字段。
如上,我们选择要提取的字段在issues列表内的JSON
结构中分别处于4个不同的嵌套级别,一环扣一环。
{ "expand": "schema,names", "issues": [ { "fields": { "issuetype": { "avatarId": 10300, "description": "", "id": "10005", "name": "New Feature", "subtask": False }, "status": { "description": "A resolution has been taken, and it is awaiting verification by reporter. From here issues are either reopened, or are closed.", "id": "5", "name": "Resolved", "statusCategory": { "colorName": "green", "id": 3, "key": "done", "name": "Done", } }, "summary": "Recovered data collection Defraglar $MFT problem" }, "id": "11861", "key": "CAE-160", }, { "fields": { ... more issues], "maxResults": 5, "startAt": 0, "total": 160 }
一个不太好的解决方案
一种选择是直接撸码,写一个查找特定字段的函数,但问题是必须对每个嵌套字段调用此函数,然后再调用.apply
到DataFrame
中的新列。
为获取我们想要的几个字段,首先我们提取fields键内的对象至列:
df = ( df["fields"] .apply(pd.Series) .merge(df, left_index=True, right_index = True) )
从上表看出,只有summary是可用的,issuetype、status等仍然埋在嵌套对象中。
下面是提取issuetype中的name的一种方法。
# 提取issue type的name到一个新列叫"issue_type" df_issue_type = ( df["issuetype"] .apply(pd.Series) .rename(columns={"name": "issue_type_name"})["issue_type_name"] ) df = df.assign(issue_type_name = df_issue_type)
像上面这样,如果嵌套层级特别多,就需要自己手撸一个递归来实现了,因为每层嵌套都需要调用一个像上面解析并添加到新列的方法。
对于编程基础薄弱的朋友,手撸一个其实还挺麻烦的,尤其是对于数据分析师,着急想用数据的时候,希望可以快速拿到结构化的数据进行分析。
下面东哥分享一个pandas
的内置解决方案。
内置的解决方案
pandas
中有一个牛逼的内置功能叫 .json_normalize
。
pandas
的文档中提到:将半结构化JSON
数据规范化为平面表。
前面方案的所有代码,用这个内置功能仅需要3行就可搞定。步骤很简单,懂了下面几个用法即可。
确定我们要想的字段,使用 . 符号连接嵌套对象。
将想要处理的嵌套列表(这里是results["issues"]
)作为参数放进 .json_normalize
中。
过滤我们定义的FIELDS列表。
FIELDS = ["key", "fields.summary", "fields.issuetype.name", "fields.status.name", "fields.status.statusCategory.name"] df = pd.json_normalize(results["issues"]) df[FIELDS]
没错,就这么简单。
其它操作
记录路径
除了像上面那样传递results["issues"]
列表之外,我们还使用record_path
参数在JSON
对象中指定列表的路径。
# 使用路径而不是直接用results["issues"] pd.json_normalize(results, record_path="issues")[FIELDS]
自定义分隔符
还可以使用sep参数自定义嵌套结构连接的分隔符,比如下面将默认的“.”替换“-”。
### 用 "-" 替换默认的 "." FIELDS = ["key", "fields-summary", "fields-issuetype-name", "fields-status-name", "fields-status-statusCategory-name"] pd.json_normalize(results["issues"], sep = "-")[FIELDS]
控制递归
如果不想递归到每个子对象,可以使用max_level
参数控制深度。在这种情况下,由于statusCategory.name
字段位于JSON
对象的第4级,因此不会包含在结果DataFrame
中。
# 只深入到嵌套第二级 pd.json_normalize(results, record_path="issues", max_level = 2)
下面是.json_normalize
的pandas
rrreee
JSON
구조를 반환하고 우리가 관심을 갖는 키가 실제로 객체의 다른 수준에 있다는 것입니다. 🎜🎜중첩된 JSON
구조는 다음과 같습니다. 🎜🎜그리고 우리가 원하는 것은 이런 것입니다. 🎜🎜다음은 API에서 반환한 데이터를 예로 들어 보겠습니다. API에는 일반적으로 관련 필드에 대한 메타데이터가 포함되어 있습니다. 이것이 우리가 원하는 필드라고 가정해 보겠습니다. 🎜JSON
구조에서 차례로 4개의 서로 다른 중첩 수준에 있습니다. 🎜rrreee🎜🎜별로 좋지 않은 해결책🎜🎜🎜한 가지 옵션은 직접 코딩하고 특정 필드를 찾는 함수를 작성하는 것이지만 문제는 이 함수를 각 중첩 필드에 대해 호출한 다음 를 호출해야 한다는 것입니다. <code>DataFrame
의 새 열에를 적용합니다. 🎜🎜원하는 여러 필드를 얻으려면 먼저 필드 키에서 열의 개체를 추출합니다. 🎜rrreee🎜위 표에서 볼 수 있듯이 요약만 사용할 수 있고 이슈 유형, 상태 등은 여전히 중첩되어 묻혀 있습니다. 사물. 🎜🎜다음은 issuetype에서 이름을 추출하는 방법입니다. 🎜rrreee🎜위처럼 중첩 수준이 너무 많으면 재귀를 직접 구현해야 합니다. 각 중첩 수준마다 위와 같은 메서드를 호출하여 구문 분석하고 새 열에 추가해야 하기 때문입니다. 🎜🎜프로그래밍 기초가 약한 친구들에게는 사실 하나 고르기가 상당히 까다롭습니다. 특히 데이터 분석가의 경우 데이터 사용에 대한 불안감이 있을 때 분석을 위해 구조화된 데이터를 빨리 얻고 싶어합니다. 🎜🎜이제 당 형제는 pandas
를 위한 내장 솔루션을 공유합니다. 🎜🎜🎜내장 솔루션🎜🎜🎜pandas
에는 .json_normalize
라는 멋진 내장 함수가 있습니다. 🎜🎜pandas
문서에서는 반구조화된 JSON
데이터를 플랫 테이블로 정규화한다고 언급합니다. 🎜🎜이 내장 함수를 사용하면 이전 솔루션의 모든 코드를 단 3줄로 완성할 수 있습니다. 단계는 매우 간단합니다. 다음 사용법을 이해하세요. 🎜🎜원하는 필드를 결정하고 . 기호를 사용하여 중첩된 개체를 연결합니다. 🎜🎜처리하려는 중첩 목록(여기서는 results["issues"]
)을 매개변수로 .json_normalize
에 넣습니다. 🎜🎜우리가 정의한 FIELDS 목록을 필터링하세요. 🎜rrreee🎜네, 정말 간단해요. 🎜🎜🎜기타 작업🎜🎜🎜🎜레코드 경로🎜🎜🎜위와 같이 results["issues"]
목록을 전달하는 것 외에도 다음에서 record_path
매개변수를 사용합니다. JSON 객체에 지정된 목록의 경로입니다. 🎜rrreee🎜🎜사용자 정의 구분 기호🎜🎜🎜 sep 매개 변수를 사용하여 중첩 구조 연결에 대한 구분 기호를 사용자 정의할 수도 있습니다. 예를 들어 아래에서 기본값 "."을 "-"로 바꿉니다. 🎜rrreee🎜🎜재귀 제어🎜🎜🎜각 하위 개체로 재귀하지 않으려면 max_level
매개변수를 사용하여 깊이를 제어할 수 있습니다. 이 경우 statusCategory.name
필드는 JSON
개체의 레벨 4에 있으므로 결과 DataFrame
에 포함되지 않습니다. 🎜rrreee🎜다음은 .json_normalize
에 대한 pandas
공식 문서 설명입니다. 이해가 안 되시면 이번에는 동 형제님이 직접 배워 보세요. 여기에 소개합니다. 🎜🎜pandas 공식 문서: https://pandas.pydata.org/pan...🎜위 내용은 Python은 중첩된 JSON을 도입하여 몇 초 만에 데이터프레임으로 변환합니다!의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!