alternate_matcher.hpp

来自「support vector clustering for vc++」· HPP 代码 · 共 212 行

HPP
212
字号
///////////////////////////////////////////////////////////////////////////////
// alternate_matcher.hpp
//
//  Copyright 2004 Eric Niebler. Distributed under 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_XPRESSIVE_DETAIL_CORE_MATCHER_ALTERNATE_MATCHER_HPP_EAN_10_04_2005
#define BOOST_XPRESSIVE_DETAIL_CORE_MATCHER_ALTERNATE_MATCHER_HPP_EAN_10_04_2005

// MS compatible compilers support #pragma once
#if defined(_MSC_VER) && (_MSC_VER >= 1020)
# pragma once
#endif

#include <vector>

#include <boost/version.hpp>
#if BOOST_VERSION <= 103200
// WORKAROUND for Fusion bug in Boost 1.32
namespace boost { namespace fusion
{
    namespace detail { struct iterator_root; }
    using detail::iterator_root;
}}
#endif

#include <boost/shared_ptr.hpp>
#include <boost/spirit/fusion/sequence/range.hpp>
#include <boost/spirit/fusion/algorithm/for_each.hpp>
#include <boost/xpressive/detail/detail_fwd.hpp>
#include <boost/xpressive/detail/core/quant_style.hpp>
#include <boost/xpressive/detail/core/state.hpp>
#include <boost/xpressive/detail/static/width_of.hpp>
#include <boost/xpressive/detail/static/is_pure.hpp>
#include <boost/xpressive/detail/dynamic/matchable.hpp>
#include <boost/xpressive/detail/utility/hash_peek_bitset.hpp>
#include <boost/xpressive/detail/utility/algorithm.hpp>
#include <boost/xpressive/detail/utility/any.hpp>

namespace boost { namespace xpressive { namespace detail
{

    ///////////////////////////////////////////////////////////////////////////////
    // alt_match_pred
    //
    template<typename BidiIter, typename Next>
    struct alt_match_pred
    {
        state_type<BidiIter> &state_;

        alt_match_pred(state_type<BidiIter> &state)
          : state_(state)
        {
        }

        template<typename Xpr>
        bool operator ()(Xpr const &xpr) const
        {
            return get_pointer(xpr)->BOOST_NESTED_TEMPLATE push_match<Next>(this->state_);
        }

    private:
        alt_match_pred &operator =(alt_match_pred const &);
    };

    ///////////////////////////////////////////////////////////////////////////////
    // alt_match
    //
    template<typename BidiIter>
    inline bool alt_match
    (
        std::vector<shared_ptr<matchable<BidiIter> const> > const &alternates
      , state_type<BidiIter> &state
      , matchable<BidiIter> const &
    )
    {
        typedef alt_match_pred<BidiIter, matchable<BidiIter> > alt_match_pred;
        return detail::any(alternates.begin(), alternates.end(), alt_match_pred(state));
    }

    template<typename BidiIter, typename Next, typename Alternates>
    inline bool alt_match
    (
        fusion::sequence_base<Alternates> const &alternates
      , state_type<BidiIter> &state
      , Next const &
    )
    {
        typedef alt_match_pred<BidiIter, Next> alt_match_pred;
        return fusion::any(alternates.cast(), alt_match_pred(state));
    }

    ///////////////////////////////////////////////////////////////////////////////
    // make_range
    template<typename Begin, typename End>
    inline fusion::range<Begin, End> make_range(Begin const &begin, End const &end)
    {
        return fusion::range<Begin, End>(begin, end);
    }

    ///////////////////////////////////////////////////////////////////////////////
    // alt_get_width_pred
    //
    template<typename BidiIter>
    struct alt_get_width_pred
    {
        state_type<BidiIter> *state_;
        std::size_t *width_;

        alt_get_width_pred(state_type<BidiIter> *state, std::size_t *width)
          : state_(state)
          , width_(width)
        {
        }

        template<typename Xpr>
        void operator ()(Xpr const &xpr) const
        {
            if(*this->width_ != unknown_width())
            {
                std::size_t that_width = get_pointer(xpr)->get_width(this->state_);
                if(*this->width_ != that_width)
                {
                    *this->width_ = unknown_width();
                }
            }
        }
    };

    ///////////////////////////////////////////////////////////////////////////////
    // alt_get_width
    //
    template<typename BidiIter>
    inline std::size_t alt_get_width
    (
        std::vector<shared_ptr<matchable<BidiIter> const> > const &alternates
      , state_type<BidiIter> *state
    )
    {
        typedef alt_get_width_pred<BidiIter> alt_get_width_pred;
        std::size_t width = alternates.front()->get_width(state);
        std::for_each(alternates.begin() + 1, alternates.end(), alt_get_width_pred(state, &width));
        return width;
    }

    template<typename BidiIter, typename Alternates>
    inline std::size_t alt_get_width
    (
        fusion::sequence_base<Alternates> const &alternates
      , state_type<BidiIter> *state
    )
    {
        typedef alt_get_width_pred<BidiIter> alt_get_width_pred;
        std::size_t width = (*fusion::begin(alternates.cast())).get_width(state);
        fusion::for_each
        (
            make_range(fusion::next(fusion::begin(alternates.cast())), fusion::end(alternates.cast()))
          , alt_get_width_pred(state, &width)
        );
        return width;
    }

    ///////////////////////////////////////////////////////////////////////////////
    // alternate_matcher
    //
    template<typename Alternates, typename Traits>
    struct alternate_matcher
      : quant_style_auto<width_of<Alternates>, is_pure<Alternates> >
    {
        typedef Alternates alternates_type;
        typedef typename Traits::char_type char_type;

        Alternates alternates_;
        mutable hash_peek_bitset<char_type> bset_;

        explicit alternate_matcher(Alternates const &alternates = Alternates())
          : alternates_(alternates)
          , bset_()
        {
        }

        template<typename BidiIter, typename Next>
        bool match(state_type<BidiIter> &state, Next const &next) const
        {
            if(!state.eos() && !this->can_match_(*state.cur_, traits_cast<Traits>(state)))
            {
                return false;
            }

            return detail::alt_match(this->alternates_, state, next);
        }

        template<typename BidiIter>
        std::size_t get_width(state_type<BidiIter> *state) const
        {
            return detail::alt_get_width(this->alternates_, state);
        }

    private:
        alternate_matcher &operator =(alternate_matcher const &);

        bool can_match_(char_type ch, Traits const &traits) const
        {
            return this->bset_.test(ch, traits);
        }
    };

}}}

#endif

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?