是否可以从对象创建引用视图?

Is it possible to create a view of references from objects?

为了说明我的意思:我有三个对象:

Foo first, even, odd;

我想构建一个包含对这些对象的引用的视图,如下所示:first&、odd&、even&、odd& ... 最多 N 个。为了能够迭代它们:

for (const auto & obj: my_view) {
    // obj is equal to first&, odd&, even&, odd&... and so on
}

或将视图传递给函数:

template <typename V>
void parse_elements(const V & v) {
    // iterate over elements here
}

是否可以使用最新的 std::ranges 或 v3 范围库而不编写我自己的容器 class?

你把事情搞复杂了。只需提取函数即可:

void extractedFunction(Foo& foo);

void yourLogic(int N)
{
    Foo first, even, odd;
    extractedFunction(foo);
    for (int i = 0; i < (N - 1) / 2; ++i) {
        extractedFunction(odd);
        extractedFunction(even);
    }
    if ((N - 1) % 2 == 1) {
        extractedFunction(odd);
    }
}

首先,我建议从 Foo 对象的数组开始,而不是不同的变量。这不是必需的,因为您可以改用 spans,但这更直接一些:

Foo first[1] {};
Foo even_odd[2] {};

Ranges-v3 具有在这种情况下很有用的循环和连接视图。它们不在标准库中:

namespace views = ranges::views;
auto even_odd_cycle = even_odd | views::cycle;
auto first_even_odd = views::concat(first, even_odd_cycle);
for (Foo& f : first_even_odd | views::take(N)) {
    // ...
}

从计数范围开始; 0,1,2,3,4... iota 是此范围的(命名不当的)名称。

然后按[&](auto n)->decltype(auto){return n?n%2?odd:even:first;}.

变换

我想这就是我想要的。

延伸阅读:

https://en.cppreference.com/w/cpp/ranges/iota_view

https://en.cppreference.com/w/cpp/algorithm/ranges/transform

我认为你需要创建一个容器,其中 value_type 是 Foo 的指针,就像那个 vector ,然后你可以调用 views::indirect 以获取对原始对象的引用。 https://godbolt.org/z/9x84v4851

#include <range/v3/all.hpp>
#include <fmt/format.h>
#include <fmt/ranges.h>
namespace views =ranges::views;

int main()
{
  int first=0;
  int a =1, b=2;
  std::vector<int*> v{&a,&b};
  auto ra=v|views::cycle|views::take(8)|views::indirect;
  auto rf=views::single(&first)|views::indirect;
  auto res=views::concat(rf,ra);
  fmt::print("{}",res);

}