1 // Copyright 2019 The Abseil Authors. 2 // 3 // Licensed under the Apache License, Version 2.0 (the "License"); 4 // you may not use this file except in compliance with the License. 5 // You may obtain a copy of the License at 6 // 7 // https://www.apache.org/licenses/LICENSE-2.0 8 // 9 // Unless required by applicable law or agreed to in writing, software 10 // distributed under the License is distributed on an "AS IS" BASIS, 11 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 12 // See the License for the specific language governing permissions and 13 // limitations under the License. 14 // 15 // ----------------------------------------------------------------------------- 16 // File: function_ref.h 17 // ----------------------------------------------------------------------------- 18 // 19 // This header file defines the `absl::FunctionRef` type for holding a 20 // non-owning reference to an object of any invocable type. This function 21 // reference is typically most useful as a type-erased argument type for 22 // accepting function types that neither take ownership nor copy the type; using 23 // the reference type in this case avoids a copy and an allocation. Best 24 // practices of other non-owning reference-like objects (such as 25 // `absl::string_view`) apply here. 26 // 27 // An `absl::FunctionRef` is similar in usage to a `std::function` but has the 28 // following differences: 29 // 30 // * It doesn't own the underlying object. 31 // * It doesn't have a null or empty state. 32 // * It never performs deep copies or allocations. 33 // * It's much faster and cheaper to construct. 34 // * It's trivially copyable and destructable. 35 // 36 // Generally, `absl::FunctionRef` should not be used as a return value, data 37 // member, or to initialize a `std::function`. Such usages will often lead to 38 // problematic lifetime issues. Once you convert something to an 39 // `absl::FunctionRef` you cannot make a deep copy later. 40 // 41 // This class is suitable for use wherever a "const std::function<>&" 42 // would be used without making a copy. ForEach functions and other versions of 43 // the visitor pattern are a good example of when this class should be used. 44 // 45 // This class is trivial to copy and should be passed by value. 46 #ifndef ABSL_FUNCTIONAL_FUNCTION_REF_H_ 47 #define ABSL_FUNCTIONAL_FUNCTION_REF_H_ 48 49 #include <cassert> 50 #include <functional> 51 #include <type_traits> 52 53 #include "absl/base/attributes.h" 54 #include "absl/functional/internal/function_ref.h" 55 #include "absl/meta/type_traits.h" 56 57 namespace absl { 58 ABSL_NAMESPACE_BEGIN 59 60 // FunctionRef 61 // 62 // Dummy class declaration to allow the partial specialization based on function 63 // types below. 64 template <typename T> 65 class FunctionRef; 66 67 // FunctionRef 68 // 69 // An `absl::FunctionRef` is a lightweight wrapper to any invocable object with 70 // a compatible signature. Generally, an `absl::FunctionRef` should only be used 71 // as an argument type and should be preferred as an argument over a const 72 // reference to a `std::function`. `absl::FunctionRef` itself does not allocate, 73 // although the wrapped invocable may. 74 // 75 // Example: 76 // 77 // // The following function takes a function callback by const reference 78 // bool Visitor(const std::function<void(my_proto&, 79 // absl::string_view)>& callback); 80 // 81 // // Assuming that the function is not stored or otherwise copied, it can be 82 // // replaced by an `absl::FunctionRef`: 83 // bool Visitor(absl::FunctionRef<void(my_proto&, absl::string_view)> 84 // callback); 85 // 86 // Note: the assignment operator within an `absl::FunctionRef` is intentionally 87 // deleted to prevent misuse; because the `absl::FunctionRef` does not own the 88 // underlying type, assignment likely indicates misuse. 89 template <typename R, typename... Args> 90 class FunctionRef<R(Args...)> { 91 private: 92 // Used to disable constructors for objects that are not compatible with the 93 // signature of this FunctionRef. 94 template <typename F, 95 typename FR = absl::base_internal::invoke_result_t<F, Args&&...>> 96 using EnableIfCompatible = 97 typename std::enable_if<std::is_void<R>::value || 98 std::is_convertible<FR, R>::value>::type; 99 100 public: 101 // Constructs a FunctionRef from any invocable type. 102 template <typename F, typename = EnableIfCompatible<const F&>> 103 // NOLINTNEXTLINE(runtime/explicit) FunctionRef(const F & f ABSL_ATTRIBUTE_LIFETIME_BOUND)104 FunctionRef(const F& f ABSL_ATTRIBUTE_LIFETIME_BOUND) 105 : invoker_(&absl::functional_internal::InvokeObject<F, R, Args...>) { 106 absl::functional_internal::AssertNonNull(f); 107 ptr_.obj = &f; 108 } 109 110 // Overload for function pointers. This eliminates a level of indirection that 111 // would happen if the above overload was used (it lets us store the pointer 112 // instead of a pointer to a pointer). 113 // 114 // This overload is also used for references to functions, since references to 115 // functions can decay to function pointers implicitly. 116 template < 117 typename F, typename = EnableIfCompatible<F*>, 118 absl::functional_internal::EnableIf<absl::is_function<F>::value> = 0> FunctionRef(F * f)119 FunctionRef(F* f) // NOLINT(runtime/explicit) 120 : invoker_(&absl::functional_internal::InvokeFunction<F*, R, Args...>) { 121 assert(f != nullptr); 122 ptr_.fun = reinterpret_cast<decltype(ptr_.fun)>(f); 123 } 124 125 // To help prevent subtle lifetime bugs, FunctionRef is not assignable. 126 // Typically, it should only be used as an argument type. 127 FunctionRef& operator=(const FunctionRef& rhs) = delete; 128 FunctionRef(const FunctionRef& rhs) = default; 129 130 // Call the underlying object. operator()131 R operator()(Args... args) const { 132 return invoker_(ptr_, std::forward<Args>(args)...); 133 } 134 135 private: 136 absl::functional_internal::VoidPtr ptr_; 137 absl::functional_internal::Invoker<R, Args...> invoker_; 138 }; 139 140 // Allow const qualified function signatures. Since FunctionRef requires 141 // constness anyway we can just make this a no-op. 142 template <typename R, typename... Args> 143 class FunctionRef<R(Args...) const> : public FunctionRef<R(Args...)> { 144 public: 145 using FunctionRef<R(Args...)>::FunctionRef; 146 }; 147 148 ABSL_NAMESPACE_END 149 } // namespace absl 150 151 #endif // ABSL_FUNCTIONAL_FUNCTION_REF_H_ 152