+2

Using C code in Python

Ta thường sử dụng C/C++ cho những công việc yêu cầu tốc độ xử lý cũng như hiệu năng cao. Vậy làm thế nào để ta có thể gọi các hàm được cung cấp trong các thư viện C từ Python? Bài viết này trả lời câu hỏi trên.

Python cung cấp module ctypes để ta có thể giao tiếp với các thư viện ngoài của C.

Load thư viện động

Để load một thư viện động, ta sử dụng hàm cdll.LoadLibrary() từ module ctypes. Đoạn code dưới đây sẽ import bộ thư viện chuẩn glibc của C trong GNU/Linux:

import ctypes as ct

libc = ct.cdll.LoadLibrary('libc.so.6')

# Or we can use CDLL
libc = ct.CDLL('libc.so.6')

Gọi hàm

Sau khi đã load được thư viện, ta có thể gọi hàm C như những hàm Python bình thường. Dưới đây ta gọi hàm printftime của C, lưu ý rằng NULL pointer trong C tương ứng với None trong Python:

libc.printf(b"Hello world! %d %s\n", 5, b"whoami")
# Hello world! 5 whomai

print(libc.time(None))
# 1150945864

None, integers, bytes objects và (unicode) strings là các kiểu dữ liệu căn bản của Python và có thể được truyền trực tiếp như là đối số khi gọi hàm C.

None tương ứng với NULL pointer, bytes objectsstrings được sử dụng như là con trỏ tới vùng nhớ chứa dữ liệu của nó (char * hoặc wchar_t *). Trong ví dụ trên, ta sử dụng bytes object (b"Hello world! %d %s\n") như là con trỏ char * khi gọi hàm printf(). Kiểu integers của Python sẽ được sử dụng như int của C.

Dưới đây ta sẽ tìm hiểu rõ hơn về các kiểu dữ liệu mà ctypes hỗ trợ.

Các kiểu dữ liệu cơ bản

ctypes định nghĩa một số kiểu dữ liệu tương ứng với các kiểu dữ liệu cơ bản của C:

Capture.PNG

Tất cả các kiểu này có thể được khởi tạo như sau:

>>> c_int()
c_long(0)
>>> c_wchar_p("Hello, World")
c_wchar_p('Hello, World')
>>> c_ushort(-3)
c_ushort(65533)

Và có thể thày đổi gía trị của nó:

>>> i = c_int(42)
>>> print(i)
c_long(42)
>>> print(i.value)
42
>>> i.value = -99
>>> print(i.value)
-99

Truyền con trỏ trong Python

Nhiều hàm trong C yêu cầu đối số truyền vào là một con trỏ, vì vậy ctypes cung cấp cho cho chúng ta hàm byref(). Gỉa sử ta có hàm hoán đổi gía trị của 2 số nguyên trong thư viện /home/vuong/swap.so:

void swap(int *a, int *b);

Làm thế nào để ta có thể gọi hàm đấy trong Python?

from ctypes import *

lib = cdll.LoadLibrary('/home/vuong/swap.so')

a = c_int(5)
b = c_int(13)
lib.swap(byref(a), byref(b))

Khai báo Structs và Unions

Để khai báo structuresunions, ta kế thừa 2 class tương ứng trong ctypes là StructureUnion. Mỗi subclass phải định nghĩa một thuộc tính _fields_, là một list các 2-tuples (gồm 2 trường nametype).

Gỉa sử ta có khai báo struct và union trong C:


struct UserInfo {
    char name[50];
    int age;
    float height;
};

union UTag {
    int ival;
    float fval;
    char *sval;
};

Ta có đoạn code tương ứng trong Python, sử dụng class StructureUnion:

from ctypes import *

class UserInfo(Structure):
    _fields_ = [
        ('name', c_char * 50), # array of 50 characters
        ('age', c_int),
        ('height', c_float)
    ]

class UTag(Union):
    _fields_ = [
        ('ival', c_int),
        ('fval', c_float),
        ('sval', c_char_p)
    ]

Đoạn code trên cũng minh họa khai báo mảng trong python.

Happy coding!


All rights reserved

Viblo
Hãy đăng ký một tài khoản Viblo để nhận được nhiều bài viết thú vị hơn.
Đăng kí