>  기사  >  백엔드 개발  >  Python은 중첩된 JSON을 도입하여 몇 초 만에 데이터프레임으로 변환합니다!

Python은 중첩된 JSON을 도입하여 몇 초 만에 데이터프레임으로 변환합니다!

coldplay.xixi
coldplay.xixi원래의
2020-12-29 09:34:393358검색

Python Tutorial이 칼럼에서는 JSON을 중첩하는 방법을 소개합니다

Python은 중첩된 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通常包含有关字段的元数据。假设下面这些是我们想要的字段。

  • key:JSON密钥,在第一级的位置。
  • summary:第二级的“字段”对象。
  • status name:第三级位置。
  • statusCategory name:位于第4个嵌套级别。

如上,我们选择要提取的字段在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
}

一个不太好的解决方案

一种选择是直接撸码,写一个查找特定字段的函数,但问题是必须对每个嵌套字段调用此函数,然后再调用.applyDataFrame中的新列。

为获取我们想要的几个字段,首先我们提取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_normalizepandasrrreee

설명: 여기서 결과는 큰 사전이고, Issue는 결과의 키이며, Issue의 값은 중첩된 JSON 개체 사전 목록입니다. 나중에 JSON 중첩 구조를 볼 수 있습니다.
문제는 API가 중첩된 JSON 구조를 반환하고 우리가 관심을 갖는 키가 실제로 객체의 다른 수준에 있다는 것입니다. 🎜🎜중첩된 JSON 구조는 다음과 같습니다. 🎜🎜그리고 우리가 원하는 것은 이런 것입니다. 🎜🎜다음은 API에서 반환한 데이터를 예로 들어 보겠습니다. API에는 일반적으로 관련 필드에 대한 메타데이터가 포함되어 있습니다. 이것이 우리가 원하는 필드라고 가정해 보겠습니다. 🎜
  • 키: 첫 번째 수준의 JSON 키입니다.
  • 요약: 두 번째 수준의 "필드" 개체입니다.
  • 상태 이름: 세 번째 수준 위치.
  • statusCategory 이름: 4번째 중첩 수준에 위치합니다.
🎜위와 같이 추출하기로 선택한 필드는 문제 목록의 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 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.