Как я могу привести двойной указатель ctype к массиву numpy? - PullRequest
1 голос
/ 09 ноября 2019

Эй, я получил этот код, который я использую для приведения массива двумерных массивов к двойному указателю ctype.

import ctypes as ct
import numpy as np

arr = np.empty([500, 500], dtype=np.uint8)

UI8Ptr = ct.POINTER(ct.c_uint8)
UI8PtrPtr = ct.POINTER(UI8Ptr)

ct_arr = np.ctypeslib.as_ctypes(arr)
UI8PtrArr = UI8Ptr * ct_arr._length_
ct_ptr = ct.cast(UI8PtrArr(*(ct.cast(row, UI8Ptr) for row in ct_arr)), UI8PtrPtr)

Как я могу привести ct_ptr обратно к массиву двумерных массивов?

1 Ответ

1 голос
/ 10 ноября 2019

Примечание : это продолжение до [SO]: как передать 2d массив из Python в C? .

Один путь будет идтичерез следующие состояния:

  1. CTypes указатель
  2. Python списки
  3. 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) .

...