You can not select more than 25 topics Topics must start with a chinese character,a letter or number, can include dashes ('-') and can be up to 35 characters long.

function_ref.h 6.2 kB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148
  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. #include <cassert>
  49. #include <functional>
  50. #include <type_traits>
  51. #include "absl/base/attributes.h"
  52. #include "absl/functional/internal/function_ref.h"
  53. #include "absl/meta/type_traits.h"
  54. namespace absl
  55. {
  56. ABSL_NAMESPACE_BEGIN
  57. // FunctionRef
  58. //
  59. // Dummy class declaration to allow the partial specialization based on function
  60. // types below.
  61. template<typename T>
  62. class FunctionRef;
  63. // FunctionRef
  64. //
  65. // An `absl::FunctionRef` is a lightweight wrapper to any invokable object with
  66. // a compatible signature. Generally, an `absl::FunctionRef` should only be used
  67. // as an argument type and should be preferred as an argument over a const
  68. // reference to a `std::function`. `absl::FunctionRef` itself does not allocate,
  69. // although the wrapped invokable may.
  70. //
  71. // Example:
  72. //
  73. // // The following function takes a function callback by const reference
  74. // bool Visitor(const std::function<void(my_proto&,
  75. // absl::string_view)>& callback);
  76. //
  77. // // Assuming that the function is not stored or otherwise copied, it can be
  78. // // replaced by an `absl::FunctionRef`:
  79. // bool Visitor(absl::FunctionRef<void(my_proto&, absl::string_view)>
  80. // callback);
  81. //
  82. // Note: the assignment operator within an `absl::FunctionRef` is intentionally
  83. // deleted to prevent misuse; because the `absl::FunctionRef` does not own the
  84. // underlying type, assignment likely indicates misuse.
  85. template<typename R, typename... Args>
  86. class FunctionRef<R(Args...)>
  87. {
  88. private:
  89. // Used to disable constructors for objects that are not compatible with the
  90. // signature of this FunctionRef.
  91. template<typename F, typename FR = absl::base_internal::invoke_result_t<F, Args&&...>>
  92. using EnableIfCompatible =
  93. typename std::enable_if<std::is_void<R>::value || std::is_convertible<FR, R>::value>::type;
  94. public:
  95. // Constructs a FunctionRef from any invokable type.
  96. template<typename F, typename = EnableIfCompatible<const F&>>
  97. // NOLINTNEXTLINE(runtime/explicit)
  98. FunctionRef(const F& f ABSL_ATTRIBUTE_LIFETIME_BOUND) :
  99. invoker_(&absl::functional_internal::InvokeObject<F, R, Args...>)
  100. {
  101. absl::functional_internal::AssertNonNull(f);
  102. ptr_.obj = &f;
  103. }
  104. // Overload for function pointers. This eliminates a level of indirection that
  105. // would happen if the above overload was used (it lets us store the pointer
  106. // instead of a pointer to a pointer).
  107. //
  108. // This overload is also used for references to functions, since references to
  109. // functions can decay to function pointers implicitly.
  110. template<
  111. typename F,
  112. typename = EnableIfCompatible<F*>,
  113. absl::functional_internal::EnableIf<absl::is_function<F>::value> = 0>
  114. FunctionRef(F* f) // NOLINT(runtime/explicit)
  115. :
  116. invoker_(&absl::functional_internal::InvokeFunction<F*, R, Args...>)
  117. {
  118. assert(f != nullptr);
  119. ptr_.fun = reinterpret_cast<decltype(ptr_.fun)>(f);
  120. }
  121. // To help prevent subtle lifetime bugs, FunctionRef is not assignable.
  122. // Typically, it should only be used as an argument type.
  123. FunctionRef& operator=(const FunctionRef& rhs) = delete;
  124. FunctionRef(const FunctionRef& rhs) = default;
  125. // Call the underlying object.
  126. R operator()(Args... args) const
  127. {
  128. return invoker_(ptr_, std::forward<Args>(args)...);
  129. }
  130. private:
  131. absl::functional_internal::VoidPtr ptr_;
  132. absl::functional_internal::Invoker<R, Args...> invoker_;
  133. };
  134. ABSL_NAMESPACE_END
  135. } // namespace absl
  136. #endif // ABSL_FUNCTIONAL_FUNCTION_REF_H_