Python调用C库-对象地址

问题描述:

    python程序使用c库(内部有C++实现的类)实现目标功能。当出现这种情况时,python调用接口使库在堆上创建对象并返回该对象地址,使用该地址和库函数完成相应的功能,调用接口释放该地址上的对象,程序会出现内存异常。

    在这里,python将保存的对象地址传递给C库函数可以完成相应的功能,即库函数可以通过该地址调用该对象的成员函数,但C库函数不能通过该地址析构对象。

    C库实现接口及计算类:

    头文件CDllForPythonApi.h

#pragma once
#include "CDllForPythonExport.h"

extern "C"
{
	// 创建计算对象
	void CDll_API *Create_Obj();

	// 删除计算对象
	int CDll_API Delete_Obj(void* obj);

	// 测试程序
	int CDll_API Do_Test(void* obj);
}

    源文件CDllForPythonApi.cpp

#include "pch.h"
#include "CDllForPythonApi.h"

using namespace std;

class Calculate
{
public:
	Calculate() {};
	~Calculate() {};

public:
	int DoTest() { return 1; }
};


void* Create_Obj()
{
	auto pObj = new Calculate();

	// 输出对象地址
	fstream file;
	file.open("LogData.txt", ios::out | ios::app);
	file << "创建对象 ->" << pObj << endl;
	file.close();

	return pObj;
}
int Delete_Obj(void* pObj)
{
	auto pCalObj = static_cast<Calculate*>(pObj);
	if (pCalObj == nullptr)
		return -1;

	// 输出对象地址
	fstream file;
	file.open("LogData.txt", ios::out | ios::app);
	file << "执行删除 ->" << pCalObj << endl;
	file.close();

	delete pCalObj;
	return 0;
}

int Do_Test(void* pObj)
{
	auto pCalObj = static_cast<Calculate*>(pObj);
	if (pCalObj == nullptr)
		return -1;

	// 输出对象地址
	fstream file;
	file.open("LogData.txt", ios::out | ios::app);
	file << "执行计算 ->" << pCalObj << endl;
	file.close();

	return pCalObj->DoTest();
}

    Python文件调用C库函数并输出结果:

#!/usr/bin/python
# -*- coding: UTF-8 -*-

import platform
import ctypes


class Calc:
    # 初始化类
    def __init__(self):
        self.obj = ctypes.c_void_p
        if platform.system() == "Windows":
            self.cdll = ctypes.cdll.LoadLibrary("./CDllForPython.dll")

    def do_create_obj(self):
        self.obj = self.cdll.Create_Obj()

    def do_delete_obj(self):
        res = self.cdll.Delete_Obj(self.obj)
        return res

    def do_test(self):
        res = self.cdll.Do_Test(self.obj)
        return res


if __name__ == '__main__':
    obj = Calc()
    obj.do_create_obj()
    res_v = obj.do_test()
    print("do_test计算结果" + str(res_v))
    res_v = obj.do_delete_obj()
    print("删除对象")

    日志文件输出的对象地址为:

 

问题处理:

    通过分析,C库创建对象返回的地址与python程序所存储的地址并不相同。通过python保存的地址可以调用对象函数执行计算,但不可以析构对象;如果在创建时,C库保存对象地址,那么可以通过该地址析构对象。综上,C库创建对象后,由C库保存对象地址并返回唯一标识,python通过接口函数和唯一标识调用C库执行程序。

问题探究:

    之所以会出现python保存的地址能够调用C库函数,却不能析构的现象,初步猜想是python机制造成的。具体是什么机制待研究,如果有清楚的大神,还望解惑!

        

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值