* mln/core/p_complex_piter.hh
(mln::p_complex_fwd_piter_<D, P>::iter)
(mln::p_complex_bkd_piter_<D, P>::iter):
New typedef.
(mln::p_complex_fwd_piter_<D, P>::face_)
(mln::p_complex_bkd_piter_<D, P>::face_):
Remove member.
(mln::p_complex_fwd_piter_<D, P>::iter_)
(mln::p_complex_bkd_piter_<D, P>::iter_):
New member.
(mln::p_complex_fwd_piter_<D, P>::p_complex_fwd_piter_)
(mln::p_complex_bkd_piter_<D, P>::p_complex_bkd_piter_):
Adjust ctors.
(mln::p_complex_fwd_piter_<D, P>::is_valid)
(mln::p_complex_fwd_piter_<D, P>::invalidate)
(mln::p_complex_fwd_piter_<D, P>::start)
(mln::p_complex_fwd_piter_<D, P>::next)
(mln::p_complex_fwd_piter_<D, P>::update):
Adjust.
(mln::p_complex_fwd_piter_<D, P>::invalid_unsigned_)
(mln::p_complex_bkd_piter_<D, P>::invalid_unsigned_):
Remove, useless.
---
milena/ChangeLog | 27 ++++++
milena/mln/core/p_complex_piter.hh | 157 ++++++++----------------------------
2 files changed, 61 insertions(+), 123 deletions(-)
diff --git a/milena/ChangeLog b/milena/ChangeLog
index b40e272..cd60428 100644
--- a/milena/ChangeLog
+++ b/milena/ChangeLog
@@ -1,5 +1,32 @@
2008-09-11 Roland Levillain <roland(a)lrde.epita.fr>
+ Factor iterators on p_complexes with iterator on complexes.
+
+ * mln/core/p_complex_piter.hh
+ (mln::p_complex_fwd_piter_<D, P>::iter)
+ (mln::p_complex_bkd_piter_<D, P>::iter):
+ New typedef.
+ (mln::p_complex_fwd_piter_<D, P>::face_)
+ (mln::p_complex_bkd_piter_<D, P>::face_):
+ Remove member.
+ (mln::p_complex_fwd_piter_<D, P>::iter_)
+ (mln::p_complex_bkd_piter_<D, P>::iter_):
+ New member.
+ (mln::p_complex_fwd_piter_<D, P>::p_complex_fwd_piter_)
+ (mln::p_complex_bkd_piter_<D, P>::p_complex_bkd_piter_):
+ Adjust ctors.
+ (mln::p_complex_fwd_piter_<D, P>::is_valid)
+ (mln::p_complex_fwd_piter_<D, P>::invalidate)
+ (mln::p_complex_fwd_piter_<D, P>::start)
+ (mln::p_complex_fwd_piter_<D, P>::next)
+ (mln::p_complex_fwd_piter_<D, P>::update):
+ Adjust.
+ (mln::p_complex_fwd_piter_<D, P>::invalid_unsigned_)
+ (mln::p_complex_bkd_piter_<D, P>::invalid_unsigned_):
+ Remove, useless.
+
+2008-09-11 Roland Levillain <roland(a)lrde.epita.fr>
+
Add default constructors to complexes on iterators.
* mln/core/complex_iter.hh
diff --git a/milena/mln/core/p_complex_piter.hh b/milena/mln/core/p_complex_piter.hh
index 98efc11..d9ce17a 100644
--- a/milena/mln/core/p_complex_piter.hh
+++ b/milena/mln/core/p_complex_piter.hh
@@ -21,7 +21,7 @@
// file, or you compile this file and link it with other files to
// produce an executable, this file does not by itself cause the
// resulting executable to be covered by the GNU General Public
-// License.
+// License.
// reasons why the executable file might be covered by the GNU General
// Public License.
@@ -36,6 +36,7 @@
# include <mln/core/internal/point_iterator_base.hh>
# include <mln/core/p_complex.hh>
# include <mln/core/complex_psite.hh>
+# include <mln/core/complex_iter.hh>
// Factor p_complex_fwd_piter_<D, P> and p_complex_bkd_piter_<D, P>.
@@ -60,6 +61,9 @@ namespace mln
typedef p_complex_fwd_piter_<D, P> self_;
typedef internal::point_iterator_base_< P, self_ > super_;
+ /// The type of the underlying complex iterator.
+ typedef complex_fwd_iter_<D> iter;
+
public:
// Make definitions from super class available.
// FIXME: Is it still meaningful for a complex?
@@ -107,27 +111,13 @@ namespace mln
/// \}
private:
- /// The face handle this iterator is pointing to.
- any_face_handle<D> face_;
+ /// The underlying complex iterator.
+ iter iter_;
/// The psite corresponding to this iterator.
psite psite_;
/// \brief The point associated to this psite.
// FIXME: Actually, this is a dummy value!
point p_;
-
- /// \brief An invalid value for both the dimension and the id of
- /// the face.
- ///
- /// Use a function instead of a static constant, since `static'
- /// variables needs to be compiled once, which requires a compiled
- /// library to avoid duplicate symbols, which is something that
- /// was not really planned in Milena. A function tagged `inlined'
- /// can appear multiple times in a program, and solves this
- /// problem. We rely on the compiler to inline this call.
- ///
- /// Of course, we could have used UINT_MAX, but it is not very
- /// C++.
- unsigned invalid_unsigned_() const;
};
@@ -153,6 +143,9 @@ namespace mln
typedef p_complex_bkd_piter_<D, P> self_;
typedef internal::point_iterator_base_< P, self_ > super_;
+ /// The type of the underlying complex iterator.
+ typedef complex_bkd_iter_<D> iter;
+
public:
// Make definitions from super class available.
// FIXME: Is it still meaningful for a complex?
@@ -200,27 +193,13 @@ namespace mln
/// \}
private:
- /// The face handle this iterator is pointing to.
- any_face_handle<D> face_;
+ /// The underlying complex iterator.
+ iter iter_;
/// The psite corresponding to this iterator.
psite psite_;
/// \brief The point associated to this psite.
// FIXME: Actually, this is a dummy value!
point p_;
-
- /// \brief An invalid value for both the dimension and the id of
- /// the face.
- ///
- /// Use a function instead of a static constant, since `static'
- /// variables needs to be compiled once, which requires a compiled
- /// library to avoid duplicate symbols, which is something that
- /// was not really planned in Milena. A function tagged `inlined'
- /// can appear multiple times in a program, and solves this
- /// problem. We rely on the compiler to inline this call.
- ///
- /// Of course, we could have used UINT_MAX, but it is not very
- /// C++.
- unsigned invalid_unsigned_() const;
};
@@ -248,15 +227,14 @@ namespace mln
: psite_(),
p_()
{
- face_.set_cplx(pc.cplx());
- // Invalidate face_.
- invalidate();
+ iter_.set_cplx(pc.cplx());
+ mln_postcondition(!is_valid());
}
template <unsigned D, typename P>
inline
p_complex_fwd_piter_<D, P>::p_complex_fwd_piter_(const p_complex_fwd_piter_<D, P>& rhs)
- : face_(rhs.face_),
+ : iter_(rhs.iter_),
psite_(rhs.psite_),
// Dummy value.
p_()
@@ -270,7 +248,7 @@ namespace mln
{
if (&rhs == this)
return *this;
- face_ = rhs.face_;
+ iter_ = rhs.iter_;
psite_ = rhs.psite_;
return *this;
}
@@ -289,7 +267,7 @@ namespace mln
bool
p_complex_fwd_piter_<D, P>::is_valid() const
{
- return face_.is_valid();
+ return iter_.is_valid();
}
template <unsigned D, typename P>
@@ -297,50 +275,24 @@ namespace mln
void
p_complex_fwd_piter_<D, P>::invalidate()
{
- face_.set_n(invalid_unsigned_());
- face_.set_face_id(invalid_unsigned_());
+ iter_.invalidate();
}
template <unsigned D, typename P>
inline
- void
+ void
p_complex_fwd_piter_<D, P>::start()
{
- face_.set_n(0u);
- face_.set_face_id(0u);
+ iter_.start();
update_();
}
template <unsigned D, typename P>
inline
- void
+ void
p_complex_fwd_piter_<D, P>::next_()
{
- if (is_valid())
- {
- unsigned n = face_.n();
- unsigned face_id = face_.face_id();
-
- if (face_id + 1 < face_.cplx().nfaces(n))
- /* FIXME: Provide accessor any_face_handle::n() returning
- a mutable reference? This way, we could just write
-
- ++face_.face_id();
-
- instead of the following. */
- face_.set_face_id(face_id + 1);
- else
- // Start to iterate on the faces of the next dimension if
- // possible.
- if (n <= D)
- {
- // FIXME: Same remark as above.
- face_.set_n(n + 1);
- face_.set_face_id(0u);
- }
- else
- invalidate();
- }
+ iter_.next_();
if (is_valid())
update_();
}
@@ -351,7 +303,7 @@ namespace mln
p_complex_fwd_piter_<D, P>::update_()
{
// Update psite_.
- psite_ = complex_psite<D, P>(face_);
+ psite_ = complex_psite<D, P>(iter_);
}
template <unsigned D, typename P>
@@ -390,13 +342,6 @@ namespace mln
return psite_;
}
- template <unsigned D, typename P>
- unsigned
- p_complex_fwd_piter_<D, P>::invalid_unsigned_() const
- {
- return std::numeric_limits<unsigned>::max();
- }
-
template <unsigned D, typename P>
inline
@@ -431,15 +376,14 @@ namespace mln
: psite_(),
p_()
{
- face_.set_cplx(pc.cplx());
- // Invalidate face_.
- invalidate();
+ iter_.set_cplx(pc.cplx());
+ mln_postcondition(!is_valid());
}
template <unsigned D, typename P>
inline
p_complex_bkd_piter_<D, P>::p_complex_bkd_piter_(const p_complex_bkd_piter_<D, P>& rhs)
- : face_(rhs.face_),
+ : iter_(rhs.iter_),
psite_(rhs.psite_),
// Dummy value.
p_()
@@ -453,7 +397,7 @@ namespace mln
{
if (&rhs == this)
return *this;
- face_ = rhs.face_;
+ iter_ = rhs.iter_;
psite_ = rhs.psite_;
return *this;
}
@@ -472,7 +416,7 @@ namespace mln
bool
p_complex_bkd_piter_<D, P>::is_valid() const
{
- return face_.is_valid();
+ return iter_.is_valid();
}
template <unsigned D, typename P>
@@ -480,50 +424,24 @@ namespace mln
void
p_complex_bkd_piter_<D, P>::invalidate()
{
- face_.set_n(invalid_unsigned_());
- face_.set_face_id(invalid_unsigned_());
+ iter_.invalidate();
}
template <unsigned D, typename P>
inline
- void
+ void
p_complex_bkd_piter_<D, P>::start()
{
- face_.set_n(D);
- face_.set_face_id(face_.cplx().template nfaces<D>() - 1);
+ iter_.start();
update_();
}
template <unsigned D, typename P>
inline
- void
+ void
p_complex_bkd_piter_<D, P>::next_()
{
- if (is_valid())
- {
- unsigned n = face_.n();
- unsigned face_id = face_.face_id();
-
- if (face_id > 0)
- /* FIXME: Provide accessor any_face_handle::n() returning
- a mutable reference? This way, we could just write
-
- ++face_.face_id();
-
- instead of the following. */
- face_.set_face_id(face_id - 1);
- else
- // Start to iterate on the faces of the previous dimension
- // if it exists.
- if (n > 0)
- {
- // FIXME: Same remark as above.
- face_.set_n(n - 1);
- face_.set_face_id(face_.cplx().nfaces(n - 1) - 1);
- }
- else
- invalidate();
- }
+ iter_.next_();
if (is_valid())
update_();
}
@@ -534,7 +452,7 @@ namespace mln
p_complex_bkd_piter_<D, P>::update_()
{
// Update psite_.
- psite_ = complex_psite<D, P>(face_);
+ psite_ = complex_psite<D, P>(iter_);
}
template <unsigned D, typename P>
@@ -573,13 +491,6 @@ namespace mln
return psite_;
}
- template <unsigned D, typename P>
- unsigned
- p_complex_bkd_piter_<D, P>::invalid_unsigned_() const
- {
- return std::numeric_limits<unsigned>::max();
- }
-
template <unsigned D, typename P>
inline
--
1.5.6.5
URL: https://svn.lrde.epita.fr/svn/oln/branches/cleanup-2008
ChangeLog:
2008-09-11 Guillaume Lazzara <z(a)lrde.epita.fr>
Add a section about graph images in tutorial.
* milena/doc/tutorial/tutorial.tex: Add a section about graph images and fix few mistakes all along the tutorial.
---
tutorial.tex | 281 ++++++++++++++++++++++++++++++++++++++++++++++++++---------
1 file changed, 239 insertions(+), 42 deletions(-)
Index: branches/cleanup-2008/milena/doc/tutorial/tutorial.tex
===================================================================
--- branches/cleanup-2008/milena/doc/tutorial/tutorial.tex (revision 2214)
+++ branches/cleanup-2008/milena/doc/tutorial/tutorial.tex (revision 2215)
@@ -144,17 +144,26 @@
The previous methods are available depending on the site set. A box
will have the bbox() method since it can be retrived in constant time: a box
-is it's own bounding box (see \ref{fig:box_bbox}). A p\_array does not have this
+is it's own bounding box (see \ref{fig:box_bbox}).
+
+\begin{lstlisting}[frame=single]
+ box2d b(2,3);
+
+ // The bbox can be retrived in constant time.
+ std::cout << b.bbox() << std::endl;
+
+ // nsites can be retrieved in constant time.
+ std::cout << "nsites = " << b.nsites() << std::endl;
+\end{lstlisting}
+
+A p\_array does not have this
method since sites do not have to be adjacent. Maintaining such information, in
order to keep getting the bbox in constant time, would be time and memory
consuming. Instead of providing a method directly in p\_array, an algorithm is
-available if this information needed (see \ref{fig:parray_bbox}).
+available if this information is needed.
P\_array and box both have a nsites method since the internal structure allow a
constant time retrieval.
-\subsubsection*{Sample code}
-
-\begin{figure}[ht!]
\begin{lstlisting}[frame=single]
p_array<point2d> arr;
@@ -166,21 +175,9 @@
// it can be retrieved in constant time.
std::cout << "nsites = " << arr.nsites() << std::endl;
\end{lstlisting}
- \caption{How to retrieve information from a p\_array.\label{fig:parray_bbox}}
-\end{figure}
-\begin{figure}[ht!]
- \begin{lstlisting}[frame=single]
- box2d b(2,3);
- // The bbox can be retrived in constant time.
- std::cout << b.bbox() << std::endl;
- // nsites can be retrieved in constant time.
- std::cout << "nsites = " << b.nsites() << std::endl;
- \end{lstlisting}
- \caption{How to retrieve information from a box.\label{fig:box_bbox}}
-\end{figure}
\clearpage
\newpage
@@ -199,8 +196,8 @@
\item A site set, also called the "domain".
\end{itemize}
-Every image type is defined on a specific site type. An image2d will always
-have a domain defined by a box2d.
+An image2d will always have its bouding box defined by a box2d whatever the
+site set used to define its domain.
The Value set, which includes all the possible values a site can have, is also
called "destination" set.
@@ -208,7 +205,7 @@
border is virtual since the image can have an extended domain as well.
That one is optional, it defines sites outside the virtual border which is
useful in algorithms when working with sites being part of the domain but close
-to the borders. The virtual border can be defined thanks to a function, an
+to the borders. The extended domain can be defined thanks to a function, an
image or a site set.
//FIXME: remove this line
@@ -373,7 +370,6 @@
\chapter{Iterators}
-Every objects
Each container object in Olena like site sets or images have iterators.
There are usually three kinds:
\begin{itemize}
@@ -413,14 +409,11 @@
Example of different forward iterations:
\begin{itemize}
\item box2d: from top to bottom then from left to right.
- \item p\_array<point2d>: from left to right.
+ \item p\_array$<$point2d$>$: from left to right.
\end{itemize}
-A for\_all() macro is available to iterate over all the sites
-(Fig. \ref{fig:for_all}). \\
+A for\_all() macro is available to iterate over all the sites: \\
-
-\begin{figure}[ht!]
\begin{lstlisting}[frame=single]
box2d b(3, 2);
mln_piter(box2d) p(b);
@@ -428,16 +421,12 @@
for_all(p)
std::cout << p; //prints every site coordinates.
\end{lstlisting}
- \caption{Use of the for\_all() macro.\label{fig:for_all}}
-\end{figure}
Note that when you declare an iterator, prefer using the "mln\_*iter" macros.
They resolve the iterator type automatically from the given container type
passed as parameter. These macros can be used with any container like images or
site sets.
-(\ref{fig:iter_allcontainers}).
-\begin{figure}[ht!]
\begin{lstlisting}[frame=single]
image2d<int> ima(box2d(2, 3));
@@ -449,23 +438,28 @@
for_all(v)
std::cout << v << std::endl;
\end{lstlisting}
- \caption{mln\_*iter macros can be used with any
-containers.\label{fig:iter_allcontainers}}
-\end{figure}
\chapter{Basic operations}
//FIXME : illustrer
-\begin{itemize}
- \item level::clone(), creates a deep copy of an object. Any shared data is
-duplicated.
- \item level::fill(), fill an object with a value (fig. \ref{fig:fill_impl}).
- \item level::paste(), paste object data to another object (fig.
-\ref{fig:paste_impl})
- \item labeling::blobs(), find and label the different components of an image.
-\end{itemize}
+\section{Basic routines}
+\begin{tabular}{|l|p{8cm}|}
+\hline
+Routine name & Description \\ \hline
+level::clone() & creates a deep copy of an object. Any shared data is
+duplicated. \\ \hline
+
+level::fill() & fill an object with a value. \\ \hline
+
+level::paste() & paste object data to another object. \\ \hline
+labeling::blobs() & find and label the different components of an image. \\
+\hline
+\end{tabular} \\
+
+
+\section{Examples}
First, create an image:
\begin{lstlisting}[frame=single]
image2d<char> imga(0, 0, 20, 20);
@@ -572,7 +566,7 @@
\caption{Implementation of the paste routine.\label{fig:paste_impl}}
\end{figure}
-
+\newpage
\section{Working with parts of an image}
Sometime it may be interesting to work only on some part of the image or to
@@ -604,6 +598,21 @@
by a function\_p2v to another Value.
The following sample code illustrates this feature.
+In order to use C functions as predicate, they must have one of the following
+prototype if you work on 2D images:
+\begin{lstlisting}[frame=single]
+//function_p2b
+bool my_function_p2b(mln::point2d p);
+
+//function_p2v
+//V is the value type used in the image.
+V my_function_p2v(mln::point2d p);
+\end{lstlisting}
+Of course, you just need to change the point type if you use another image
+type. For instance, you would use point3d with 3D images.
+The returned value type V for function\_p2v depends on the image value type.
+With image2d<int>, V would be int.
+
In this section, all along the examples, the image "ima" will refer to the
following declaration:
\begin{lstlisting}[frame=single]
@@ -675,9 +684,197 @@
fill(inplace(lab.domain(2)), color::red);
\end{lstlisting}
+Here is an example using a C function:
+\begin{lstlisting}[frame=single]
+bool row_oddity(mln::point2d p)
+{
+ return p.row() % 2;
+}
+
+void my_function()
+{
+ ...
+ //Prints only sites which are on odd lines.
+ debug::println(ima | row_oddity};
+ ...
+}
+\end{lstlisting}
\chapter{Graphes and images}
+\section{Description}
+Olena enables the possibility of using graphes with images.
+Graphes can help you to handle directly parts of an image and represent their
+relationship.
+Specific data can be associated to each vertex and/or edges.
+
+//FIXME: Add more explanations?
+
+\section{Example}
+
+First, create a graph which looks like the following:
+\begin{lstlisting}[frame=single]
+ 0 1 2 3 4
+ .-----------
+ 0 | 0 2
+ 1 | \ / |
+ 2 | 1 |
+ 3 | \ |
+ 4 | 3-4
+\end{lstlisting}
+
+To do so, we need to create a site set containing the sites we are going to use
+as vertices:
+
+\begin{lstlisting}
+std::vector<point2d> sites;
+
+// Site associated to...
+sites.push_back(point2d(0,0)); // ... vertex 0.
+sites.push_back(point2d(2,2)); // ... vertex 1.
+sites.push_back(point2d(0,4)); // ... vertex 2.
+sites.push_back(point2d(4,3)); // ... vertex 3.
+sites.push_back(point2d(4,4)); // ... vertex 4.
+\end{lstlisting}
+
+Then populate the graph with vertices:
+\begin{lstlisting}
+util::graph<point2d> g;
+
+for (unsigned i = 0; i < points.size(); ++i)
+ g.add_vertex (points[i]);
+\end{lstlisting}
+
+Finally, populate the graph with edges:
+\begin{lstlisting}
+g.add_edge(0, 1); // Associated to edge 0.
+g.add_edge(1, 2); // Associated to edge 1.
+g.add_edge(1, 3); // Associated to edge 2.
+g.add_edge(3, 4); // Associated to edge 3.
+g.add_edge(4, 2); // Associated to edge 4.
+\end{lstlisting}
+
+Now, a graph structure is created but contains only site relationship
+information. We may like to map data to it. Vertices and edges are mapped to
+indexes. Indexes start from 0 and respect the insertion order.
+Therefore, the idea is to provide a function which returns the associated data
+according to the index given as parameter. Combining this function and the
+graph, which is actually a sort of site set, we get an image. Since this is an
+image, you can use it with algorithms and for\_all loops.
+
+\begin{lstlisting}[frame=single]
+
+Value my_data_fun(vertex_index_t index)
+{
+ if (index == 0)
+ return value1;
+ else if (index > 1 && < 4)
+ return value2;
+ return value3;
+}
+
+Value my_data_fun(edge_index_t index)
+{
+ if (index == 0)
+ return value1;
+ else if (index > 1 && < 4)
+ return value2;
+ return value3;
+}
+
+void my_fun()
+{
+ ...
+ // Constructs an image which associates
+ // data and edges. A site is actually an edge.
+ mln_VAR(graph_edge_ima, my_data_fun | g.edges());
+
+ //Prints each edge.
+ mln_piter p(graph_edge_ima);
+ for_all(p)
+ std::cout << p << std::endl;
+
+ // Constructs an image which associates
+ // data and vertices. A site is actually a vertex.
+ mln_VAR(graph_vertices_ima, my_data_fun2 | g.vertices());
+
+ //Prints each vertex
+ mln_piter p(graph_vertices_ima);
+ for_all(p)
+ std::Cout << p << std::endl;
+ ...
+}
+\end{lstlisting}
+
+Note that like any image in Olena, graph images share their data. Therefore,
+while constructing a graph image from a graph and a function, the graph is not
+copied and this is NOT a costly operation.
+
+Like other images, graph images also have an overloaded operator() to access the
+data associated to a vertex or an edge.
+\begin{lstlisting}
+ //Prints each edge's associated value.
+ mln_piter p(graph_edge_ima);
+ for_all(p)
+ std::cout << graph_edge_ima(p) << std::endl;
+
+ //Prints each vertex's associated value.
+ mln_piter p(graph_vertex_ima);
+ for_all(p)
+ std::cout << graph_vertex_ima(p) << std::endl;
+\end{lstlisting}
+
+Of course, creating a graph image is not necessary and you can work directly
+with the graph and container/function mapping indexes and data.
+
+\begin{lstlisting}
+// Iterator on vertices.
+mln_Viter V(g);
+
+// Prints each vertex and its associated value.
+for_all(V)
+{
+ std::cout << V << " : " << my_data_fun(V) << std::endl;
+}
+\end{lstlisting}
+
+Graphes have iterators like any other site sets and also provide
+specific iterators in order to iterate over graphes in a more intuitive way.
+
+\begin{lstlisting}
+// Iterator on vertices.
+mln_Viter V(g);
+
+// Iterator on V's edges.
+mln_Eiter n(g, V);
+
+// Prints the graph
+// List all edges for each vertex.
+for_all(V)
+{
+ std::cout << V << " : ";
+ for_all(n)
+ std::cout << n << " ";
+ std::cout << std::endl;
+}
+
+// Iterator on edges.
+mln_Eiter E(g);
+
+// Iterator on vertices adjacent to E.
+mln_Viter n(g, E);
+
+// Prints the graph
+// List all adjacent vertices for each edge.
+for_all(E)
+{
+ std::cout << E << " : ";
+ for_all(n)
+ std::cout << n << " ";
+ std::cout << std::endl;
+}
+\end{lstlisting}
+//FIXME talk about p_vertices and p_edges.
\end{document}
\ No newline at end of file