Python никогда не может вызвать DLL или C ++ файл, включая логику наследования? - PullRequest
0 голосов
/ 12 февраля 2019
class A{
  A();
  void a();
  virtual void v();
}

сделать это в liba.so

class B : A {
  B();
  void b();
  void v();
}
extern "C" {
  A* newB() {
    return new B();
  }
  void calla(A*b) {
    b->a();
  }
  void callv(A*b) {
     b->v();
  }
}

сделать это в libb.so

в python:

from ctypes import cdll 
lib = cdll.LoadLibrary('./libb.so')

class B(object):
  def __init__(self):
    self.obj = lib.newb()
  def a(self):
    lib.calla(self.obj)
  def v(self):
    lib.callv(self.obj)

b = B()
b.a()  # call base class function, that's ok
b.v()  # call derive class virtual function , segment fault!

Означает ли это, что Python не можетиспользовать dll, включая логику наследования?

кажется, что Python не может вызвать класс c ++, так как вход в его функцию изменил логику наследования bc, кто-нибудь может поговорить об этом?

Спасибо

1 Ответ

0 голосов
/ 12 февраля 2019

Примечания :

Я подготовил полный (и фиктивный) пример, чтобы проиллюстрировать поведение.

ах :

#pragma once
#define COUT() std::cout << __FILE__ << ":" << __LINE__ << "(" << __FUNCTION__ << ")\n"

#if defined(_WIN32)
#  define DLL_EXPORT __declspec(dllexport)
#else
#  define DLL_EXPORT
#endif


class DLL_EXPORT A {
public:
    A();
    virtual ~A();
    void a();
    virtual void v();
};

a.cpp :

#include "a.h"
#include <iostream>


A::A() {
    COUT();
}

A::~A() {
    COUT();
}

void A::a() {
    COUT();
}

void A::v() {
    COUT();
}

чч :

#pragma once
#include "a.h"


class B : public A {
public:
    B();
    void b();
    void v();
};


extern "C" {
    DLL_EXPORT A *newB() {
        return new B();
    }

    DLL_EXPORT void calla(A *b) {
        b->a();
    }

    DLL_EXPORT void callv(A *b) {
        b->v();
    }

    DLL_EXPORT void delB(A *b)
    {
        delete b;
    }
}

b.cpp :

#include "b.h"
#include <iostream>


B::B() : 
    A() {
    COUT();
}

void B::b() {
    COUT();
}

void B::v() {
    COUT();
}

code.py :

#!/usr/bin/env python3

import sys
import ctypes


LIB_NAME = "./libb.so"


class B(object):

    def __init__(self, lib_name=LIB_NAME):
        self.lib = ctypes.cdll.LoadLibrary(lib_name)
        self.lib.newB.restype = ctypes.c_void_p
        self.obj = self.lib.newB()

    def a(self):
        self.lib.calla.argtypes = [ctypes.c_void_p]
        self.lib.calla(self.obj)

    def v(self):
        self.lib.callv.argtypes = [ctypes.c_void_p]
        self.lib.callv(self.obj)

    def __del__(self):
        self.lib.delB.argtypes = [ctypes.c_void_p]
        self.lib.delB(self.obj)
        self.obj = None
        self.lib = None


def main():
    b = B()
    b.a()
    b.v()


if __name__ == "__main__":
    print("Python {:s} on {:s}\n".format(sys.version, sys.platform))
    main()

Выход :

[cfati@cfati-5510-0:/cygdrive/e/Work/Dev/StackOverflow/q054646019]> ls
a.cpp  a.h  b.cpp  b.h  code.py
[cfati@cfati-5510-0:/cygdrive/e/Work/Dev/StackOverflow/q054646019]> g++ -shared -fPIC -o liba.so a.cpp
[cfati@cfati-5510-0:/cygdrive/e/Work/Dev/StackOverflow/q054646019]> g++ -shared -fPIC -o libb.so b.cpp ./liba.so
[cfati@cfati-5510-0:/cygdrive/e/Work/Dev/StackOverflow/q054646019]> ls
a.cpp  a.h  b.cpp  b.h  code.py  liba.so  libb.so
[cfati@cfati-5510-0:/cygdrive/e/Work/Dev/StackOverflow/q054646019]> python3 code.py
Python 3.6.4 (default, Jan  7 2018, 15:53:53)
[GCC 6.4.0] on cygwin

a.cpp:6(A)
b.cpp:7(B)
a.cpp:14(a)
b.cpp:15(v)
a.cpp:10(~A)
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...