рдкрд╛рдЗрдерди (ctypes) рд╕реЗ C / C ++

рдореБрдЦреНрдп

рдореИрдВрдиреЗ рдкрд┐рдЫрд▓реЗ рд▓реЗрдЦ рдореЗрдВ C рд╕реЗ рдкрд╛рдпрдерди рдХреЛ рдХреЙрд▓ рдХрд░рдиреЗ рдХреЗ рддрд░реАрдХреЗ рдХреЗ рдмрд╛рд░реЗ рдореЗрдВ рд▓рд┐рдЦрд╛ рдерд╛, рдЕрдм рд╣рдо рдЗрд╕рдХреЗ рдмрд╛рд░реЗ рдореЗрдВ рдмрд╛рдд рдХрд░реЗрдВрдЧреЗ рдХрд┐ Python3 рд╕реЗ C / C ++ рдХреЛ рдХреИрд╕реЗ рдХреЙрд▓ рдХрд░реЗрдВред рдПрдХ рдмрд╛рд░ рдЬрдм рдореИрдВрдиреЗ рдЗрд╕ рдмрд╛рд░реЗ рдореЗрдВ рд▓рд┐рдЦрдирд╛ рд╢реБрд░реВ рдХрд┐рдпрд╛, рддреЛ рд╣рдо рдкреВрд░реЗ рд╡рд┐рд╖рдп рдХреЛ рдЕрдВрдд рддрдХ рдкреНрд░рдХрдЯ рдХрд░реЗрдВрдЧреЗред рдЗрд╕рдХреЗ рдЕрд▓рд╛рд╡рд╛, рдпрд╣рд╛рдБ рдХреБрдЫ рднреА рдЬрдЯрд┐рд▓ рдирд╣реАрдВ рд╣реИред


рд╕реА


рдпрд╣рд╛рдВ рд╕рдм рдХреБрдЫ рд╕рд░рд▓ рд╣реИ, рдЕрдЬрдЧрд░ рдмрд┐рдирд╛ рдХрд┐рд╕реА рд╕рдорд╕реНрдпрд╛ рдХреЗ рд╕реА рдлрд╝рдВрдХреНрд╢рди рдХреЛ рдХреЙрд▓ рдХрд░ рд╕рдХрддрд╛ рд╣реИред


test.c:


#include "test.h" int a = 5; double b = 5.12345; char c = 'X'; int func_ret_int(int val) { printf("get func_ret_int: %d\n", val); return val; } double func_ret_double(double val) { printf("get func_ret_double: %f\n", val); return val; } char * func_ret_str(char *val) { printf("get func_ret_str: %s\n", val); return val; } char func_many_args(int val1, double val2, char val3, short val4) { printf("get func_many_args: int - %d, double - %f, char - %c, short - %d\n", val1, val2, val3, val4); return val3; } 

test.h:


 #ifndef _TEST_H_ #define _TEST_H_ #ifdef __cplusplus extern "C" { #endif #include <stdio.h> #include <string.h> #include <unistd.h> int func_ret_int(int val); double func_ret_double(double val); char *func_ret_str(char *val); char func_many_args(int val1, double val2, char val3, short val4) #ifdef __cplusplus } #endif #endif /* _TEST_H_ */ 

рдХреИрд╕реЗ рд╕рдВрдХрд▓рд┐рдд рдХрд░реЗрдВ:


 gcc -fPIC -shared -o libtest.so test.c 

рд╕реНрд░реЛрдд рдПрдХ рдЧрддрд┐рд╢реАрд▓ рдкреБрд╕реНрддрдХрд╛рд▓рдп рдореЗрдВ рд╕рдВрдХрд▓рд┐рдд рд╣реЛрддрд╛ рд╣реИ рдФрд░ рд▓рдбрд╝рд╛рдИ рдХреЗ рд▓рд┐рдП рддреИрдпрд╛рд░ рд╣реЛрддрд╛ рд╣реИред
рд╣рдо рдкрд╛рдпрдерди рдХреЛ рдкрд╛рд╕ рдХрд░рддреЗ рд╣реИрдВред рдЙрджрд╛рд╣рд░рдг рджрд┐рдЦрд╛рддрд╛ рд╣реИ рдХрд┐ рдХрд┐рд╕реА рдлрд╝рдВрдХреНрд╢рди рдХреЗ рд▓рд┐рдП рддрд░реНрдХ рдХреИрд╕реЗ рдкрд╛рд░рд┐рдд рдХрд░реЗрдВ, рдлрд╝рдВрдХреНрд╢рди рдХрд╛ рдкрд░рд┐рдгрд╛рдо рдкреНрд░рд╛рдкреНрдд рдХрд░реЗрдВ, рдФрд░ рд╡реИрд╢реНрд╡рд┐рдХ рдЪрд░ рдХреЗ рдореВрд▓реНрдпреЛрдВ рдХреЛ рдХреИрд╕реЗ рдкреНрд░рд╛рдкреНрдд рдХрд░реЗрдВ рдФрд░ рдХреИрд╕реЗ рдмрджрд▓реЗрдВред


main.py:


 #!/usr/bin/python3 #-*- coding: utf-8 -*- import ctypes #   test = ctypes.CDLL('./objs/libtest.so') ## #    ## # ,    int test.func_ret_int.restype = ctypes.c_int # ,     int test.func_ret_int.argtypes = [ctypes.c_int, ] # ,    double test.func_ret_double.restype = ctypes.c_double # ,     double test.func_ret_double.argtypes = [ctypes.c_double] # ,    char * test.func_ret_str.restype = ctypes.c_char_p # ,     char * test.func_ret_str.argtypes = [ctypes.POINTER(ctypes.c_char), ] # ,    char test.func_many_args.restype = ctypes.c_char # ,     int, double. char, short test.func_many_args.argtypes = [ctypes.c_int, ctypes.c_double, ctypes.c_char, ctypes.c_short] print('ret func_ret_int: ', test.func_ret_int(101)) print('ret func_ret_double: ', test.func_ret_double(12.123456789)) #      ,       . print('ret func_ret_str: ', test.func_ret_str('Hello!'.encode('utf-8')).decode("utf-8") ) print('ret func_many_args: ', test.func_many_args(15, 18.1617, 'X'.encode('utf-8'), 32000).decode("utf-8")) print() ## #    ## # ,    int a = ctypes.c_int.in_dll(test, "a") print('ret a: ', a.value) #   . a.value = 22 a = ctypes.c_int.in_dll(test, "a") print('ret a: ', a.value) # ,    double b = ctypes.c_double.in_dll(test, "b") print('ret b: ', b.value) # ,    char c = ctypes.c_char.in_dll(test, "c") print('ret c: ', c.value.decode("utf-8")) 

рд╕рднреА рд╕рдВрднрд╡ рдбреЗрдЯрд╛ рдкреНрд░рдХрд╛рд░ рдФрд░ рдЙрдирдХреЗ рдкрджрдирд╛рдо рдкрд╛рдпрдерди рдкреНрд░рд▓реЗрдЦрди рдореЗрдВ рдкрд╛рдП рдЬрд╛ рд╕рдХрддреЗ рд╣реИрдВред


рд╕рдВрд░рдЪрдирд╛рдУрдВ рдХреЗ рд╕рд╛рде рдХрд╛рдо рдХрд░реЗрдВ


рд╕реА - рдЯреЗрд╕реНрдЯ рдореЗрдВ рд╕рдВрд░рдЪрдирд╛ рдХреА рдШреЛрд╖рдгрд╛ред


 typedef struct test_st_s test_st_t; struct test_st_s { int val1; double val2; char val3; }; 

рд╣рдорд╛рд░реА рд╕рдВрд░рдЪрдирд╛ рдХреЗ рд╕рд╛рде рдХрд╛рдо рдХрд░рдиреЗ рдХреЗ рд▓рд┐рдП рдХрд╛рд░реНрдп:


 test_st_t * func_ret_struct(test_st_t *test_st) { if (test_st) { printf("C get test_st: val1 - %d, val2 - %f, val3 - %c\n", test_st->val1, test_st->val2, test_st->val3); } return test_st; } 

рдЕрдЬрдЧрд░:


 import sys import struct #    Python   C class test_st_t(ctypes.Structure): _fields_ = [('val1', ctypes.c_int), ('val2', ctypes.c_double), ('val3', ctypes.c_char)] # ,    test_st_t * test.func_ret_struct.restype = ctypes.POINTER(test_st_t) # ,     void * test.func_ret_struct.argtypes = [ctypes.c_void_p] #   test_st = test_st_t(19, 3.5, 'Z'.encode('utf-8')) # Python None == Null C ret = test.func_ret_struct(None) print('ret func_ret_struct: ', ret) #   None,      ret = test.func_ret_struct(ctypes.byref(test_st)) #    C print('ret val1 = {}\nret val2 = {}\nret val3 = {}'.format(ret.contents.val1, ret.contents.val2, ret.contents.val3.decode("utf-8"))) 

рд╕реА ++


рдпрд╣ рдпрд╣рд╛рдБ рдереЛрдбрд╝рд╛ рдЕрдзрд┐рдХ рдЬрдЯрд┐рд▓ рд╣реИ, рдХреНрдпреЛрдВрдХрд┐ ctypes рдХреЗрд╡рд▓ C рдлрд╝рдВрдХреНрд╢рди рдХреЗ рд╕рд╛рде рдХрд╛рдо рдХрд░ рд╕рдХрддрд╛ рд╣реИ ред рдпрд╣ рд╣рдорд╛рд░реЗ рд▓рд┐рдП рдХреЛрдИ рд╕рдорд╕реНрдпрд╛ рдирд╣реАрдВ рд╣реИ, рдмрд╕ C, C ++ рдХреЛрдб рдХреЛ рдмрд╛рдБрдз рджреЗрдЧрд╛ред


C ++ рд╡рд░реНрдЧ рд╡рд┐рдзрд┐рдпрд╛рдБ рдФрд░ C рдмрд╛рдЗрдВрдбрд┐рдВрдЧ:


 #include "test.hpp" /* *   */ std::string test::ret_str(std::string val) { std::cout << "get ret_str: " << val << std::endl; return val; } int test::ret_int(int val) { std::cout << "get ret_int: " << val << std::endl; return val; } double test::ret_double(double val) { std::cout << "get ret_double: " << val << std::endl; return val; } /* *  C    C++ */ //   test,     . test *test_new() { return new test(); } //   test. void test_del(test *test) { delete test; } /* *   . */ //    ret_str char *test_ret_str(test *test, char *val) { // char *  std::string std::string str = test->ret_str(std::string(val)); // std::string  char * char *ret = new char[str.length() + 1]; strcpy(ret, str.c_str()); return ret; } //    ret_int int test_ret_int(test *test, int val) { return test->ret_int(val); } //    ret_double double test_ret_double(test *test, double val) { return test->ret_double(val); } /* *   . */ //    a int test_get_a(test *test) { return test->a; } //    b double test_get_b(test *test) { return test->b; } //    c char test_get_c(test *test) { return test->c; } 

рд▓реЗрдХрд┐рди рд╡рд╣рд╛рдБ рдПрдХ рдЪреЗрддрд╛рд╡рдиреА рд╣реИ, рдмрд╛рдзреНрдпрдХрд╛рд░реА рдХреЛ рдмрд╛рд╣рд░реА рд╕реА рдХреЗ рд░реВрдк рдореЗрдВ рдШреЛрд╖рд┐рдд рдХрд┐рдпрд╛ рдЬрд╛рдирд╛ рдЪрд╛рд╣рд┐рдП ред рддрд╛рдХрд┐ ++ рдХрдВрдкрд╛рдЗрд▓рд░ рдмрд╛рдзреНрдпрдХрд╛рд░реА рдлрд╝рдВрдХреНрд╢рди рдирд╛рдореЛрдВ рдХреЛ рдЕрдзрд┐рднрд╛рд░ рди рдбрд╛рд▓реЗрдВред рдпрджрд┐ рд╡рд╣ рдХрд░рддрд╛ рд╣реИ, рддреЛ рд╣рдо ctypes рдХреЗ рдорд╛рдзреНрдпрдо рд╕реЗ рдЕрдкрдиреЗ рдХрд╛рд░реНрдпреЛрдВ рдХреЗ рд╕рд╛рде рдХрд╛рдо рдирд╣реАрдВ рдХрд░ рдкрд╛рдПрдВрдЧреЗред
test.hpp:


 #include <iostream> #include <string.h> class test { public: int a = 5; double b = 5.12345; char c = 'X'; std::string ret_str(std::string val); int ret_int(int val); double ret_double(double val); }; #ifdef __cplusplus extern "C" { #endif test *test_new(); void test_del(test *test); char *test_ret_str(test *test, char *val); int test_ret_int(test *test, int val); double test_ret_double(test *test, double val); int test_get_a(test *test); double test_get_b(test *test); char test_get_c(test *test); #ifdef __cplusplus } #endif 

рдХреИрд╕реЗ рд╕рдВрдХрд▓рд┐рдд рдХрд░реЗрдВ:
g ++ -fPIC -sared -o libtestpp.so test.cpp


рдЕрдЬрдЧрд░ рдЙрддрдирд╛ рд╣реА рдЖрд╕рд╛рди рд╣реИред


 #   testpp = ctypes.CDLL('./objs/libtestpp.so') # ,     testpp.test_new.restype = ctypes.c_void_p #   test test = testpp.test_new() ## #    ## # ,    char * testpp.test_ret_str.restype = ctypes.c_char_p # ,     void *  char * testpp.test_ret_str.argtypes = [ctypes.c_void_p, ctypes.c_char_p] # ,    int testpp.test_ret_int.restype = ctypes.c_int # ,     void *  int testpp.test_ret_int.argtypes = [ctypes.c_void_p, ctypes.c_int] # ,    double testpp.test_ret_double.restype = ctypes.c_double # ,     void *  double testpp.test_ret_double.argtypes = [ctypes.c_void_p, ctypes.c_double] print('  :') #   1-       print('ret test_ret_str: ', testpp.test_ret_str(test, 'Hello!'.encode('utf-8')).decode("utf-8")) print('ret test_ret_int: ', testpp.test_ret_int(test, 123)) print('ret test_ret_double: ', testpp.test_ret_double(test, 9.87654321)) ## #    ## # ,    int testpp.test_get_a.restype = ctypes.c_int # ,     void * testpp.test_get_a.argtypes = [ctypes.c_void_p] # ,    double testpp.test_get_b.restype = ctypes.c_double # ,     void * testpp.test_get_b.argtypes = [ctypes.c_void_p] # ,    char testpp.test_get_c.restype = ctypes.c_char # ,     void * testpp.test_get_c.argtypes = [ctypes.c_void_p] print('\n  :') print('ret test_get_a: ', testpp.test_get_a(test)) print('ret test_get_b: ', testpp.test_get_b(test)) print('ret test_get_c: ', testpp.test_get_c(test).decode("utf-8")) # ,     void * testpp.test_del.argtypes = [ctypes.c_void_p] #   testpp.test_del(test) 

Ctypes рдХреЗ рдкреЗрд╢реЗрд╡рд░реЛрдВ рдФрд░ рд╡рд┐рдкрдХреНрд╖реЛрдВ


рдкреЗрд╢реЗрд╡рд░реЛрдВ :


  • рдЖрдк рдХрд┐рд╕реА рднреА рдкрд╣рд▓реЗ рд╕реЗ рд╕рдВрдХрд▓рд┐рдд рд╕реА рд▓рд╛рдЗрдмреНрд░реЗрд░реА рдХреЛ рдХрдиреЗрдХреНрдЯ рдХрд░ рд╕рдХрддреЗ рд╣реИрдВ

рд╡рд┐рдкрдХреНрд╖ :


  • рдкрд╛рдпрдерди рдореЗрдВ, рдЖрдкрдХреЛ рдпрд╣ рд╡рд░реНрдгрди рдХрд░рдиреЗ рдХреА рдЖрд╡рд╢реНрдпрдХрддрд╛ рд╣реИ рдХрд┐ рд╕реА рдлрд╝рдВрдХреНрд╢рди рдХреНрдпрд╛ рд▓реМрдЯрд╛рддрд╛ рд╣реИ рдФрд░ рддрд░реНрдХ рдХреЗ рд░реВрдк рдореЗрдВ рд╕реНрд╡реАрдХрд╛рд░ рдХрд░рддрд╛ рд╣реИред

рдХреЛрдб рдиреЗ рд╕реНрдкрд╖реНрдЯ рд░реВрдк рд╕реЗ рдЯрд┐рдкреНрдкрдгреА рдХрд░рдиреЗ рдХреА рдХреЛрд╢рд┐рд╢ рдХреА, рдпрд╣рд╛рдБ рдХрдо рд▓рд┐рдЦрдиреЗ рдХреЗ рд▓рд┐рдП)


рдореБрдЭреЗ рдЙрдореНрдореАрдж рд╣реИ рдХрд┐ рдпрд╣ рдЙрдкрдпреЛрдЧреА рд╣реЛрдЧрд╛ред


рдзрдиреНрдпрд╡рд╛рдж


рдЙрд╕рдХреА рдорджрдж рдХреЗ рд▓рд┐рдП DollaR84 ред
рдорд┐рд▓реА рддреНрд░реБрдЯрд┐рдпреЛрдВ рдХреЗ рд▓рд┐рдП рдкрд╛рд▓рд┐рдЪ 239 ред


рд╕рдВрджрд░реНрдн


Source: https://habr.com/ru/post/hi466499/


All Articles