是否可以从对象创建引用视图?
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
对象的数组开始,而不是不同的变量。这不是必需的,因为您可以改用 span
s,但这更直接一些:
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);
}
为了说明我的意思:我有三个对象:
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
对象的数组开始,而不是不同的变量。这不是必需的,因为您可以改用 span
s,但这更直接一些:
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
#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);
}