Section 23.3.7 Class vector<bool>
[vector.bool], paragraph 1 states:
template <class Allocator> class vector<bool, Allocator> {
public:
// types:
typedef bool const_reference;
...
However this program fails to compile when using libc++:
#include <vector>
#include <type_traits>
int
main()
{
static_assert(std::is_same<std::vector<bool>::const_reference, bool>{}, "?");
}
Furthermore I note that the C++ standard has been consistent in this specification all the way back to C++98. And I further note that libc++ has consistently not followed this specification since the first introduction of libc++.
What is the motivation for this non-conformance?
The motivation for this extension, which is detectable by a conforming program, and thus non-conforming, is to make
vector<bool>
behave more likevector<char>
with respect to references (const and otherwise).Introduction
Since 1998,
vector<bool>
has been derided as "not quite a container." LWG 96, one of the very first LWG issues, launched the debate. Today, 17 years later,vector<bool>
remains largely unchanged.This paper goes into some specific examples on how the behavior of
vector<bool>
differs from every other instantiation ofvector
, thus hurting generic code. However the same paper discusses at length the very nice performance propertiesvector<bool>
can have if properly implemented.Summary:
vector<bool>
isn't a bad container. It is actually quite useful. It just has a bad name.Back to
const_reference
As introduced above, and detailed here, what is bad about
vector<bool>
is that it behaves differently in generic code than othervector
instantiations. Here is a concrete example:The standard specification says that the assert marked
// Fires!
will trigger, but only whentest
is run with avector<bool>
. When run with avector<char>
(or anyvector
besidesbool
when an appropriate non-defaultT
is assigned), the test passes.The libc++ implementation sought to minimize the negative effects of having
vector<bool>
behave differently in generic code. One thing it did to achieve this is to makevector<T>::const_reference
a proxy-reference, just like the specifiedvector<T>::reference
, except that you can't assign through it. That is, on libc++,vector<T>::const_reference
is essentially a pointer to the bit inside thevector
, instead of a copy of that bit.On libc++ the above
test
passes for bothvector<char>
andvector<bool>
.At what cost?
The downside is that this extension is detectible, as shown in the question. However, very few programs actually care about the exact type of this alias, and more programs care about the behavior.
To give the libc++ client better behavior in generic code, and perhaps after sufficient field testing, propose this extension to a future C++ standard for the betterment of the entire C++ industry.
Such a proposal might come in the form of a new container (e.g.
bit_vector
) that has much the same API as today'svector<bool>
, but with a few upgrades such as theconst_reference
discussed here. Followed by deprecation (and eventual removal) of thevector<bool>
specialization.bitset
could also use a little upgrading in this department, e.g. addconst_reference
, and a set of iterators.That is, in hindsight
bitset
is tovector<bool>
(which should be renamed tobit_vector
-- or whatever), asarray
is tovector
. And the analogy ought to hold true whether or not we are talking aboutbool
as thevalue_type
ofvector
andarray
.There are multiple examples of C++11 and C++14 features that started out as extensions in libc++. This is how standards evolve. Actual demonstrated positive field experience carries strong influence. The standards folk are a conservative bunch when it comes to changing existing specifications (as they should be). Guessing, even when you are sure you are guessing correctly, is a risky strategy for evolving an internationally recognized standard.