Evgenii Legotckoi
Evgenii Legotckoi18 июля 2018 г. 3:14

Напишите собственные привязки к Python

Сегодня мы взглянем, как вы можете создавать привязки для собственного проекта.

В Qt Company рады сообщить, что Qt для Python будет также включать в себя Shiboken  - ваш основной инструмент создания привязки.

Прочтите материал ниже, и вы получите представление о том, как создавать привязки Python для простой библиотеки C ++.

Будем надеяться, это способствует тому, что вы сделаете тоже самое с собственными пользовательскими библиотеками.

Как и в любом проекте Qt мы рады представить статьи по Shiboken, тем самым улучшая его понимание для всех.

Пример библиотеки

Основными целями этого поста  будет использование нами немного бессмысленной пользовательской библиотеки под названием Universe. Она предусматривает два класса: Icecream (мороженое) и Truck (грузовик).


Icecreams характеризуется вкусом, а Truck служит, как средство доставки Icecream для детей в окрестности. Довольно просто.

Мы хотели бы использовать эти классы внутри Python. Вариант использования – добавление дополнительных вкусов мороженного или проверка успешности доставки мороженного. Говоря простыми словами, мы хотим обеспечить привязки Python для Icecream и Truck, чтобы мы могли использовать их в собственном скрипте Python.

Мы пропустим некоторую часть для краткости, но вы можете посмотреть в репозитории полный исходный код. Ссылка на репозиторий pyside-setup/examples/samplebinding .

C++ библиотека

Во-первых, давайте взглянем на заголовочный файл Icecream

class Icecream
{
public:
    Icecream(const std::string &flavor);
    virtual Icecream *clone();
    virtual ~Icecream();
    virtual const std::string getFlavor();

private:
    std::string m_flavor;
};

и заголовочный файл Truck

class Truck {
public:
    Truck(bool leaveOnDestruction = false);
    Truck(const Truck &other);
    Truck& operator=(const Truck &other);
    ~Truck();

    void addIcecreamFlavor(Icecream *icecream);
    void printAvailableFlavors() const;

    bool deliver() const;
    void arrive() const;
    void leave() const;

    void setLeaveOnDestruction(bool value);
    void setArrivalMessage(const std::string &message);

private:
    void clearFlavors();

    bool m_leaveOnDestruction = false;
    std::string m_arrivalMessage = "A new icecream truck has arrived!\n";
    std::vector m_flavors;
};

Большинство API должны быть достаточно просты для понимания, но мы подведем итоги важнейших частей:

  • Icecream является полиморфным типом и предназначается для переопределения
  • getFlavor() вернет вкус в зависимости от фактического производного типа
  • Truck сохраняет вектор принадлежащих ему объектов Icecream, которые могут быть добавлены через addIcecreamFlavor()
  • Сообщение прибытия «грузовика» можно настроить с помощью setArrivalMessage()
  • deliver() скажет нам, была ли доставка «мороженого» успешной или нет

Система типов Shiboken

Чтобы информировать shiboken об API, мы нуждаемся в привязках, для этого мы делаем файл-заголовок, который включает в себя интересующие нас типы:

#ifndef BINDINGS_H
#define BINDINGS_H
#include "icecream.h"
#include "truck.h"
#endif // BINDINGS_H

Кроме того, shiboken также требуется XML typesystem файл, который определяет отношение между типами C++ и Python:

<?xml version="1.0"?>
<typesystem package="Universe">
    <primitive-type name="bool"/>
    <primitive-type name="std::string"/>
    <object-type name="Icecream">
        <modify-function signature="clone()">
            <modify-argument index="0">
                <define-ownership owner="c++"/>
            </modify-argument>
        </modify-function>
    </object-type>
    <value-type name="Truck">
        <modify-function signature="addIcecreamFlavor(Icecream*)">
            <modify-argument index="1">
                <define-ownership owner="c++"/>
            </modify-argument>
        </modify-function>
    </value-type>
</typesystem>

Первая важная вещь на которую нужно обратить внимание – то, что мы объявляем "bool" и "std::string", как примитивные типы. Некоторые из методов C++ используют их как параметр / возвращаемые типы данных, и поэтому shiboken должен знать о них. Затем он может генерировать соответствующий код преобразования между C ++ и Python.

Большинство примитивных типов C ++ обрабатываются shiboken, не требуя дополнительного кода. Затем мы объявляем два вышеупомянутых класса. Один из них как “object-type”(объект-тип), а другой как “value-type” (значение-объект).

Основное отличие состоит в том, что “object-type” передаются в сгенерированном коде, как указатели, тогда как “value-type” копируются (семантика значений).

Задав имена классов в файле системы типов, shiboken автоматически попытается создать привязки для всех методов, объявленных в классах, поэтому нет необходимости указывать все имена методов вручную...

Если вы не хотите каким-либо образом изменить функцию, то это приводит нас к следующей теме «Правилам владения».

Shiboken не может волшебным образом узнать, кто отвечает за освобождение объектов C++, размещенных в коде Python’а.

Может быть множество случаев: Python должен освободить память C ++, когда количество ссылок на объект Python становится равным нулю или Python никогда не должен удалять объект C++, допуская, что он будет удален в определенный момент внутри библиотеки C ++. Или, возможно, это родительский элемент для другого объекта (например, как в QWidgets). В нашем случае метод clone() вызывается только внутри библиотеки C++, и мы предполагаем, что код C++ позаботится о освобождении памяти клонированного объекта.

Что касается addIcecreamFlavor(), мы знаем, что Truck располагается в объекте Icecream, и будет удален сразу же после уничтожения Truck. Таким образом, вновь, право владения устанавливается на “c++.”

Если бы мы не указали правила владения, то в этом случае объекты C++ будут удалены, когда соответствующие имена Python выйдут за пределы области видимости.

Сборка

Чтобы собрать пользовательскую библиотеку Universe и затем сгенерировать для нее привязки, мы предоставляем хорошо документированный, в основном общий файл CMakeLists.txt, который вы можете повторно использовать для своих собственных библиотек.

В основном это сводится к вызову «cmake.», чтобы настроить проект, а затем построить с помощью цепочки инструментов по вашему выбору (мы рекомендуем генератор «(N) Makefiles»).

В результате создания проекта вы получаете две общие библиотеки: libuniverse. (so/dylib/dll) и Universe. (so/pyd).

Первая – это пользовательская библиотека C++ и последняя – это модуль Python, который может быть импортирован из скрипта (документ, тестовый драйвер) Python.

Конечно, есть также промежуточные файлы, созданные shiboken (файлы .h / .cpp, созданные для создания привязок Python). Не волнуйтесь о них, если вам не нужно устранять неполадки или по какой-либо причине не удается скомпилировать или не ведет себя так, как должно. Тогда вы можете отправить Qt Company отчет об ошибке!

И, наконец, мы добираемся до части Python.

Использование Python модуля

В следующем маленьком скрипте мы будем использовать модуль Universe, наследуемся от Icecream, внедрим виртуальные методы, инстанцируем объекты и многое другое

from Universe import Icecream, Truck

class VanillaChocolateIcecream(Icecream):
    def __init__(self, flavor=""):
        super(VanillaChocolateIcecream, self).__init__(flavor)

    def clone(self):
        return VanillaChocolateIcecream(self.getFlavor())

    def getFlavor(self):
        return "vanilla sprinked with chocolate"

class VanillaChocolateCherryIcecream(VanillaChocolateIcecream):
    def __init__(self, flavor=""):
        super(VanillaChocolateIcecream, self).__init__(flavor)

    def clone(self):
        return VanillaChocolateCherryIcecream(self.getFlavor())

    def getFlavor(self):
        base_flavor = super(VanillaChocolateCherryIcecream, self).getFlavor()
        return base_flavor + " and a cherry"

if __name__ == '__main__':
    leave_on_destruction = True
    truck = Truck(leave_on_destruction)

    flavors = ["vanilla", "chocolate", "strawberry"]
    for f in flavors:
        icecream = Icecream(f)
        truck.addIcecreamFlavor(icecream)

    truck.addIcecreamFlavor(VanillaChocolateIcecream())
    truck.addIcecreamFlavor(VanillaChocolateCherryIcecream())

    truck.arrive()
    truck.printAvailableFlavors()
    result = truck.deliver()

    if result:
        print("All the kids got some icecream!")
    else:
        print("Aww, someone didn't get the flavor they wanted...")

    if not result:
        special_truck = Truck(truck)
        del truck

        print("")
        special_truck.setArrivalMessage("A new SPECIAL icecream truck has arrived!\n")
        special_truck.arrive()
        special_truck.addIcecreamFlavor(Icecream("SPECIAL *magical* icecream"))
        special_truck.printAvailableFlavors()
        special_truck.deliver()
        print("Now everyone got the flavor they wanted!")
        special_truck.leave()

После импортирования классов из нашего модуля, мы создаем два производных (вторичных) типа Icecream, которым настроили “вкусы”.

Затем мы создаем truck (грузовик), добавляем к нему некоторые обычные варианты (разновидности) Icecreams (мороженого) и два специальных.

Мы стараемся отправить (доставить) Icecream (мороженое).

Если же доставка не удалась, мы создаем новый truck (грузовик) со старыми вариантами, скопированными, и новый волшебный вариант, который, несомненно, удовлетворит всех клиентов.

В приведенном выше скрипте кратко показано использование выводов из типов C++, переопределение виртуальных методов, создание и уничтожение объектов и т. д.

Рекомендуем хостинг TIMEWEB
Рекомендуем хостинг TIMEWEB
Стабильный хостинг, на котором располагается социальная сеть EVILEG. Для проектов на Django рекомендуем VDS хостинг.

Вам это нравится? Поделитесь в социальных сетях!

Комментарии

Только авторизованные пользователи могут публиковать комментарии.
Пожалуйста, авторизуйтесь или зарегистрируйтесь
Дмитрий

C++ - Тест 004. Указатели, Массивы и Циклы

  • Результат:60баллов,
  • Очки рейтинга-1
Дмитрий

C++ - Тест 003. Условия и циклы

  • Результат:92баллов,
  • Очки рейтинга8
d
  • dsfs
  • 26 апреля 2024 г. 4:56

C++ - Тест 004. Указатели, Массивы и Циклы

  • Результат:80баллов,
  • Очки рейтинга4
Последние комментарии
k
kmssr8 февраля 2024 г. 18:43
Qt Linux - Урок 001. Автозапуск Qt приложения под Linux как сделать автозапуск для флэтпака, который не даёт создавать файлы в ~/.config - вот это вопрос ))
АК
Анатолий Кононенко5 февраля 2024 г. 1:50
Qt WinAPI - Урок 007. Работаем с ICMP Ping в Qt Без строки #include <QRegularExpressionValidator> в заголовочном файле не работает валидатор.
EVA
EVA25 декабря 2023 г. 10:30
Boost - статическая линковка в CMake проекте под Windows Ошибка LNK1104 часто возникает, когда компоновщик не может найти или открыть файл библиотеки. В вашем случае, это файл libboost_locale-vc142-mt-gd-x64-1_74.lib из библиотеки Boost для C+…
J
JonnyJo25 декабря 2023 г. 8:38
Boost - статическая линковка в CMake проекте под Windows Сделал всё по-как у вас, но выдаёт ошибку [build] LINK : fatal error LNK1104: не удается открыть файл "libboost_locale-vc142-mt-gd-x64-1_74.lib" Хоть убей, не могу понять в чём дел…
G
Gvozdik18 декабря 2023 г. 21:01
Qt/C++ - Урок 056. Подключение библиотеки Boost в Qt для компиляторов MinGW и MSVC Для решения твой проблемы добавь в файл .pro строчку "LIBS += -lws2_32" она решит проблему , лично мне помогло.
Сейчас обсуждают на форуме
G
George137 мая 2024 г. 0:27
добавить qlineseries в функции в функции: "GPlotter::addSeries(QString title, QVector &arr)" я вызываю метод setChart(...), я в конструктор передал адрес на QChartView элемент
BlinCT
BlinCT5 мая 2024 г. 5:46
Написать свой GraphsView Всем привет. В Qt есть давольно старый обьект дял работы с графиками ChartsView и есть в 6.7 новый но очень сырой и со слабым функционалом GraphsView. По этой причине я хочу написать х…
PS
Peter Son3 мая 2024 г. 17:57
Best Indian Food Restaurant In Cincinnati OH Ready to embark on a gastronomic journey like no other? Join us at App india restaurant and discover why we're renowned as the Best Indian Food Restaurant In Cincinnati OH . Whether y…
Evgenii Legotckoi
Evgenii Legotckoi2 мая 2024 г. 14:07
Мобильное приложение на C++Qt и бэкенд к нему на Django Rest Framework Добрый день. По моему мнению - да, но то, что будет касаться вызовов к функционалу Андроида, может создать огромные трудности.
IscanderChe
IscanderChe30 апреля 2024 г. 4:22
Во Flask рендер шаблона не передаётся в браузер Доброе утро! Имеется вот такой шаблон: <!doctype html><html> <head> <title>{{ title }}</title> <link rel="stylesheet" href="{{ url_…

Следите за нами в социальных сетях