Table of Contents
Preface
Introduction
Generalize these methods
I find that almost no one uses it**=
Home Backend Development Python Tutorial Detailed explanation of enhanced arithmetic assignment '-=' operation

Detailed explanation of enhanced arithmetic assignment '-=' operation

Sep 11, 2020 pm 05:11 PM
python Assignment

Detailed explanation of enhanced arithmetic assignment '-=' operation

Related learning recommendations: python tutorial

Preface

This article is a series of articles on Python syntax sugar one. The latest source code can be found in the desugar project (github.com/brettcannon…

Introduction

Python has something called enhanced arithmetic assignment(augmented arithmetic assignment) . Maybe you are not familiar with this name. It actually means assignment while doing mathematical operations. For example, a -= b is the enhanced arithmetic assignment of subtraction.

Enhanced assignment was added in Python 2.0 version. (Annotation: Introduced in PEP-203)

Analysis-=

Because Python does not allow overwriting assignments, compared to other operations that have special/magic methods , the way it implements enhanced assignment may not be exactly the same as you imagine.

First of all, you must know that a -= b is semantically the same as a = a-b. But also realize that if you know in advance that you want to assign an object to a variable name, it may be more efficient than a - b blind operation.

For example, at least The advantage is that you can avoid creating a new object: if an object can be modified in place, then returning self is more efficient than reconstructing a new object.

Therefore, Python provides a __isub__() method. If it is defined on the left side of the assignment (usually called an lvalue), the value on the right side (usually called an rvalue) will be called. So for a -= b, it will try to call a. __isub__(b).

If the result of the call is NotImplemented, or there is no result at all, then Python will fall back to regular binary arithmetic operations: a - b. Article on binary operations, the translation is here)

In the end, no matter which method is used, the return value will be assigned to a.

The following is a simple pseudo code,a - = b is decomposed into:

# 实现 a -= b 的伪代码if hasattr(a, "__isub__"):
    _value = a.__isub__(b)    if _value is not NotImplemented:
        a = _value    else:
        a = a - b    del _value else:
     a = a - b复制代码
Copy after login

Generalize these methods

Since we already implemented binary arithmetic, it is not too complicated to generalize the enhanced arithmetic operation.

By passing in a binary arithmetic operation function, and doing some introspection (and handling possible TypeErrors), it can be neatly summarized as:

def _create_binary_inplace_op(binary_op: _BinaryOp) -> Callable[[Any, Any], Any]:

    binary_operation_name = binary_op.__name__[2:-2]
    method_name = f"__i{binary_operation_name}__"
    operator = f"{binary_op._operator}="

    def binary_inplace_op(lvalue: Any, rvalue: Any, /) -> Any:
        lvalue_type = type(lvalue)        try:
            method = debuiltins._mro_getattr(lvalue_type, method_name)        except AttributeError:            pass
        else:
            value = method(lvalue, rvalue)            if value is not NotImplemented:                return value        try:            return binary_op(lvalue, rvalue)        except TypeError as exc:            # If the TypeError is due to the binary arithmetic operator, suppress
            # it so we can raise the appropriate one for the agumented assignment.
            if exc._binary_op != binary_op._operator:                raise
        raise TypeError(            f"unsupported operand type(s) for {operator}: {lvalue_type!r} and {type(rvalue)!r}"
        )

    binary_inplace_op.__name__ = binary_inplace_op.__qualname__ = method_name
    binary_inplace_op.__doc__ = (        f"""Implement the augmented arithmetic assignment `a {operator} b`."""
    )    return binary_inplace_op复制代码
Copy after login

This makes the definition of -= support _create_binary_inplace_op(__ sub__) , and can infer other things: the function name, what __i*__ function is called, and which callable to call when a binary arithmetic operation goes wrong.

I find that almost no one uses it**=

While writing the code for this article, I encountered a strange test error of **=. Of all the tests that ensure that __pow__ is called appropriately, there is one that fails for the operator module in the Python standard library.

My code is usually fine, and if there are differences between the code and CPython's code, it usually means I'm doing something wrong.

However, no matter how carefully I troubleshoot the code, I can't pinpoint why my tests pass and the standard library fails.

I decided to take a closer look at what's going on under the hood of CPython. Starting with the disassembled bytecode:

>>> def test(): a **= b... >>> import dis>>> dis.dis(test)  1           0 LOAD_FAST                0 (a)              2 LOAD_GLOBAL              0 (b)              4 INPLACE_POWER              6 STORE_FAST               0 (a)              8 LOAD_CONST               0 (None)             10 RETURN_VALUE复制代码
Copy after login

Through it, I found INPLACE_POWER in the eval loop:

        case TARGET(INPLACE_POWER): {
            PyObject *exp = POP();
            PyObject *base = TOP();
            PyObject *res = PyNumber_InPlacePower(base, exp, Py_None);
            Py_DECREF(base);
            Py_DECREF(exp);
            SET_TOP(res);            if (res == NULL)                goto error;
            DISPATCH();
        }复制代码
Copy after login

Source: github.com/python/cpyt …

Then find PyNumber_InPlacePower():

PyObject *PyNumber_InPlacePower(PyObject *v, PyObject *w, PyObject *z){    if (v->ob_type->tp_as_number &&
        v->ob_type->tp_as_number->nb_inplace_power != NULL) {        return ternary_op(v, w, z, NB_SLOT(nb_inplace_power), "**=");
    }    else {        return ternary_op(v, w, z, NB_SLOT(nb_power), "**=");
    }
}复制代码
Copy after login

Source: github.com/python/cpyt…

Heaved a sigh of relief~The code shows if defined If __ipow__ is present, it will be called, but __pow__ will be called only if there is no __ipow__.

However, the correct approach should be: If there is a problem when calling __ipow__, NotImplemented is returned or there is no return at all, then __pow__ and __rpow__ should be called.

In other words, the above code unexpectedly skips the fallback semantics of a**b when __ipow__ is present!

Actually, this issue was partially discovered and a bug submitted about 11 months ago. I fixed the issue and documented it on python-dev.

As of now, it seems that this will be fixed in Python 3.10, we also need to add a notice about **= being buggy in the documentation for 3.8 and 3.9 (the problem may be early, but older The Python version is already in security-only maintenance mode, so the documentation will not change).

The fixed code will most likely not be ported because it is a semantic change, and it is difficult to tell if someone accidentally relied on the problematic semantics. But it took so long for this problem to be noticed, which suggests that **= is not widely used, otherwise the problem would have been discovered long ago.

If you want to know more about programming learning, please pay attention to the php training column!

The above is the detailed content of Detailed explanation of enhanced arithmetic assignment '-=' operation. For more information, please follow other related articles on the PHP Chinese website!

Statement of this Website
The content of this article is voluntarily contributed by netizens, and the copyright belongs to the original author. This site does not assume corresponding legal responsibility. If you find any content suspected of plagiarism or infringement, please contact admin@php.cn

Hot AI Tools

Undresser.AI Undress

Undresser.AI Undress

AI-powered app for creating realistic nude photos

AI Clothes Remover

AI Clothes Remover

Online AI tool for removing clothes from photos.

Undress AI Tool

Undress AI Tool

Undress images for free

Clothoff.io

Clothoff.io

AI clothes remover

AI Hentai Generator

AI Hentai Generator

Generate AI Hentai for free.

Hot Article

R.E.P.O. Energy Crystals Explained and What They Do (Yellow Crystal)
1 months ago By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. Best Graphic Settings
1 months ago By 尊渡假赌尊渡假赌尊渡假赌
Will R.E.P.O. Have Crossplay?
1 months ago By 尊渡假赌尊渡假赌尊渡假赌

Hot Tools

Notepad++7.3.1

Notepad++7.3.1

Easy-to-use and free code editor

SublimeText3 Chinese version

SublimeText3 Chinese version

Chinese version, very easy to use

Zend Studio 13.0.1

Zend Studio 13.0.1

Powerful PHP integrated development environment

Dreamweaver CS6

Dreamweaver CS6

Visual web development tools

SublimeText3 Mac version

SublimeText3 Mac version

God-level code editing software (SublimeText3)

PHP and Python: Code Examples and Comparison PHP and Python: Code Examples and Comparison Apr 15, 2025 am 12:07 AM

PHP and Python have their own advantages and disadvantages, and the choice depends on project needs and personal preferences. 1.PHP is suitable for rapid development and maintenance of large-scale web applications. 2. Python dominates the field of data science and machine learning.

How is the GPU support for PyTorch on CentOS How is the GPU support for PyTorch on CentOS Apr 14, 2025 pm 06:48 PM

Enable PyTorch GPU acceleration on CentOS system requires the installation of CUDA, cuDNN and GPU versions of PyTorch. The following steps will guide you through the process: CUDA and cuDNN installation determine CUDA version compatibility: Use the nvidia-smi command to view the CUDA version supported by your NVIDIA graphics card. For example, your MX450 graphics card may support CUDA11.1 or higher. Download and install CUDAToolkit: Visit the official website of NVIDIACUDAToolkit and download and install the corresponding version according to the highest CUDA version supported by your graphics card. Install cuDNN library:

Detailed explanation of docker principle Detailed explanation of docker principle Apr 14, 2025 pm 11:57 PM

Docker uses Linux kernel features to provide an efficient and isolated application running environment. Its working principle is as follows: 1. The mirror is used as a read-only template, which contains everything you need to run the application; 2. The Union File System (UnionFS) stacks multiple file systems, only storing the differences, saving space and speeding up; 3. The daemon manages the mirrors and containers, and the client uses them for interaction; 4. Namespaces and cgroups implement container isolation and resource limitations; 5. Multiple network modes support container interconnection. Only by understanding these core concepts can you better utilize Docker.

Python vs. JavaScript: Community, Libraries, and Resources Python vs. JavaScript: Community, Libraries, and Resources Apr 15, 2025 am 12:16 AM

Python and JavaScript have their own advantages and disadvantages in terms of community, libraries and resources. 1) The Python community is friendly and suitable for beginners, but the front-end development resources are not as rich as JavaScript. 2) Python is powerful in data science and machine learning libraries, while JavaScript is better in front-end development libraries and frameworks. 3) Both have rich learning resources, but Python is suitable for starting with official documents, while JavaScript is better with MDNWebDocs. The choice should be based on project needs and personal interests.

MiniOpen Centos compatibility MiniOpen Centos compatibility Apr 14, 2025 pm 05:45 PM

MinIO Object Storage: High-performance deployment under CentOS system MinIO is a high-performance, distributed object storage system developed based on the Go language, compatible with AmazonS3. It supports a variety of client languages, including Java, Python, JavaScript, and Go. This article will briefly introduce the installation and compatibility of MinIO on CentOS systems. CentOS version compatibility MinIO has been verified on multiple CentOS versions, including but not limited to: CentOS7.9: Provides a complete installation guide covering cluster configuration, environment preparation, configuration file settings, disk partitioning, and MinI

How to operate distributed training of PyTorch on CentOS How to operate distributed training of PyTorch on CentOS Apr 14, 2025 pm 06:36 PM

PyTorch distributed training on CentOS system requires the following steps: PyTorch installation: The premise is that Python and pip are installed in CentOS system. Depending on your CUDA version, get the appropriate installation command from the PyTorch official website. For CPU-only training, you can use the following command: pipinstalltorchtorchvisiontorchaudio If you need GPU support, make sure that the corresponding version of CUDA and cuDNN are installed and use the corresponding PyTorch version for installation. Distributed environment configuration: Distributed training usually requires multiple machines or single-machine multiple GPUs. Place

How to choose the PyTorch version on CentOS How to choose the PyTorch version on CentOS Apr 14, 2025 pm 06:51 PM

When installing PyTorch on CentOS system, you need to carefully select the appropriate version and consider the following key factors: 1. System environment compatibility: Operating system: It is recommended to use CentOS7 or higher. CUDA and cuDNN:PyTorch version and CUDA version are closely related. For example, PyTorch1.9.0 requires CUDA11.1, while PyTorch2.0.1 requires CUDA11.3. The cuDNN version must also match the CUDA version. Before selecting the PyTorch version, be sure to confirm that compatible CUDA and cuDNN versions have been installed. Python version: PyTorch official branch

How to install nginx in centos How to install nginx in centos Apr 14, 2025 pm 08:06 PM

CentOS Installing Nginx requires following the following steps: Installing dependencies such as development tools, pcre-devel, and openssl-devel. Download the Nginx source code package, unzip it and compile and install it, and specify the installation path as /usr/local/nginx. Create Nginx users and user groups and set permissions. Modify the configuration file nginx.conf, and configure the listening port and domain name/IP address. Start the Nginx service. Common errors need to be paid attention to, such as dependency issues, port conflicts, and configuration file errors. Performance optimization needs to be adjusted according to the specific situation, such as turning on cache and adjusting the number of worker processes.

See all articles