https://gcc.gnu.org/bugzilla/show_bug.cgi?id=81482

            Bug ID: 81482
           Summary: by-value lambda capture in remove_if
           Product: gcc
           Version: unknown
            Status: UNCONFIRMED
          Severity: normal
          Priority: P3
         Component: libstdc++
          Assignee: unassigned at gcc dot gnu.org
          Reporter: fabian-gcc at kloetzl dot info
  Target Milestone: ---

Hi all,

There is a little quirks with `remove_if` and lambdas that mutably capture by
value. As a contrived example let's print all the odd numbers from a list and
also count how many elements were removed already.

----
#include <vector>
#include <algorithm>
#include <cstdio>

int main(int argc, char *argv[])
{
        auto numbers = std::vector<int>{0,1,2,3,4,5,6,7,8,9};

        auto count = 0;
        auto odd = [=] (int n) mutable {
                if (n & 1) {
                        printf("%i %i\n", n, count);
                        count++;
                        return true;
                }
                return false;
        };

        std::remove_if(std::begin(numbers), std::end(numbers), odd);
        return 0;
}
----

Note that the use of `printf` is to avoid a libc++ bug [1].

The expected output is as follows.
----
1 0
3 1
5 2
7 3
9 4
----

However, with GCC 7.1.1 and libstdc++ 3.3.6 (current arch) the output is
different.

----
1 0
3 0
5 1
7 2
9 3
----

Clang with libc++ produces the desired output. As far as I can tell, both
behaviours are correct, according to the standard. But I guess the second is
unexpected.

A possible fix would be to pass the lambda by reference to `find_if`, instead
of by-value which makes an implicit copy [2]. That is how libc++ does it [3].

Some other standard algorithms might also be affected by this, didn't check. ☺

Best,
Fabian


[1]: https://bugs.debian.org/cgi-bin/bugreport.cgi?bug=808086
[2]:
https://github.com/gcc-mirror/gcc/blob/master/libstdc%2B%2B-v3/include/bits/stl_algo.h#L863
[3]:
https://github.com/llvm-mirror/libcxx/blob/master/include/algorithm#L2218-L2219

Reply via email to