Python Pandas: รับดัชนีแถวที่คอลัมน์ตรงกับค่าที่แน่นอน


276

รับ DataFrame พร้อมคอลัมน์ "BoolCol" เราต้องการหาดัชนีของ DataFrame ที่ค่าสำหรับ "BoolCol" == True

ขณะนี้ฉันมีวิธีการทำซ้ำซึ่งทำงานได้อย่างสมบูรณ์แบบ:

for i in range(100,3000):
    if df.iloc[i]['BoolCol']== True:
         print i,df.iloc[i]['BoolCol']

แต่นี่ไม่ใช่วิธีที่ถูกต้องในการทำแพนด้า หลังจากการวิจัยบางอย่างฉันกำลังใช้รหัสนี้:

df[df['BoolCol'] == True].index.tolist()

อันนี้ให้ดัชนีรายการของฉัน แต่พวกเขาไม่ตรงกันเมื่อฉันตรวจสอบพวกเขาโดยทำ:

df.iloc[i]['BoolCol']

ผลที่ได้คือจริงเท็จ !!

สิ่งใดจะเป็นวิธีที่ถูกต้องของนุ่นในการทำเช่นนี้?

คำตอบ:


428

df.iloc[i]ส่งกลับแถวith ไม่ได้อ้างถึงป้ายกำกับดัชนีเป็นดัชนีที่ใช้ 0dfii

ในทางกลับกันแอตทริบิวต์indexจะส่งกลับป้ายดัชนีจริงไม่ใช่ดัชนีแถว:

df.index[df['BoolCol'] == True].tolist()

หรือเทียบเท่า

df.index[df['BoolCol']].tolist()

คุณสามารถเห็นความแตกต่างค่อนข้างชัดเจนโดยการเล่นกับ DataFrame ด้วยดัชนีที่ไม่ใช่ค่าเริ่มต้นที่ไม่เท่ากับตำแหน่งตัวเลขของแถว:

df = pd.DataFrame({'BoolCol': [True, False, False, True, True]},
       index=[10,20,30,40,50])

In [53]: df
Out[53]: 
   BoolCol
10    True
20   False
30   False
40    True
50    True

[5 rows x 1 columns]

In [54]: df.index[df['BoolCol']].tolist()
Out[54]: [10, 40, 50]

หากคุณต้องการที่จะใช้ดัชนี ,

In [56]: idx = df.index[df['BoolCol']]

In [57]: idx
Out[57]: Int64Index([10, 40, 50], dtype='int64')

จากนั้นคุณสามารถเลือกแถวโดยใช้locแทนiloc :

In [58]: df.loc[idx]
Out[58]: 
   BoolCol
10    True
40    True
50    True

[3 rows x 1 columns]

โปรดทราบว่าlocสามารถยอมรับอาร์เรย์บูลีน :

In [55]: df.loc[df['BoolCol']]
Out[55]: 
   BoolCol
10    True
40    True
50    True

[3 rows x 1 columns]

หากคุณมีอาร์เรย์บูลีนmaskและต้องการค่าดัชนีลำดับคุณสามารถคำนวณได้โดยใช้np.flatnonzero :

In [110]: np.flatnonzero(df['BoolCol'])
Out[112]: array([0, 3, 4])

ใช้df.ilocเพื่อเลือกแถวตามดัชนีลำดับ:

In [113]: df.iloc[np.flatnonzero(df['BoolCol'])]
Out[113]: 
   BoolCol
10    True
40    True
50    True

9
df.query('BoolCol')แต่อีกทางหนึ่งคือการทำ
Phillip Cloud

3
ฉันรู้ว่ามันเก่า แต่ฉันสงสัยว่ามีวิธีง่ายๆในการรับหมายเลขดัชนี 0 จากแบบสอบถาม ฉันต้องการหมายเลข iloc เพราะฉันต้องการเลือกบางแถวก่อนและหลังแถวพบเงื่อนไขบางอย่าง ดังนั้นแผนของฉันคือให้ดัชนี 0 ของแถวตรงตามเงื่อนไขจากนั้นสร้างส่วนสำหรับใช้ใน iloc () สิ่งเดียวที่ฉันเห็นคือ get_loc แต่มันใช้อาร์เรย์ไม่ได้
sheridp

3
@sheridp: หากคุณมีหน้ากากแบบบูลที่คุณสามารถหาดัชนีลำดับที่maskเป็นโดยใช้True np.flatnonzeroฉันได้แก้ไขโพสต์ด้านบนเพื่อแสดงความหมายของฉัน
unutbu

8
ข้อเสนอแนะของคุณindices = np.flatnonzero(df[col_name] == category_name)ทำให้ฉันได้รับสิ่งที่ชื่อของคำถามที่ถามซึ่งหายากบนอินเทอร์เน็ต
ClimbsRocks

หากคุณต้องการกลับดัชนีสิ่งที่เป็นค่าใช้จ่ายของ df [dftest] .index? สิ่งนี้สร้างเฟรมข้อมูลระดับกลางหรือไม่ (ซึ่งข้อมูลอาจเป็นกิกะไบต์) สิ่งที่เกี่ยวกับ dftest? สิ่งนี้ไม่ได้จัดสรรวัตถุระหว่างกลางที่มีขนาดใหญ่มากซึ่งดัชนีที่ส่งคืนอาจมีขนาดเล็กมากหรือแม้แต่ว่างเปล่า สิ่งเหล่านี้ได้รับการปรับให้เหมาะสมอย่างน่าอัศจรรย์โดยใช้มุมมองที่ขี้เกียจ ถ้าไม่เช่นนั้นแน่นอนจะต้องมีวิธีที่มีประสิทธิภาพ
user48956

31

สามารถทำได้โดยใช้ numpy where () ฟังก์ชั่น:

import pandas as pd
import numpy as np

In [716]: df = pd.DataFrame({"gene_name": ['SLC45A1', 'NECAP2', 'CLIC4', 'ADC', 'AGBL4'] , "BoolCol": [False, True, False, True, True] },
       index=list("abcde"))

In [717]: df
Out[717]: 
  BoolCol gene_name
a   False   SLC45A1
b    True    NECAP2
c   False     CLIC4
d    True       ADC
e    True     AGBL4

In [718]: np.where(df["BoolCol"] == True)
Out[718]: (array([1, 3, 4]),)

In [719]: select_indices = list(np.where(df["BoolCol"] == True)[0])

In [720]: df.iloc[select_indices]
Out[720]: 
  BoolCol gene_name
b    True    NECAP2
d    True       ADC
e    True     AGBL4

แม้ว่าคุณจะไม่ต้องการดัชนีสำหรับการแข่งขันเสมอ แต่ในกรณีที่คุณต้องการ:

In [796]: df.iloc[select_indices].index
Out[796]: Index([u'b', u'd', u'e'], dtype='object')

In [797]: df.iloc[select_indices].index.tolist()
Out[797]: ['b', 'd', 'e']

2

วิธีง่ายๆคือการรีเซ็ตดัชนีของ DataFrame ก่อนการกรอง:

df_reset = df.reset_index()
df_reset[df_reset['BoolCol']].index.tolist()

แฮ็คบิต แต่มันรวดเร็ว!


1

ก่อนอื่นคุณสามารถตรวจสอบได้queryเมื่อคอลัมน์เป้าหมายเป็นประเภทbool (PS: เกี่ยวกับวิธีใช้โปรดตรวจสอบลิงก์ )

df.query('BoolCol')
Out[123]: 
    BoolCol
10     True
40     True
50     True

หลังจากที่เรากรอง df ดั้งเดิมโดยคอลัมน์บูลีนเราสามารถเลือกดัชนี

df=df.query('BoolCol')
df.index
Out[125]: Int64Index([10, 40, 50], dtype='int64')

นอกจากนี้แพนด้ายังมีnonzeroเราเพียงแค่เลือกตำแหน่งของTrueแถวและใช้มันฝานDataFrameหรือindex

df.index[df.BoolCol.nonzero()[0]]
Out[128]: Int64Index([10, 40, 50], dtype='int64')

1

หากคุณต้องการใช้วัตถุ dataframe เพียงครั้งเดียวให้ใช้:

df['BoolCol'].loc[lambda x: x==True].index

0

ผมขยายคำถามนี้ว่าเป็นวิธีการที่จะได้รับrow, columnและvalueทุกแมตช์ที่คุ้มค่า?

นี่คือทางออก:

import pandas as pd
import numpy as np


def search_coordinate(df_data: pd.DataFrame, search_set: set) -> list:
    nda_values = df_data.values
    tuple_index = np.where(np.isin(nda_values, [e for e in search_set]))
    return [(row, col, nda_values[row][col]) for row, col in zip(tuple_index[0], tuple_index[1])]


if __name__ == '__main__':
    test_datas = [['cat', 'dog', ''],
                  ['goldfish', '', 'kitten'],
                  ['Puppy', 'hamster', 'mouse']
                  ]
    df_data = pd.DataFrame(test_datas)
    print(df_data)
    result_list = search_coordinate(df_data, {'dog', 'Puppy'})
    print(f"\n\n{'row':<4} {'col':<4} {'name':>10}")
    [print(f"{row:<4} {col:<4} {name:>10}") for row, col, name in result_list]

เอาท์พุท:

          0        1       2
0       cat      dog        
1  goldfish           kitten
2     Puppy  hamster   mouse


row  col        name
0    1           dog
2    0         Puppy
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.