【发布时间】:2016-09-26 20:53:00
【问题描述】:
我通过将 c 代码集成到我的 python 程序中来加速我的程序。我正在使用ctypes 从 python 中执行 c 中的函数。
c 程序:
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <math.h>
#define MAX_ITERATIONS 1000
static void calculate_pixel(int x, int y, int size, float*** img){
int i = 0;
float a = 0.0, b = 0.0;
while(a*a+b*b <= 4.0 && i < MAX_ITERATIONS){
float temp_a = a;
a = a*a - b*b + x;
b = 2*a*b + y;
i++;
}
float temp[3];
memset(&temp, 0, sizeof(float)*3);
if(i != MAX_ITERATIONS){
float brightness = logf(1.75 + i - logf(log(a*a + b*b))) / log(MAX_ITERATIONS);
temp[0] = brightness;
temp[1] = brightness;
temp[2] = 1;
}
memcpy(img[x][y], &temp, sizeof(float)*3);
}
float*** mandel(int size){
ssize_t len = (size)*sizeof(float*);
float*** img = malloc(len);
int x, y;
for(x = 0; x < size; x++){
img[x] = malloc(len);
for(y = 0; y < size; y++){
img[x][y] = malloc(sizeof(float)*3);
calculate_pixel(x, y, size, img);
}
}
return img;
}
Python 程序:
from ctypes import *
import matplotlib.pyplot as pl
size = 1000
lib = './mandelbrot3.so'
dll = cdll.LoadLibrary(lib)
dll.mandel.argtypes = [c_int]
#what i get in return from the c program
dll.mandel.restype = POINTER(POINTER(POINTER(c_float*3)*size)*size)
#calling function "mandel" in c program
res = dll.mandel(size)
#printing first value, does work this way
print(res.contains[0].contains[0].contains[0])
#creating a picture with pyplot, and inserting the array this way,
#does not work because its pointers
pl.imshow(res.contains)
pl.show()
dll.mandel.restype是一个大小为 1000*1000*3 的三元组指针。这是创建一个大小为 1000*1000 像素的图片,3 个浮点数是 rgb 值。
所以我的问题是我从 c 程序中得到的只是一个三元组指针。而且我需要能够将其转换为普通的 3d python 列表或 numpy 数组。有没有比使用 for 循环读取指针数组中的所有元素并将它们插入新列表或 numpy 数组更好的方法?
【问题讨论】:
-
这看起来不像一个 3D 数组,而是一个trippe-pointer。这些是根本不同的数据结构。
-
我也应该发布我的 c 代码吗?
-
注意:成为一个三星级的 C 程序员并不是恭维。
-
不是说...只是试图解决这个任务。如果您对我的烂摊子有更好的解决方案,请分享:)
标签: python c arrays python-3.x ctypes