在C++中,“類函數(shù)指針”和傳統(tǒng)的“函數(shù)指針”,是兩個(gè)完全不同的東西。
成都創(chuàng)新互聯(lián)專注于網(wǎng)站建設(shè),為客戶提供成都網(wǎng)站建設(shè)、成都網(wǎng)站設(shè)計(jì)、網(wǎng)頁(yè)設(shè)計(jì)開(kāi)發(fā)服務(wù),多年建網(wǎng)站服務(wù)經(jīng)驗(yàn),各類網(wǎng)站都可以開(kāi)發(fā),品牌網(wǎng)站設(shè)計(jì),公司官網(wǎng),公司展示網(wǎng)站,網(wǎng)站設(shè)計(jì),建網(wǎng)站費(fèi)用,建網(wǎng)站多少錢,價(jià)格優(yōu)惠,收費(fèi)合理。
你取一個(gè)類的成員函數(shù)的地址,得到的是一個(gè)類函數(shù)指針,也叫成員函數(shù)指針。即使你的成員函數(shù)定義看起來(lái)和普通函數(shù)原型完全一樣,它也和這個(gè)原型的普通函數(shù)指針完全不同,彼此之間不能轉(zhuǎn)換。
PyEval_SetTrace要求傳入的是一個(gè)傳統(tǒng)的函數(shù)指針,你傳入一個(gè)類函數(shù)指針,當(dāng)然是不行的。編譯器報(bào)錯(cuò)是說(shuō)無(wú)法將一個(gè)類函數(shù)指針轉(zhuǎn)換為函數(shù)指針。
C++的類函數(shù)指針,是一個(gè)非常難用的東西,有非常多奇怪的特性,而且不同編譯器對(duì)它的支持大不相同,是C++著名的復(fù)雜性來(lái)源之一,建議不要使用。
你想要的東西,實(shí)際上是一個(gè)“委托”的概念,不過(guò)可惜的是C++并不支持委托。使用boost::function可以實(shí)現(xiàn)類似功能,但python的C API接口卻不支持boost::function,所以也不行。
你這種情況,最簡(jiǎn)單的方法還是用傳統(tǒng)的函數(shù)指針,使用普通函數(shù)包裝下類的成員函數(shù),然后把普通函數(shù)的指針傳給python。
int* GrabImage();
int GetPixel(int* image, int x, int y);
void SetPixel(int* image, int x, int y, int color);
如果您曾經(jīng)使用過(guò)C或C ++等低級(jí)語(yǔ)言,那么您可能已經(jīng)聽(tīng)說(shuō)過(guò)指針。指針允許您在部分代碼中創(chuàng)建高效率。它們也會(huì)給初學(xué)者帶來(lái)困惑,并且可能導(dǎo)致各種內(nèi)存管理錯(cuò)誤,即使對(duì)于專家也是如此。那么在Python中有指針的存在嗎?
指針廣泛用于C和C ++。本質(zhì)上,它們是保存另一個(gè)變量的內(nèi)存地址的變量。有關(guān)指針的更新,可以考慮在C指針上查看此概述。
為什么Python沒(méi)有指針?
實(shí)際上指針為何不存在的原因現(xiàn)在還不知道,也許指針違背了Python的禪宗。指針鼓勵(lì)隱含的變化而不是明確的變化。但通常情況下,它們很復(fù)雜而不是很簡(jiǎn)單,特別是對(duì)于初學(xué)者。更糟糕的是,當(dāng)他們用指針指向自己的方法,或做一些非常危險(xiǎn)的事情,比如從你無(wú)法獲取的的一些變量中讀取數(shù)據(jù)。
Python更傾向于嘗試從用戶那里抽象出內(nèi)存地址來(lái)實(shí)現(xiàn)具體細(xì)節(jié),所以Python通常關(guān)注可用性而不是速度。因此,Python中的指針并沒(méi)有多大意義。但是在有些情況下,Python會(huì)為您提供使用指針的一些好處。
想要理解Python中的指針,需要理解Python實(shí)現(xiàn)指針功能的具體細(xì)節(jié)。簡(jiǎn)單來(lái)說(shuō),需要了解這些知識(shí)點(diǎn):
不可變對(duì)象和可變對(duì)象【Python中的對(duì)象】
Python變量/名稱【Python中的變量】
【在Python中模擬實(shí)現(xiàn)指針】
test.c(動(dòng)態(tài)庫(kù)源代碼)
[cpp] view plain copy
// 編譯生成動(dòng)態(tài)庫(kù): gcc -g -fPIC -shared -o libtest.so test.c
#include stdio.h
#include string.h
#include stdlib.h
typedef struct StructPointerTest
{
char name[20];
int age;
}StructPointerTest, *StructPointer;
StructPointer test() // 返回結(jié)構(gòu)體指針
{
StructPointer p = (StructPointer)malloc(sizeof(StructPointerTest));
strcpy(p-name, "Joe");
p-age = 20;
return p;
}
編譯:gcc -g -fPIC -shared -o libtest.so test.c
call.py(python調(diào)用C語(yǔ)言生成的動(dòng)態(tài)庫(kù)):
[python] view plain copy
#!/bin/env python
# coding=UTF-8
from ctypes import *
#python中結(jié)構(gòu)體定義
class StructPointer(Structure):
_fields_ = [("name", c_char * 20), ("age", c_int)]
if __name__ == "__main__":
lib = cdll.LoadLibrary("./libtest.so")
lib.test.restype = POINTER(StructPointer)
p = lib.test()
print "%s: %d" %(p.contents.name, p.contents.age)
最后運(yùn)行結(jié)果:
[plain] view plain copy
[zcm@c_py #112]$make clean
rm -f *.o libtest.so
[zcm@c_py #113]$make
gcc -g -fPIC -shared -o libtest.so test.c
[zcm@c_py #114]$./call.py
Joe: 20
[zcm@c_py #115]$