dlib C++ Library - queue_kernel_1.h

// Copyright (C) 2003 Davis E. King (davis@dlib.net)
// License: Boost Software License See LICENSE.txt for the full license.
#ifndef DLIB_QUEUE_KERNEl_1_
#define DLIB_QUEUE_KERNEl_1_
#include "queue_kernel_abstract.h"
#include "../algs.h"
#include "../interfaces/enumerable.h"
#include "../interfaces/remover.h"
#include "../serialize.h"
namespace dlib
{
 template <
 typename T,
 typename mem_manager = default_memory_manager
 >
 class queue_kernel_1 : public enumerable<T>,
 public remover<T>
 {
 /*!
 INITIAL VALUE
 queue_size == 0 
 current_element == 0
 at_start_ == true
 
 CONVENTION
 queue_size == the number of elements in the queue
 at_start() == at_start_
 current_element_valid() == (current_element != 0)
 element() == current_element->item
 if (queue_size > 0)
 {
 in points to the last element to be inserted into the queue
 out points to the next element to be dequeued
 each node points to the node inserted after it except for the most 
 recently inserted node
 current_element == 0
 }
 
 !*/
 struct node
 {
 node* last;
 T item;
 };
 public:
 typedef T type;
 typedef mem_manager mem_manager_type;
 queue_kernel_1 (
 ) :
 in(0),
 out(0),
 queue_size(0),
 current_element(0),
 at_start_(true)
 {
 }
 virtual ~queue_kernel_1 (
 ); 
 inline void clear(
 );
 void enqueue (
 T& item
 );
 void dequeue (
 T& item
 );
 void cat (
 queue_kernel_1& item
 );
 T& current (
 );
 const T& current (
 ) const;
 
 void swap (
 queue_kernel_1& item
 );
 // functions from the remover interface
 inline void remove_any (
 T& item
 );
 // functions from the enumerable interface
 inline size_t size (
 ) const;
 inline bool at_start (
 ) const;
 inline void reset (
 ) const;
 bool current_element_valid (
 ) const;
 inline const T& element (
 ) const;
 inline T& element (
 );
 bool move_next (
 ) const;
 private:
 void delete_nodes (
 node* start,
 unsigned long length
 );
 /*!
 requires
 - start points to a node in a singly linked list 
 - start->last points to the next node in the list 
 - there are at least length nodes in the list beginning with start
 ensures
 - length nodes have been deleted starting with the node pointed 
 to by start
 !*/
 // data members
 node* in;
 node* out;
 unsigned long queue_size;
 mutable node* current_element;
 mutable bool at_start_;
 // restricted functions
 queue_kernel_1(queue_kernel_1&); // copy constructor
 queue_kernel_1& operator=(queue_kernel_1&); // assignment operator
 };
 template <
 typename T,
 typename mem_manager
 >
 inline void swap (
 queue_kernel_1<T,mem_manager>& a, 
 queue_kernel_1<T,mem_manager>& b 
 ) { a.swap(b); } 
 template <
 typename T,
 typename mem_manager
 >
 void deserialize (
 queue_kernel_1<T,mem_manager>& item, 
 std::istream& in
 )
 {
 try
 {
 item.clear();
 unsigned long size;
 deserialize(size,in);
 T temp;
 for (unsigned long i = 0; i < size; ++i)
 {
 deserialize(temp,in);
 item.enqueue(temp);
 }
 }
 catch (serialization_error& e)
 { 
 item.clear();
 throw serialization_error(e.info + "\n while deserializing object of type queue_kernel_1"); 
 }
 }
// ----------------------------------------------------------------------------------------
// ----------------------------------------------------------------------------------------
 // member function definitions
// ----------------------------------------------------------------------------------------
// ----------------------------------------------------------------------------------------
 template <
 typename T,
 typename mem_manager
 >
 queue_kernel_1<T,mem_manager>::
 ~queue_kernel_1 (
 )
 {
 delete_nodes(out,queue_size);
 }
// ----------------------------------------------------------------------------------------
 template <
 typename T,
 typename mem_manager
 >
 void queue_kernel_1<T,mem_manager>::
 clear (
 )
 {
 delete_nodes(out,queue_size);
 queue_size = 0;
 // put the enumerator at the start
 reset();
 }
// ----------------------------------------------------------------------------------------
 template <
 typename T,
 typename mem_manager
 >
 void queue_kernel_1<T,mem_manager>::
 enqueue (
 T& item
 )
 {
 // make new node
 node* temp = new node;
 // swap item into new node
 exchange(item,temp->item);
 
 if (queue_size == 0)
 out = temp;
 else
 in->last = temp;
 // make in point to the new node
 in = temp;
 
 ++queue_size;
 // put the enumerator at the start
 reset();
 }
// ----------------------------------------------------------------------------------------
 template <
 typename T,
 typename mem_manager
 >
 void queue_kernel_1<T,mem_manager>::
 dequeue (
 T& item
 )
 {
 // swap out into item
 exchange(item,out->item);
 --queue_size;
 if (queue_size == 0)
 {
 delete out;
 }
 else
 {
 node* temp = out;
 
 // move out pointer to the next element in the queue
 out = out->last;
 // delete old node
 delete temp;
 }
 // put the enumerator at the start
 reset();
 }
// ----------------------------------------------------------------------------------------
 template <
 typename T,
 typename mem_manager
 >
 void queue_kernel_1<T,mem_manager>::
 cat (
 queue_kernel_1<T,mem_manager>& item
 )
 {
 if (item.queue_size > 0)
 {
 if (queue_size > 0)
 {
 in->last = item.out;
 }
 else
 {
 out = item.out;
 }
 in = item.in;
 queue_size += item.queue_size;
 item.queue_size = 0;
 }
 // put the enumerator at the start
 reset();
 }
// ----------------------------------------------------------------------------------------
 template <
 typename T,
 typename mem_manager
 >
 T& queue_kernel_1<T,mem_manager>::
 current (
 )
 {
 return out->item;
 }
// ----------------------------------------------------------------------------------------
 template <
 typename T,
 typename mem_manager
 >
 const T& queue_kernel_1<T,mem_manager>::
 current (
 ) const
 {
 return out->item;
 }
// ----------------------------------------------------------------------------------------
 template <
 typename T,
 typename mem_manager
 >
 void queue_kernel_1<T,mem_manager>::
 swap (
 queue_kernel_1<T,mem_manager>& item
 )
 {
 node* in_temp = in;
 node* out_temp = out;
 unsigned long queue_size_temp = queue_size;
 node* current_element_temp = current_element;
 bool at_start_temp = at_start_;
 in = item.in;
 out = item.out;
 queue_size = item.queue_size;
 current_element = item.current_element;
 at_start_ = item.at_start_;
 item.in = in_temp;
 item.out = out_temp;
 item.queue_size = queue_size_temp;
 item.current_element = current_element_temp;
 item.at_start_ = at_start_temp;
 }
// ----------------------------------------------------------------------------------------
// ----------------------------------------------------------------------------------------
 // enumerable function definitions
// ----------------------------------------------------------------------------------------
// ----------------------------------------------------------------------------------------
 template <
 typename T,
 typename mem_manager
 >
 bool queue_kernel_1<T,mem_manager>::
 at_start (
 ) const
 {
 return at_start_;
 }
// ----------------------------------------------------------------------------------------
 template <
 typename T,
 typename mem_manager
 >
 size_t queue_kernel_1<T,mem_manager>::
 size (
 ) const
 {
 return queue_size;
 }
// ----------------------------------------------------------------------------------------
 template <
 typename T,
 typename mem_manager
 >
 void queue_kernel_1<T,mem_manager>::
 reset (
 ) const
 {
 at_start_ = true;
 current_element = 0;
 }
// ----------------------------------------------------------------------------------------
 template <
 typename T,
 typename mem_manager
 >
 bool queue_kernel_1<T,mem_manager>::
 current_element_valid (
 ) const
 {
 return (current_element != 0);
 }
// ----------------------------------------------------------------------------------------
 template <
 typename T,
 typename mem_manager
 >
 const T& queue_kernel_1<T,mem_manager>::
 element (
 ) const
 {
 return current_element->item;
 }
// ----------------------------------------------------------------------------------------
 template <
 typename T,
 typename mem_manager
 >
 T& queue_kernel_1<T,mem_manager>::
 element (
 )
 {
 return current_element->item;
 }
// ----------------------------------------------------------------------------------------
 template <
 typename T,
 typename mem_manager
 >
 bool queue_kernel_1<T,mem_manager>::
 move_next (
 ) const
 {
 if (at_start_)
 {
 at_start_ = false;
 // if the queue is empty then there is nothing to do
 if (queue_size == 0)
 {
 return false;
 }
 else
 {
 current_element = out;
 return true;
 }
 }
 else
 {
 // if we are at the last element then the enumeration has finished
 if (current_element == in || current_element == 0)
 {
 current_element = 0;
 return false;
 }
 else
 {
 current_element = current_element->last;
 return true;
 } 
 }
 }
// ----------------------------------------------------------------------------------------
// ----------------------------------------------------------------------------------------
 // remover function definitions
// ----------------------------------------------------------------------------------------
// ----------------------------------------------------------------------------------------
 template <
 typename T,
 typename mem_manager
 >
 void queue_kernel_1<T,mem_manager>::
 remove_any (
 T& item
 ) 
 {
 dequeue(item);
 }
// ----------------------------------------------------------------------------------------
// ----------------------------------------------------------------------------------------
 // private member function definitions
// ----------------------------------------------------------------------------------------
// ----------------------------------------------------------------------------------------
 template <
 typename T,
 typename mem_manager
 >
 void queue_kernel_1<T,mem_manager>::
 delete_nodes (
 node* start,
 unsigned long length
 )
 {
 node* temp;
 while (length)
 {
 temp = start->last;
 delete start;
 start = temp;
 --length;
 }
 }
// ----------------------------------------------------------------------------------------
}
#endif // DLIB_QUEUE_KERNEl_1_

AltStyle によって変換されたページ (->オリジナル) /