Things to be Aware Of露
TLS Support露
Support for Transport Layer Security (TLS) is increasingly important, particularly on mobile platforms where an application is typically a front end to a cloud-based server. As both Python and Qt implement different APIs that support TLS, a PyQt application has a choice as to which to use. This is particularly important when deploying an application as the support may have to be included with, or built into, the application itself.
Ideally the TLS implementation provided by the target would be used (e.g. CryptoAPI on Windows, Secure Transport on macOS and iOS). This would mean that security updates, including certificate updates, would be handled by the vendor of the target operating system and could be ignored by the application. Unfortunately there is no common TLS API. The resolution to this problem is the subject of PEP 543 but that has yet to be implemented.
Python uses OpenSSL as its TLS implementation. Python v3.7.4 and later use OpenSSL v1.1.1. Python v3.7.0 to v3.7.3 use OpenSSL v1.1.0. Earlier versions of Python use OpenSSL v1.0.2. On Windows and macOS the standard Python binary installers include copies of the corresponding OpenSSL libraries.
Qt has support for the native TLS implementation on macOS and iOS but on other platforms (except for Linux) a deployed application must include it鈥檚 own OpenSSL implementaion.
enums露
New in version 5.11.
PyQt (or rather SIP) wraps C/C++ enums using a dedicated Python type. Members of the enum are visible at the same scope as the enum itself.
Qt is making increasing use of C++11 scoped enums and support for them was
added to SIP v4.19.4. Scoped enums are implemented using the standard
enum.Enum
Python type. In this case members of the enum are only
visible within the scope of the enum.
The difference in visibility is unfortunate as it requires the Python programmer to be aware of the nature of the underlying C++ enum.
With SIP v4.19.9 members of traditional C/C++ enums are now also visible within the scope of the enum. It is strongly recommended that enum members are always referenced by specifying the scope of the enum. PyQt6 will not allow any other method of access.
Crashes On Exit露
When the Python interpreter leaves a scope (for example when it returns from a function) it will potentially garbage collect all objects local to that scope. The order in which it is done is, in effect, random. Theoretically this can cause problems because it may mean that the C++ destructors of any wrapped Qt instances are called in an order that Qt isn鈥檛 expecting and may result in a crash. However, in practice, this is only likely to be a problem when the application is terminating.
As a way of mitigating this possiblity PyQt5 ensures that the C++ destructors of any QObject instances owned by Python are invoked before the destructor of any QCoreApplication instance is invoked. Note however that the order in which the QObject destructors are invoked is still random.
Keyword Arguments露
PyQt5 supports the use of keyword arguments for optional arguments. Although the PyQt5 and Qt documentation may indicate that an argument has a particular name, you may find that PyQt5 actually uses a different name. This is because the name of an argument is not part of the Qt API and there is some inconsistency in the way that similar arguments are named. Different versions of Qt may use a different name for an argument which wouldn鈥檛 affect the C++ API but would break the Python API.
The docstrings that PyQt5 generates for all classes, functions and methods will contain the correct argument names. In a future version of PyQt5 the documentation will also be guaranteed to contain the correct argument names.
Python Strings, Qt Strings and Unicode露
Qt uses the QString
class to represent Unicode strings, and the
QByteArray
to represent byte arrays or strings. In Python v3 the
corresponding native object types are str
and bytes
. In Python v2 the
corresponding native object types are unicode
and str
.
PyQt5 does its best to automatically convert between objects of the various types. Explicit conversions can be easily made where necessary.
In some cases PyQt5 will not perform automatic conversions where it is necessary to distinguish between different overloaded methods.
For Python v3 the following conversions are done by default.
If Qt expects a
char *
(or aconst
version) then PyQt5 will accept astr
that contains only ASCII characters, abytes
, aQByteArray
, or a Python object that implements the buffer protocol.If Qt expects a
char
(or aconst
version) then PyQt5 will accept the same types as forchar *
and also require that a single character is provided.If Qt expects a
signed char *
or anunsigned char *
(or aconst
version) then PyQt5 will accept abytes
.If Qt expects a
signed char
or anunsigned char
(or aconst
version) then PyQt5 will accept abytes
of length 1.If Qt expects a
QString
then PyQt5 will accept astr
, abytes
that contains only ASCII characters, aQByteArray
orNone
.If Qt expects a
QByteArray
then PyQt5 will also accept abytes
.If Qt expects a
QByteArray
then PyQt5 will also accept astr
that contains only Latin-1 characters.
For Python v2 the following conversions are done by default.
If Qt expects a
char *
,signed char *
or anunsigned char *
(or aconst
version) then PyQt5 will accept aunicode
that contains only ASCII characters, astr
, aQByteArray
, or a Python object that implements the buffer protocol.If Qt expects a
char
,signed char
or anunsigned char
(or aconst
version) then PyQt5 will accept the same types as forchar *
,signed char *
andunsigned char *
and also require that a single character is provided.If Qt expects a
QString
then PyQt5 will accept aunicode
, astr
that contains only ASCII characters, aQByteArray
orNone
.If Qt expects a
QByteArray
then PyQt5 will accept astr
.If Qt expects a
QByteArray
then PyQt5 will accept aunicode
that contains only Latin-1 characters.
Note that the different behaviour between Python v2 and v3 is due to v3鈥檚 reduced support for the buffer protocol.
Historically QString
distinguishes between empty strings and null strings.
Current versions of Qt treat null strings as empty strings but there may be
other C++ code that PyQt5 applications call that maintains the distinction.
Consequently PyQt5 will convert None
to a null QString
. The reverse
conversion is not done and both a null and an empty QString
will be
converted to an empty (i.e. zero length) Python string.
Garbage Collection露
C++ does not garbage collect unreferenced class instances, whereas Python does. In the following C++ fragment both colours exist even though the first can no longer be referenced from within the program:
col = new QColor();
col = new QColor();
In the corresponding Python fragment, the first colour is destroyed when the
second is assigned to col
:
col = QColor()
col = QColor()
In Python, each colour must be assigned to different names. Typically this is done within class definitions, so the code fragment would be something like:
self.col1 = QColor()
self.col2 = QColor()
Sometimes a Qt class instance will maintain a pointer to another instance and will eventually call the destructor of that second instance. The most common example is that a QObject (and any of its sub-classes) keeps pointers to its children and will automatically call their destructors. In these cases, the corresponding Python object will also keep a reference to the corresponding child objects.
So, in the following Python fragment, the first
QLabel is not destroyed when the second is
assigned to lab
because the parent QWidget
still has a reference to it:
parent = QWidget()
lab = QLabel("First label", parent)
lab = QLabel("Second label", parent)
Multiple Inheritance露
It is not possible to define a new Python class that sub-classes from more than one Qt class. The exception is classes specifically intended to act as mixin classes such as those (like QQmlParserStatus) that implement Qt interfaces.
Access to Protected Member Functions露
When an instance of a C++ class is not created from Python it is not possible to access the protected member functions of that instance. Attempts to do so will raise a Python exception. Also, any Python methods corresponding to the instance鈥檚 virtual member functions will never be called.
None
and NULL
露
Throughout PyQt5, the None
value can be specified wherever NULL
is
acceptable to the underlying C++ code.
Equally, NULL
is converted to None
whenever it is returned by the
underlying C++ code.
Support for void *
露
PyQt5 (actually SIP) represents void *
values as objects of type
sip.voidptr. Such values are often used to pass the addresses of
external objects between different Python modules. To make this easier, a
Python integer (or anything that Python can convert to an integer) can be used
whenever a sip.voidptr is expected.
A sip.voidptr may be converted to a Python integer by using the
int()
builtin function.
A sip.voidptr may be converted to a Python string by using its
asstring()
method. The asstring()
method
takes an optional integer argument which is the length of the data in bytes.
A sip.voidptr may also be given a size (ie. the size of the block
of memory that is pointed to) by calling its setsize()
method. If it has a size then it is also able to support Python鈥檚 buffer
protocol and behaves like a Python memoryview
object so that the block of
memory can be treated as a mutable list of bytes. It also means that the
Python struct module can be used to unpack and pack binary data
structures in memory, memory mapped files or shared memory.