Why allow `propagate_on_container_swap == false` in Allocators, when it might cause undefined behaviour?

后端 未结 2 1037
庸人自扰
庸人自扰 2021-02-07 18:29

Note: Originally asked by Matt Mcnabb as a comment on Why can swapping standard library containers be problematic in C++11 (involving allocators)?.

2条回答
  •  一个人的身影
    2021-02-07 19:00

    I can think of a few real-life scenarios where the construct allowed by the Standard both makes sense, and is required, however; I'll first try to answer this question from a broader perspective, not involving any specific problem.


    THE EXPLANATION

    Allocators are this magical things responsible for allocating, constructing, destructing, and deallocating memory and entities. Since C++11 when stateful allocators came into play an allocator can do much more than previously, but it all boils down to the previously mentioned four operations.

    Allocators have loads of requirements, one of them being that a1 == a2 (where a1 and a2 are allocators of the same type) must yield true only if memory allocated by one can be deallocated by the other [1].

    The above requirement of operator== means that two allocators comparing equal can do things differently, as long as they still have a mutual understanding of how memory is allocated.

    The above is why the Standard allows propagate_on_container_* to be equal to std::false_type; we might want to change the contents of two containers which allocators have the same deallocation behavior, but leave the other behavior (not related to basic memory management) behind.


    [1] as stated in [allocator.requirements]p2 (table 28)


    THE (SILLY) STORY

    Imagine that we have an Allocator named Watericator, it gathers water upon requested allocation, and hands it to the requested container.

    Watericator is a stateful Allocator, and upon constructing our instance we can choose two modes;

    1. employ Eric, who fetches water down at the fresh water spring, while also measures (and reports) water level and purity.

    2. employ Adam, who uses the tap out in the backyard and doesn't care anything about logging. Adam is a lot faster than Eric.


    No matter where the water comes from we always dispose of it in the same way; by watering our plants. Even if we have one instance where Eric is supplying us water (memory), and another where Adam is using the tap, both Watericators compare equal as far as operator== is concerned.

    Allocations done by one can be deallocated by the other.


    The above might be a silly similie, but imagine we have an allocator which does logging upon every allocation, and we uses this on a container somewhere in our code that interests us; we later want to move the elements out from this container into another one.. but we are no longer interested in all that logging.

    Without stateful allocators, and the option to turn propagate_on_container_* off, we would be forced to either 1) copy every element involved 2) be stuck with that (no longer required) logging.

提交回复
热议问题