-
-
Notifications
You must be signed in to change notification settings - Fork 19.2k
Array Interface and Categorical internals Refactor #19268
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Changes from all commits
2ef5216
57e8b0f
01bd42f
ce81706
87a70e3
8c61886
cb41803
65d5a61
57c749b
6736b0f
e4acb59
0e9337b
df68f3b
2746a43
34d2b99
a484d61
04b2e72
df0fa12
e778053
d15a722
b5f736d
240e8f6
f9b0b49
7913186
df18c3b
ab2f045
520876f
4dfa39c
e252103
7110b2a
c59dca0
fc688a5
fbc8466
030bb19
0f4c2d7
f9316e0
9c06b13
7d2cf9c
afae8ae
cd0997e
1d6eb04
92aed49
34134f2
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1 +1,2 @@ | ||
from .base import ExtensionArray # noqa | ||
from .categorical import Categorical # noqa |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,247 @@ | ||
"""An interface for extending pandas with custom arrays.""" | ||
from pandas.errors import AbstractMethodError | ||
|
||
_not_implemented_message = "{} does not implement {}." | ||
|
||
|
||
class ExtensionArray(object): | ||
"""Abstract base class for custom 1-D array types. | ||
pandas will recognize instances of this class as proper arrays | ||
with a custom type and will not attempt to coerce them to objects. They | ||
may be stored directly inside a :class:`DataFrame` or :class:`Series`. | ||
Notes | ||
----- | ||
The interface includes the following abstract methods that must be | ||
implemented by subclasses: | ||
* __getitem__ | ||
* __len__ | ||
* dtype | ||
* nbytes | ||
* isna | ||
* take | ||
* copy | ||
* _formatting_values | ||
* _concat_same_type | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. The two lines above can be removed here and mentioned in the list below (actually only formatting_values, as concat_same_type is already there) |
||
Some additional methods are required to satisfy pandas' internal, private | ||
block API. | ||
* _concat_same_type | ||
* _can_hold_na | ||
This class does not inherit from 'abc.ABCMeta' for performance reasons. | ||
Methods and properties required by the interface raise | ||
``pandas.errors.AbstractMethodError`` and no ``register`` method is | ||
provided for registering virtual subclasses. | ||
ExtensionArrays are limited to 1 dimension. | ||
They may be backed by none, one, or many NumPy ararys. For example, | ||
``pandas.Categorical`` is an extension array backed by two arrays, | ||
one for codes and one for categories. An array of IPv6 address may | ||
be backed by a NumPy structured array with two fields, one for the | ||
lower 64 bits and one for the upper 64 bits. Or they may be backed | ||
by some other storage type, like Python lists. Pandas makes no | ||
assumptions on how the data are stored, just that it can be converted | ||
to a NumPy array. | ||
Extension arrays should be able to be constructed with instances of | ||
the class, i.e. ``ExtensionArray(extension_array)`` should return | ||
an instance, not error. | ||
Additionally, certain methods and interfaces are required for proper | ||
this array to be properly stored inside a ``DataFrame`` or ``Series``. | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. this is repetitive with above (the list of methods that are required), and there is also a typo in "for |
||
""" | ||
# ------------------------------------------------------------------------ | ||
# Must be a Sequence | ||
# ------------------------------------------------------------------------ | ||
def __getitem__(self, item): | ||
# type (Any) -> Any | ||
"""Select a subset of self. | ||
Parameters | ||
---------- | ||
item : int, slice, or ndarray | ||
* int: The position in 'self' to get. | ||
* slice: A slice object, where 'start', 'stop', and 'step' are | ||
integers or None | ||
* ndarray: A 1-d boolean NumPy ndarray the same length as 'self' | ||
Returns | ||
------- | ||
item : scalar or ExtensionArray | ||
Notes | ||
----- | ||
For scalar ``item``, return a scalar value suitable for the array's | ||
type. This should be an instance of ``self.dtype.type``. | ||
For slice ``key``, return an instance of ``ExtensionArray``, even | ||
if the slice is length 0 or 1. | ||
For a boolean mask, return an instance of ``ExtensionArray``, filtered | ||
to the values where ``item`` is True. | ||
""" | ||
raise AbstractMethodError(self) | ||
|
||
def __setitem__(self, key, value): | ||
# type: (Any, Any) -> None | ||
raise NotImplementedError(_not_implemented_message.format( | ||
type(self), '__setitem__') | ||
) | ||
|
||
def __len__(self): | ||
"""Length of this array | ||
Returns | ||
------- | ||
length : int | ||
""" | ||
# type: () -> int | ||
raise AbstractMethodError(self) | ||
|
||
# ------------------------------------------------------------------------ | ||
# Required attributes | ||
# ------------------------------------------------------------------------ | ||
@property | ||
def dtype(self): | ||
# type: () -> ExtensionDtype | ||
"""An instance of 'ExtensionDtype'.""" | ||
raise AbstractMethodError(self) | ||
|
||
@property | ||
def shape(self): | ||
# type: () -> Tuple[int, ...] | ||
return (len(self),) | ||
|
||
@property | ||
def ndim(self): | ||
# type: () -> int | ||
"""Extension Arrays are only allowed to be 1-dimensional.""" | ||
return 1 | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. this should be tested on registration of the sub-type There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. What do you mean "registration"? We could override If people want to mess with this, that's fine, their stuff just won't work with pandas. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. what I mean is that when you register things, we should actually test that the interface is respected. If we had final methods this would not be necessary, but if someone override ndim this is a problem. |
||
|
||
@property | ||
def nbytes(self): | ||
# type: () -> int | ||
"""The number of bytes needed to store this object in memory. | ||
If this is expensive to compute, return an approximate lower bound | ||
on the number of bytes needed. | ||
""" | ||
raise AbstractMethodError(self) | ||
|
||
# ------------------------------------------------------------------------ | ||
# Additional Methods | ||
# ------------------------------------------------------------------------ | ||
def isna(self): | ||
# type: () -> np.ndarray | ||
"""Boolean NumPy array indicating if each value is missing. | ||
This should return a 1-D array the same length as 'self'. | ||
""" | ||
raise AbstractMethodError(self) | ||
|
||
# ------------------------------------------------------------------------ | ||
# Indexing methods | ||
# ------------------------------------------------------------------------ | ||
def take(self, indexer, allow_fill=True, fill_value=None): | ||
# type: (Sequence[int], bool, Optional[Any]) -> ExtensionArray | ||
"""Take elements from an array. | ||
Parameters | ||
---------- | ||
indexer : sequence of integers | ||
indices to be taken. -1 is used to indicate values | ||
that are missing. | ||
allow_fill : bool, default True | ||
If False, indexer is assumed to contain no -1 values so no filling | ||
will be done. This short-circuits computation of a mask. Result is | ||
undefined if allow_fill == False and -1 is present in indexer. | ||
fill_value : any, default None | ||
Fill value to replace -1 values with. By default, this uses | ||
the missing value sentinel for this type, ``self._fill_value``. | ||
Notes | ||
----- | ||
This should follow pandas' semantics where -1 indicates missing values. | ||
Positions where indexer is ``-1`` should be filled with the missing | ||
value for this type. | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Let's strongly consider exposing a helper function to make this easier to write, or at least an example of what this would look like (we can save this for later). It's not obvious how to write this with NumPy. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I added an example that I hope gets things correct for an extension type backed by a NumPy structured array. One trouble with this providing a helper function is that we don't know much about how the extension array is actually storing the data. Although, we could rely on the assumption that the underlying storage is convertible to a NumPy array, and proceed from there. Though this would perhaps be sub-optimal for many extension arrays. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. For GeometryArray, we have followed the same idea the write There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. this is fine as it is a pandas standard. |
||
This is called by ``Series.__getitem__``, ``.loc``, ``iloc``, when the | ||
indexer is a sequence of values. | ||
Examples | ||
-------- | ||
Suppose the extension array somehow backed by a NumPy structured array | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. let's make this just "NumPy array", as this is not specific to structured arrays |
||
and that the underlying structured array is stored as ``self.data``. | ||
Then ``take`` may be written as | ||
.. code-block:: python | ||
def take(self, indexer, allow_fill=True, fill_value=None): | ||
mask = indexer == -1 | ||
result = self.data.take(indexer) | ||
result[mask] = self._fill_value | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. One question here is: should the keyword argument In any case some clarification would be helpful, also if it is just in the signature for compatibility but may be ignored (maybe in a follow-up). There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. re I think that since ExtensionArray.take returns an ExtensionArray, most implementations will just ignore fill_value. I'll clarify it in the docs. |
||
return type(self)(result) | ||
""" | ||
raise AbstractMethodError(self) | ||
|
||
def copy(self, deep=False): | ||
# type: (bool) -> ExtensionArray | ||
"""Return a copy of the array. | ||
Parameters | ||
---------- | ||
deep : bool, default False | ||
Also copy the underlying data backing this array. | ||
Returns | ||
------- | ||
ExtensionArray | ||
""" | ||
raise AbstractMethodError(self) | ||
|
||
# ------------------------------------------------------------------------ | ||
# Block-related methods | ||
# ------------------------------------------------------------------------ | ||
@property | ||
def _fill_value(self): | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. need to list this in the very top doc-string |
||
# type: () -> Any | ||
"""The missing value for this type, e.g. np.nan""" | ||
return None | ||
|
||
def _formatting_values(self): | ||
# type: () -> np.ndarray | ||
# At the moment, this has to be an array since we use result.dtype | ||
"""An array of values to be printed in, e.g. the Series repr""" | ||
raise AbstractMethodError(self) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Maybe we can provide a default implementation of There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Yes, I suppose that'll be OK for many implementations. |
||
|
||
@classmethod | ||
def _concat_same_type(cls, to_concat): | ||
# type: (Sequence[ExtensionArray]) -> ExtensionArray | ||
"""Concatenate multiple array | ||
Parameters | ||
---------- | ||
to_concat : sequence of this type | ||
Returns | ||
------- | ||
ExtensionArray | ||
""" | ||
raise AbstractMethodError(cls) | ||
|
||
def _can_hold_na(self): | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. need to list this in the very top doc-string |
||
# type: () -> bool | ||
"""Whether your array can hold missing values. True by default. | ||
Notes | ||
----- | ||
Setting this to false will optimize some operations like fillna. | ||
""" | ||
return True |
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -43,6 +43,8 @@ | |
from pandas.util._validators import validate_bool_kwarg | ||
from pandas.core.config import get_option | ||
|
||
from .base import ExtensionArray | ||
|
||
|
||
def _cat_compare_op(op): | ||
def f(self, other): | ||
|
@@ -148,7 +150,7 @@ def _maybe_to_categorical(array): | |
""" | ||
|
||
|
||
class Categorical(PandasObject): | ||
class Categorical(ExtensionArray, PandasObject): | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. By having our internal arrays inherit from There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. yeah, the methods in PandasObject needs to be ABC in the ExtensionArray There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. On the other hand, I don't think all methods/attributes of PandasObject should be added to the public ExtensionArray (to keep those internal + to not clutter the ExtensionArray API) There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. FYI, I'm consistently testing these changes against
Neither inherit from |
||
""" | ||
Represents a categorical variable in classic R / S-plus fashion | ||
|
||
|
@@ -2130,6 +2132,20 @@ def repeat(self, repeats, *args, **kwargs): | |
return self._constructor(values=codes, categories=self.categories, | ||
ordered=self.ordered, fastpath=True) | ||
|
||
# Implement the ExtensionArray interface | ||
@property | ||
def _can_hold_na(self): | ||
return True | ||
|
||
@classmethod | ||
def _concat_same_type(self, to_concat): | ||
from pandas.core.dtypes.concat import _concat_categorical | ||
|
||
return _concat_categorical(to_concat) | ||
|
||
def _formatting_values(self): | ||
return self | ||
|
||
# The Series.cat accessor | ||
|
||
|
||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'd advocate leaving base.py open for (near-)future usage as pandas-internal base and putting the "use this if you want to write your own" file in e.g. extension.py
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I have a slight preference for
base.py
since it's a base class for all extension arrays. I don't think that havingExtensionArray
inarrays.base
precludes having a pandas-internal base there as well.There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It will be publicly exposed through pd.api.extensions anyway I think
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Adding stuff to the public API is waiting on #19304