Próbuję zmodyfikować kod Brandona Rhodesa Routines that examine the internals of a CPython dictionary, aby działał dla wersji CPython 3.3.python 3.3 dict: jak przekonwertować strukturę PyDictKeysObject na Pythona?
Wierzę, że udało mi się przetłumaczyć tę strukturę.
typedef PyDictKeyEntry *(*dict_lookup_func)
(PyDictObject *mp, PyObject *key, Py_hash_t hash, PyObject ***value_addr);
struct _dictkeysobject {
Py_ssize_t dk_refcnt;
Py_ssize_t dk_size;
dict_lookup_func dk_lookup;
Py_ssize_t dk_usable;
PyDictKeyEntry dk_entries[1];
};
myślę dodaje dobrze wygląda teraz:
from ctypes import Structure, c_ulong, POINTER, cast, py_object, CFUNCTYPE
LOOKUPFUNC = CFUNCTYPE(POINTER(PyDictKeyEntry), POINTER(PyDictObject),
py_object, c_ulong, POINTER(POINTER(py_object)))
class PyDictKeysObject(Structure):
"""A key object"""
_fields_ = [
('dk_refcnt', c_ssize_t),
('dk_size', c_ssize_t),
('dk_lookup', LOOKUPFUNC),
('dk_usable', c_ssize_t),
('dk_entries', PyDictKeyEntry * 1),
]
PyDictKeysObject._dk_entries = PyDictKeysObject.dk_entries
PyDictKeysObject.dk_entries = property(lambda s:
cast(s._dk_entries, POINTER(PyDictKeyEntry * s.dk_size))[0])
Ta linia kodu działa teraz, gdzie d == {0: 0, 1: 1, 2: 2, 3: 3}
:
obj = cast(id(d), POINTER(PyDictObject)).contents # works!!`
Oto moje tłumaczenie z C struct PyDictObject:
class PyDictObject(Structure): # an incomplete type
"""A dictionary object."""
def __len__(self):
"""Return the number of dictionary entry slots."""
pass
def slot_of(self, key):
"""Find and return the slot at which `key` is stored."""
pass
def slot_map(self):
"""Return a mapping of keys to their integer slot numbers."""
pass
PyDictObject._fields_ = [
('ob_refcnt', c_ssize_t),
('ob_type', c_void_p),
('ma_used', c_ssize_t),
('ma_keys', POINTER(PyDictKeysObject)),
('ma_values', POINTER(py_object)), # points to array of ptrs
]
Uwaga: możesz utworzyć link do [hg.python.org bezpośrednio] (http://hg.python.org/cpython/file/3.3/Objects/dictobject.c#l72). Wypróbuj 'ctypes.CFUNCTYPE' dla zdefiniowanego' dict_lookup_func'. – jfs
AKTUALIZACJA: Teraz zadeklarowałem typ dk_lookup przy użyciu CFUNCTYPE: – LeslieK
@ J.F.Sebastian: Dziękuję. Teraz zadeklarowałem typ dk_lookup przy użyciu CFUNCTYPE. Czy dk_entries wygląda dobrze? Kod C używa dk_entries [1]. – LeslieK