Update contrib.
1 ///////////////////////////////////////////////////////////////////////////////
2 /// \file basic_regex.hpp
3 /// Contains the definition of the basic_regex\<\> class template and its
4 /// associated helper functions.
6 // Copyright 2004 Eric Niebler. Distributed under the Boost
7 // Software License, Version 1.0. (See accompanying file
8 // LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
10 #ifndef BOOST_XPRESSIVE_BASIC_REGEX_HPP_EAN_10_04_2005
11 #define BOOST_XPRESSIVE_BASIC_REGEX_HPP_EAN_10_04_2005
13 // MS compatible compilers support #pragma once
14 #if defined(_MSC_VER) && (_MSC_VER >= 1020)
18 #ifdef BOOST_XPRESSIVE_DEBUG_TRACKING_POINTER
21 #include <boost/mpl/bool.hpp>
22 #include <boost/xpressive/proto/proto_fwd.hpp>
23 #include <boost/xpressive/regex_constants.hpp>
24 #include <boost/xpressive/detail/detail_fwd.hpp>
25 #include <boost/xpressive/detail/core/regex_impl.hpp>
27 namespace boost { namespace xpressive
30 ///////////////////////////////////////////////////////////////////////////////
33 /// \brief Class template basic_regex\<\> is a class for holding a compiled regular expression.
34 template<typename BidiIter>
37 typedef BidiIter iterator_type;
38 typedef typename iterator_value<BidiIter>::type char_type;
39 typedef std::basic_string<char_type> string_type;
40 typedef regex_constants::syntax_option_type flag_type;
42 /// \post regex_id() == 0
43 /// \post mark_count() == 0
49 /// \param that The basic_regex object to copy.
50 /// \post regex_id() == that.regex_id()
51 /// \post mark_count() == that.mark_count()
52 basic_regex(basic_regex<BidiIter> const &that)
57 /// \param that The basic_regex object to copy.
58 /// \post regex_id() == that.regex_id()
59 /// \post mark_count() == that.mark_count()
61 basic_regex<BidiIter> &operator =(basic_regex<BidiIter> const &that)
63 this->impl_ = that.impl_;
67 /// Construct from a static regular expression.
69 /// \param xpr The static regular expression
70 /// \pre Xpr is the type of a static regular expression.
71 /// \post regex_id() != 0
72 /// \post mark_count() \>= 0
73 template<typename Xpr>
74 basic_regex(Xpr const &xpr)
77 this->operator =(xpr);
80 /// Construct from a static regular expression.
82 /// \param xpr The static regular expression.
83 /// \pre Xpr is the type of a static regular expression.
84 /// \post regex_id() != 0
85 /// \post mark_count() \>= 0
86 /// \throw std::bad_alloc on out of memory
88 template<typename Xpr>
89 basic_regex<BidiIter> &operator =(Xpr const &xpr)
91 detail::static_compile(xpr, *this->impl_.get());
95 /// Returns the count of capturing sub-expressions in this regular expression
97 std::size_t mark_count() const
99 return this->impl_ ? this->impl_->mark_count_ : 0;
102 /// Returns a token which uniquely identifies this regular expression.
104 regex_id_type regex_id() const
106 return this->impl_ ? this->impl_->xpr_.get() : 0;
109 /// Swaps the contents of this basic_regex object with another.
111 /// \param that The other basic_regex object.
112 /// \attention This is a shallow swap that does not do reference tracking. If you embed
113 /// a basic_regex object by reference in another regular expression and then swap its
114 /// contents with another basic_regex object, the change will not be visible to the enclosing
115 /// regular expression. It is done this way to ensure that swap() cannot throw.
117 void swap(basic_regex<BidiIter> &that) // throw()
119 this->impl_.swap(that.impl_);
122 /// Factory method for building a regex object from a string.
123 /// Equivalent to regex_compiler\< BidiIter \>().compile(str, flags);
125 /// \param str The std::basic_string containing the regular expression.
126 /// \param flags Optional bitmask of type syntax_option_type to control how str is interpreted.
127 static basic_regex<BidiIter> compile(string_type const &str, flag_type flags = regex_constants::ECMAScript)
129 return regex_compiler<BidiIter>().compile(str, flags);
132 // for binding actions to this regex when it is nested statically in another regex
134 template<typename Action>
137 proto::unary_op<basic_regex<BidiIter>, proto::noop_tag>
138 , proto::unary_op<Action, proto::noop_tag>
139 , proto::right_shift_tag
141 operator [](detail::action_matcher<Action> const &action) const
143 return proto::noop(*this) >> proto::noop(*static_cast<Action const *>(&action));
147 #ifdef BOOST_XPRESSIVE_DEBUG_TRACKING_POINTER
150 friend std::ostream &operator <<(std::ostream &sout, basic_regex<BidiIter> const &rex)
159 friend struct detail::core_access<BidiIter>;
161 // Avoid a common programming mistake. Construction from a string is ambiguous. It could mean
162 // sregex rx = sregex::compile(str); // compile the string into a regex
164 // sregex rx = as_xpr(str); // treat the string as a literal
165 // Since there is no easy way to disambiguate, disallow it and force users to say what they mean
167 basic_regex(char_type const *);
169 basic_regex(string_type const &);
171 // used from parser, via core_access
173 explicit basic_regex(detail::regex_impl<BidiIter> const &that)
176 this->impl_.tracking_copy(that);
180 bool match_(detail::state_type<BidiIter> &state) const
182 return this->impl_->xpr_->match(state);
185 // Returns true if this basic_regex object does not contain a valid regular expression.
187 bool invalid_() const
189 return !this->impl_ || !this->impl_->xpr_;
193 void dump_(std::ostream &sout) const;
195 // the tracking_ptr manages lazy-init, COW, cycle-breaking, and
196 // reference/dependency tracking.
197 detail::tracking_ptr<detail::regex_impl<BidiIter> > impl_;
201 #ifdef BOOST_XPRESSIVE_DEBUG_TRACKING_POINTER
202 ///////////////////////////////////////////////////////////////////////////////
205 template<typename BidiIter>
206 inline void basic_regex<BidiIter>::dump_(std::ostream &sout) const
210 sout << "<null> refs={} deps={}";
214 sout << *this->impl_;
220 ///////////////////////////////////////////////////////////////////////////////
222 /// \brief Swaps the contents of two basic_regex objects.
223 /// \param left The first basic_regex object.
224 /// \param right The second basic_regex object.
225 /// \attention This is a shallow swap that does not do reference tracking.
226 /// If you embed a basic_regex object by reference in another regular expression
227 /// and then swap its contents with another basic_regex object, the change will
228 /// not be visible to the enclosing regular expression. It is done this way to
229 /// ensure that swap() cannot throw.
231 template<typename BidiIter>
232 inline void swap(basic_regex<BidiIter> &left, basic_regex<BidiIter> &right) // throw()
237 }} // namespace boost::xpressive
239 #endif // BOOST_XPRESSIVE_REGEX_HPP_EAN_10_04_2005