EVOLUTION-MANAGER
Edit File: discrete_frechet_distance.hpp
// Boost.Geometry // Copyright (c) 2018 Yaghyavardhan Singh Khangarot, Hyderabad, India. // Contributed and/or modified by Yaghyavardhan Singh Khangarot, // as part of Google Summer of Code 2018 program. // This file was modified by Oracle on 2018. // Modifications copyright (c) 2018 Oracle and/or its affiliates. // Contributed and/or modified by Adam Wulkiewicz, on behalf of Oracle // Use, modification and distribution is subject to the Boost Software License, // Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at // http://www.boost.org/LICENSE_1_0.txt) #ifndef BOOST_GEOMETRY_ALGORITHMS_DISCRETE_FRECHET_DISTANCE_HPP #define BOOST_GEOMETRY_ALGORITHMS_DISCRETE_FRECHET_DISTANCE_HPP #include <algorithm> #ifdef BOOST_GEOMETRY_DEBUG_FRECHET_DISTANCE #include <iostream> #endif #include <iterator> #include <utility> #include <vector> #include <limits> #include <boost/geometry/algorithms/detail/throw_on_empty_input.hpp> #include <boost/geometry/algorithms/not_implemented.hpp> #include <boost/geometry/core/assert.hpp> #include <boost/geometry/core/tag.hpp> #include <boost/geometry/core/tags.hpp> #include <boost/geometry/core/point_type.hpp> #include <boost/geometry/strategies/distance.hpp> #include <boost/geometry/strategies/distance_result.hpp> #include <boost/geometry/util/range.hpp> namespace boost { namespace geometry { #ifndef DOXYGEN_NO_DETAIL namespace detail { namespace discrete_frechet_distance { template <typename size_type1 , typename size_type2,typename result_type> class coup_mat { public: coup_mat(size_type1 w, size_type2 h) : m_data(w * h,-1), m_width(w), m_height(h) {} result_type & operator()(size_type1 i, size_type2 j) { BOOST_GEOMETRY_ASSERT(i < m_width && j < m_height); return m_data[j * m_width + i]; } private: std::vector<result_type> m_data; size_type1 m_width; size_type2 m_height; }; struct linestring_linestring { template <typename Linestring1, typename Linestring2, typename Strategy> static inline typename distance_result < typename point_type<Linestring1>::type, typename point_type<Linestring2>::type, Strategy >::type apply(Linestring1 const& ls1, Linestring2 const& ls2, Strategy const& strategy) { typedef typename distance_result < typename point_type<Linestring1>::type, typename point_type<Linestring2>::type, Strategy >::type result_type; typedef typename boost::range_size<Linestring1>::type size_type1; typedef typename boost::range_size<Linestring2>::type size_type2; boost::geometry::detail::throw_on_empty_input(ls1); boost::geometry::detail::throw_on_empty_input(ls2); size_type1 const a = boost::size(ls1); size_type2 const b = boost::size(ls2); //Coupling Matrix CoupMat(a,b,-1); coup_mat<size_type1,size_type2,result_type> coup_matrix(a,b); result_type const not_feasible = -100; //findin the Coupling Measure for (size_type1 i = 0 ; i < a ; i++ ) { for(size_type2 j=0;j<b;j++) { result_type dis = strategy.apply(range::at(ls1,i), range::at(ls2,j)); if(i==0 && j==0) coup_matrix(i,j) = dis; else if(i==0 && j>0) coup_matrix(i,j) = (std::max)(coup_matrix(i,j-1), dis); else if(i>0 && j==0) coup_matrix(i,j) = (std::max)(coup_matrix(i-1,j), dis); else if(i>0 && j>0) coup_matrix(i,j) = (std::max)((std::min)(coup_matrix(i,j-1), (std::min)(coup_matrix(i-1,j), coup_matrix(i-1,j-1))), dis); else coup_matrix(i,j) = not_feasible; } } #ifdef BOOST_GEOMETRY_DEBUG_FRECHET_DISTANCE //Print CoupLing Matrix for(size_type i = 0; i <a; i++) { for(size_type j = 0; j <b; j++) std::cout << coup_matrix(i,j) << " "; std::cout << std::endl; } #endif return coup_matrix(a-1,b-1); } }; }} // namespace detail::frechet_distance #endif // DOXYGEN_NO_DETAIL #ifndef DOXYGEN_NO_DISPATCH namespace dispatch { template < typename Geometry1, typename Geometry2, typename Tag1 = typename tag<Geometry1>::type, typename Tag2 = typename tag<Geometry2>::type > struct discrete_frechet_distance : not_implemented<Tag1, Tag2> {}; template <typename Linestring1, typename Linestring2> struct discrete_frechet_distance < Linestring1, Linestring2, linestring_tag, linestring_tag > : detail::discrete_frechet_distance::linestring_linestring {}; } // namespace dispatch #endif // DOXYGEN_NO_DISPATCH /*! \brief Calculate discrete Frechet distance between two geometries (currently works for LineString-LineString) using specified strategy. \ingroup discrete_frechet_distance \tparam Geometry1 \tparam_geometry \tparam Geometry2 \tparam_geometry \tparam Strategy A type fulfilling a DistanceStrategy concept \param geometry1 Input geometry \param geometry2 Input geometry \param strategy Distance strategy to be used to calculate Pt-Pt distance \qbk{distinguish,with strategy} \qbk{[include reference/algorithms/discrete_frechet_distance.qbk]} \qbk{ [heading Available Strategies] \* [link geometry.reference.strategies.strategy_distance_pythagoras Pythagoras (cartesian)] \* [link geometry.reference.strategies.strategy_distance_haversine Haversine (spherical)] [/ \* more (currently extensions): Vincenty\, Andoyer (geographic) ] [heading Example] [discrete_frechet_distance_strategy] [discrete_frechet_distance_strategy_output] } */ template <typename Geometry1, typename Geometry2, typename Strategy> inline typename distance_result < typename point_type<Geometry1>::type, typename point_type<Geometry2>::type, Strategy >::type discrete_frechet_distance(Geometry1 const& geometry1, Geometry2 const& geometry2, Strategy const& strategy) { return dispatch::discrete_frechet_distance < Geometry1, Geometry2 >::apply(geometry1, geometry2, strategy); } // Algorithm overload using default Pt-Pt distance strategy /*! \brief Calculate discrete Frechet distance between two geometries (currently work for LineString-LineString). \ingroup discrete_frechet_distance \tparam Geometry1 \tparam_geometry \tparam Geometry2 \tparam_geometry \param geometry1 Input geometry \param geometry2 Input geometry \qbk{[include reference/algorithms/discrete_frechet_distance.qbk]} \qbk{ [heading Example] [discrete_frechet_distance] [discrete_frechet_distance_output] } */ template <typename Geometry1, typename Geometry2> inline typename distance_result < typename point_type<Geometry1>::type, typename point_type<Geometry2>::type >::type discrete_frechet_distance(Geometry1 const& geometry1, Geometry2 const& geometry2) { typedef typename strategy::distance::services::default_strategy < point_tag, point_tag, typename point_type<Geometry1>::type, typename point_type<Geometry2>::type >::type strategy_type; return discrete_frechet_distance(geometry1, geometry2, strategy_type()); } }} // namespace boost::geometry #endif // BOOST_GEOMETRY_ALGORITHMS_DISCRETE_FRECHET_DISTANCE_HPP