libcarla/include/system/boost/geometry/algorithms/discrete_frechet_distance.hpp
2024-10-18 13:19:59 +08:00

297 lines
9.3 KiB
C++

// 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-2021.
// Modifications copyright (c) 2018-2021 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/dummy_geometries.hpp>
#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/detail.hpp>
#include <boost/geometry/strategies/discrete_distance/cartesian.hpp>
#include <boost/geometry/strategies/discrete_distance/geographic.hpp>
#include <boost/geometry/strategies/discrete_distance/spherical.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
{
// TODO: The implementation should calculate comparable distances
template <typename SizeType1, typename SizeType2, typename ResultType>
class coup_mat
{
public:
coup_mat(SizeType1 w, SizeType2 h)
: m_data(w * h,-1), m_width(w), m_height(h)
{}
ResultType & operator()(SizeType1 i, SizeType2 j)
{
BOOST_GEOMETRY_ASSERT(i < m_width && j < m_height);
return m_data[j * m_width + i];
}
private:
std::vector<ResultType> m_data;
SizeType1 m_width;
SizeType2 m_height;
};
struct linestring_linestring
{
template <typename Linestring1, typename Linestring2, typename Strategies>
static inline auto apply(Linestring1 const& ls1, Linestring2 const& ls2,
Strategies const& strategies)
{
typedef typename distance_result
<
typename point_type<Linestring1>::type,
typename point_type<Linestring2>::type,
Strategies
>::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);
// We can assume the inputs are not empty
auto const strategy = strategies.distance(dummy_point(), dummy_point());
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
namespace resolve_strategy {
template
<
typename Strategies,
bool IsUmbrella = strategies::detail::is_umbrella_strategy<Strategies>::value
>
struct discrete_frechet_distance
{
template <typename Geometry1, typename Geometry2>
static inline auto apply(Geometry1 const& geometry1, Geometry2 const& geometry2,
Strategies const& strategies)
{
return dispatch::discrete_frechet_distance
<
Geometry1, Geometry2
>::apply(geometry1, geometry2, strategies);
}
};
template <typename Strategy>
struct discrete_frechet_distance<Strategy, false>
{
template <typename Geometry1, typename Geometry2>
static inline auto apply(Geometry1 const& geometry1, Geometry2 const& geometry2,
Strategy const& strategy)
{
using strategies::discrete_distance::services::strategy_converter;
return dispatch::discrete_frechet_distance
<
Geometry1, Geometry2
>::apply(geometry1, geometry2,
strategy_converter<Strategy>::get(strategy));
}
};
template <>
struct discrete_frechet_distance<default_strategy, false>
{
template <typename Geometry1, typename Geometry2>
static inline auto apply(Geometry1 const& geometry1, Geometry2 const& geometry2,
default_strategy const&)
{
typedef typename strategies::discrete_distance::services::default_strategy
<
Geometry1, Geometry2
>::type strategies_type;
return dispatch::discrete_frechet_distance
<
Geometry1, Geometry2
>::apply(geometry1, geometry2, strategies_type());
}
};
} // namespace resolve_strategy
/*!
\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 auto discrete_frechet_distance(Geometry1 const& geometry1,
Geometry2 const& geometry2,
Strategy const& strategy)
{
return resolve_strategy::discrete_frechet_distance
<
Strategy
>::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 auto discrete_frechet_distance(Geometry1 const& geometry1,
Geometry2 const& geometry2)
{
return resolve_strategy::discrete_frechet_distance
<
default_strategy
>::apply(geometry1, geometry2, default_strategy());
}
}} // namespace boost::geometry
#endif // BOOST_GEOMETRY_ALGORITHMS_DISCRETE_FRECHET_DISTANCE_HPP