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

C++, Qt5, bind, 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++, переопределение виртуальных методов, создание и уничтожение объектов и т. д.

We recommend hosting TIMEWEB
We recommend hosting TIMEWEB
Stable hosting, on which the social network EVILEG is located. For projects on Django we recommend VDS hosting.
Support the author Donate

Comments

Only authorized users can post comments.
Please, Log in or Sign up
Donate

Hello, Dear Users of EVILEG!!!

If the site helped you, then support the development of the site financially, please.

You can do it by following ways:

Thank you, Evgenii Legotckoi

LP
Nov. 12, 2019, 8:22 a.m.
Lev Parhimovich

C++ - Test 006. Enumerations

  • Result:50points,
  • Rating points-4
LP
Nov. 12, 2019, 7:35 a.m.
Lev Parhimovich

C++ - Test 005. Structures and Classes

  • Result:66points,
  • Rating points-1
LP
Nov. 12, 2019, 7:26 a.m.
Lev Parhimovich

C ++ - Test 004. Pointers, Arrays and Loops

  • Result:50points,
  • Rating points-4
Last comments
b
Nov. 9, 2019, 8:28 a.m.
bastonc

спасибо ещё раз. огромное, за уделённое время
b
Nov. 9, 2019, 8:24 a.m.
bastonc

Спасибо Вам большое. Буду изучать.
Nov. 9, 2019, 5:58 a.m.
Evgenij Legotskoj

Добрый день. По первым двум вопросам вы найдёте ответ в этой статье - PyQt5 - Урок 008. Работа с QTableWidget (Обновление урока 006) Что касается последнего вопроса, то я вам…
Nov. 9, 2019, 2:50 a.m.
Evgenij Legotskoj

Как и обещал, вы можете посмотреть новую статью QML - Урок 037. Кастомизация кнопок в QML (Обновление урока 002) . Там же найдёте ссылку на Git репозиторий. Не забудьте поставить звёз…
b
Nov. 8, 2019, 7:40 a.m.
bastonc

Приветствую. Подскажите пожалуйста пару моментов. 1. Как сделать столбец не редактируемый, а остальные ячейки остаются редактируемыми 2. Как оталвливать события двойного клика для реда…
Now discuss on the forum
KZ
Nov. 13, 2019, 10:07 a.m.
Konstantin Znamenskii

Добрый день. Я хочу узнать, как наиболее грамотно обновлять Maintenance Tool у пользователя. Я периодически выкладываю обновления пакетов самой программы, но мне также нужно вносить изменения в …
Nov. 13, 2019, 9:33 a.m.
Pavel.K

Приложение трэкинг задач. Есть вложения. Добавляем вложение и отправляем его на сервер. Для синхронного вызова методов в синхронизации, используем QEventLoop при отправке вложений. В момент син…
Nov. 13, 2019, 6:41 a.m.
Ruslan Polupan

Ну ка кбы уже лет 10 как работает :-) Просто нужна отдельная прога для добавления таких записей, отдать клиентам чтобы мозг не парили....
Nov. 13, 2019, 3:13 a.m.
Evgenij Legotskoj

Добрый день. Думаю, что да. Выбранный стиль можно подгружать при запуске программы. Во всяком случае, есть такой пример на C++ - Controls Gallery . И там есть такой код #incl…
Nov. 12, 2019, 9:35 a.m.
Evgenij Legotskoj

Добрый день. Для корректной вставки ссылок на youtube вставляйте их через пустую строку, таким образом Первая ссылка Вторая ссылкаТретья ссылка
EVILEG
About
Services
© EVILEG 2015-2019
Recommend hosting TIMEWEB