使用模板化类重载lexical_Cast

Overloading lexical_Cast with a templated class

本文关键字:重载 lexical Cast      更新时间:2023-10-16

我正在尝试扩展lexical_cast以处理字符串->cv::点转换,代码如下:

#include <iostream>
#include <opencv2/opencv.hpp>
#include <boost/lexical_cast.hpp>
#include <boost/algorithm/string/classification.hpp>
#include <boost/algorithm/string/split.hpp>
namespace boost {
  template<>
    cv::Point2f lexical_cast(const std::string &str) {
      std::vector<std::string> parts;
      boost::split(parts, str, boost::is_any_of(","));
      cv::Point2f R;
      R.x = boost::lexical_cast<float>(parts[0]);
      R.y = boost::lexical_cast<float>(parts[1]);
      return R;
    }
}
int main(int argc, char **argv) {
  auto p = boost::lexical_cast<cv::Point2f>(std::string("1,2"));
  std::cout << "p = " << p << std::endl;
  return 0;
}

而且效果很好。。然而,cv::Point2f实际上是cv::Point_<T>,其中T可以是int、float、double等。我无论如何都找不到将模板化的arg公开给lexical_cast,这样我就可以拥有一个可以处理所有cv::Point_<T>类型的lexical_cst函数。

template <typename T>
struct point_type {};
template <typename T>
struct point_type<cv::Point_<T>> { using type = T; };
namespace boost {
  template <typename T, typename U = typename point_type<T>::type>
    T lexical_cast(const std::string &str)
    {
      std::vector<std::string> parts;
      boost::split(parts, str, boost::is_any_of(","));
      T R;
      R.x = boost::lexical_cast<U>(parts[0]);
      R.y = boost::lexical_cast<U>(parts[1]);
      return R;
    }
}

演示


前面的解决方案稍微复杂一点,如果你不喜欢lexical_cast:的这个隐含的第二个模板参数

#include <type_traits>
template <typename T>
struct is_point : std::false_type {};
template <typename T>
struct is_point<cv::Point_<T>> : std::true_type {};
template <typename T>
struct point_type;
template <typename T>
struct point_type<cv::Point_<T>> { using type = T; };
namespace boost {
  template <typename T>
    auto lexical_cast(const std::string &str)
      -> typename std::enable_if<is_point<T>::value, T>::type
    {
      std::vector<std::string> parts;
      boost::split(parts, str, boost::is_any_of(","));
      using U = typename point_type<T>::type;
      T R;
      R.x = boost::lexical_cast<U>(parts[0]);
      R.y = boost::lexical_cast<U>(parts[1]);
      return R;
    }
}

演示2