Примечание : это продолжение до [SO]: как передать 2d массив из Python в C? .
Один путь будет идтичерез следующие состояния:
- CTypes указатель
- Python списки
- NumPy массив
Автоматически возникают 2 наблюдения:
- Это очень неэффективно, но, к сожалению, я не знаю, как избавиться от # 2. (который является виновником)
- Указатель содержит информацию о базовом типе, но не об измерении (ях) массива, из которого он был создан, поэтому вы должны их "сохранить",так как они понадобятся вам при «деконструкции» указателя
code00.py :
#!/usr/bin/env python3
import sys
import ctypes as ct
import numpy as np
def main():
dim0 = 3
dim1 = 4
arr0 = np.empty([dim0, dim1], dtype=np.uint8)
print("Initial array:\n{0:}".format(arr0))
UI8Ptr = ct.POINTER(ct.c_uint8)
UI8PtrPtr = ct.POINTER(UI8Ptr)
ct_arr = np.ctypeslib.as_ctypes(arr0)
UI8PtrArr = UI8Ptr * ct_arr._length_
ct_ptr = ct.cast(UI8PtrArr(*(ct.cast(row, UI8Ptr) for row in ct_arr)), UI8PtrPtr)
arr1 = np.array([[ct_ptr[i][j] for j in range(dim1)] for i in range(dim0)], dtype=ct_ptr._type_._type_._type_)
print("\nFinal array:\n{0:}".format(arr1))
print("\nEqual arrays: {0:}".format(np.array_equal(arr0, arr1)))
if __name__ == "__main__":
print("Python {0:s} {1:d}bit on {2:s}\n".format(" ".join(item.strip() for item in sys.version.split("\n")), 64 if sys.maxsize > 0x100000000 else 32, sys.platform))
main()
print("\nDone.")
Вывод :
[cfati@CFATI-5510-0:e:\Work\Dev\StackOverflow\q058781199]> "e:\Work\Dev\VEnvs\py_064_03.07.03_test0\Scripts\python.exe" code00.py
Python 3.7.3 (v3.7.3:ef4ec6ed12, Mar 25 2019, 22:22:05) [MSC v.1916 64 bit (AMD64)] 64bit on win32
Initial array:
[[ 96 100 101 115]
[105 114 101 100]
[ 96 96 32 115]]
Final array:
[[ 96 100 101 115]
[105 114 101 100]
[ 96 96 32 115]]
Equal arrays: True
Done.
Для получения дополнительной информации по этой теме проверьте [SO]: ошибка при преобразовании массива типов ctypes в указатель типа void (ответ @ CristiFati) .