c++23中的新功能之十六std::forward_like

文章讨论了C++中如何处理在模板函数中,特别是涉及类型内部成员如指针、容器和自定义类的完美转发问题,引入了std::forward_like并与其与DeducingThis共用。作者通过实例展示了std::forward_like在处理不同类型参数和成员访问中的作用。

一、介绍

前面说过,c++标准其实是分成两块推进的。一块是语言标准,另外一块是应用库标准。线程同步还麻烦呢,别说两个组的大佬,不同步的现象肯定会出现。在老的标准里还比较少,
在c++11以后经常发现,后续版本会对前面的版本打补丁,其实好多就是因为这种情况。
在前面分析过很多通过完美转发来实现的例子,比如才分析过的单例。在c++11以后,使用完美转发加上右值引用几乎可以用来常见的参数处理和转发控制(当然还是有一些是有问题的)。
但是这里会有一个应用场景出现(总有但是),如果想转发类型内部成员怎么办?类似于下面:

struct Data
{
  int d;
};
void getData(int d){}
template <typename T>
void send(T && t)
{
  getData(std::forward<T>(t).d);
}

代码跑起来很正常。但这个代码可不可以扩展呢?比如在Data内部有指针,有自定义类,有STL中的容器?可以试一下,用std::vector或者自己写一个类。程序跑起来仍然没有什么问题,但这就没问题了吗?

二、std::forward_like

在上一节提出了问题,首先回答这个问题要先回到完美转发的目的来。std::forward的目的是在模板编程时能够保持原参数的类型和值类型状态(左或右值等),从而准确的传递参数。这其实在c++非泛型编程中可以窥探出它的效果。举一个简单的例子,有一个函数接收int参数,但传入short也是没有问题的。原因是c++的隐式转换,这种转换很常见,但风险也非常大。最典型的就是有符号类型和无符号类型的转变时的溢出问题。它在非泛型编程中,还是比较容易发现和查找的,但是在泛型编程,也就是模板中,很难发现。同时,在后期的类型萃取中也会导致异常。等等还有其它一些问题,都是完美转发被广泛应用的一个原因。
所以,这时回到原来的问题上,对于普通(基础)类型或者自定义的一些普通类对象,做为右值仍然会被传递成为右值,这个不会有什么问题。但对于指针和容器或者自定义的一些特殊情况的类型,就有问题了。
std::forwardstd::vector中,标准库对[]有两个重载即只区分了常量和非常量。也就是说vector::[]const返回一个左值常量,可move(t)(也就是完美转发)后,move(t).vec[id]仍然是一个左值。它不是一个右值,也就是完美转发的过程中虽然把const传递了下去,但是左右值丢失了。
那么利用forward(t).容器内容,这种情况,c限定符保留了下来,但左右值失去了,那么使用std::forward的目的也失去了。而指针更甚一层,它的处理只是被传回一个非常量的左值(除非原来指针指向就是一个常量)。这样有没有完美转发,意义已经不存在了。
在自定义类中,如果稍微复杂一些,一定可能包含上述的情况,那么,完美转发的意义就大大失色了。c++的大佬们当然不会坐视这个问题,于是提出了std::forward_like:

getData(std::forward_like<T>(t.d));
//如果含有指针:
getData(std::forward_like<T>(*t.ptr));

它的形式和std::forward有一些不同,可以理解成直接完美转发成员了。

三、和Deducing This共用

在前面分析过Deducing This,std::forward_like可以和这个属性共用,用来转发自己的成员:

template<typename T>
struct Data
{
    T* value;
    template<typename Owner>
    decltype(auto) operator*(this Owner&& owner)
    { return std::forward_like<Owner>(*owner.value); }
};

再看一个常用的lambda表达式应用:

template <typename F>
auto check(F&& f) {
    return [f = std::forward<F>(f)](this auto&& owner)noexcept(!std::invoke(std::forward_like<decltype(owner)>(f)));
}

这里的就可以从模板参数的起始来完美转发相关的常量和左右值了。此处的noexcept使用到了其对表达式的处理方式,即noexcept(expression),noexcept和noexcept(true)等价,表示不抛出异常,为false时表示可能抛出异常。在新的标准里throw()这种方式已经被抛弃。

三、例程

再看一个cppreference上的例子:

#include <cstddef>
#include <iostream>
#include <memory>
#include <optional>
#include <type_traits>
#include <utility>
#include <vector>

struct TypeTeller
{
    void operator()(this auto&& self)
    {
        using SelfType = decltype(self);
        using UnrefSelfType = std::remove_reference_t<SelfType>;
        if constexpr (std::is_lvalue_reference_v<SelfType>)
        {
            if constexpr (std::is_const_v<UnrefSelfType>)
                std::cout << "const lvalue\n";
            else
                std::cout << "mutable lvalue\n";
        }
        else
        {
            if constexpr (std::is_const_v<UnrefSelfType>)
                std::cout << "const rvalue\n";
            else
                std::cout << "mutable rvalue\n";
        }
    }
};

struct FarStates
{
    std::unique_ptr<TypeTeller> ptr;
    std::optional<TypeTeller> opt;
    std::vector<TypeTeller> container;

    auto&& from_opt(this auto&& self)
    {
        return std::forward_like<decltype(self)>(self.opt.value());
        // It is OK to use std::forward<decltype(self)>(self).opt.value(),
        // because std::optional provides suitable accessors.
    }

    auto&& operator[](this auto&& self, std::size_t i)
    {
        return std::forward_like<decltype(self)>(container.at(i));
        // It is not so good to use std::forward<decltype(self)>(self)[i], because
        // containers do not provide rvalue subscript access, although they could.
    }

    auto&& from_ptr(this auto&& self)
    {
        if (!self.ptr)
            throw std::bad_optional_access{};
        return std::forward_like<decltype(self)>(*self.ptr);
        // It is not good to use *std::forward<decltype(self)>(self).ptr, because
        // std::unique_ptr<TypeTeller> always dereferences to a non-const lvalue.
    }
};

int main()
{
    FarStates my_state{
        .ptr{std::make_unique<TypeTeller>()},
        .opt{std::in_place, TypeTeller{} },
        .container{std::vector<TypeTeller>(1)},
    };

    my_state.from_ptr();
    my_state.from_opt();
    my_state[0]();

    std::cout << '\n';

    std::as_const(my_state).from_ptr();
    std::as_const(my_state).from_opt();
    std::as_const(my_state)[0]();

    std::cout << '\n';

    std::move(my_state).from_ptr();
    std::move(my_state).from_opt();
    std::move(my_state)[0]();

    std::cout << '\n';

    std::move(std::as_const(my_state)).from_ptr();
    std::move(std::as_const(my_state)).from_opt();
    std::move(std::as_const(my_state))[0]();

    std::cout << '\n';
}

运行结果:

mutable lvalue
mutable lvalue
mutable lvalue

const lvalue
const lvalue
const lvalue

mutable rvalue
mutable rvalue
mutable rvalue

const rvalue
const rvalue
const rvalue

注意,编译器需要支持。

四、总结

大家一起来打补丁吧。估计大佬们的心里也是有多少匹马在奔腾,但奔腾向何方,这个只有他们自己知道。快有快的好,慢有慢的好。哪个才是最合适的,只有试试才知道。然后,大佬们说:试试就试试。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值