You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
concurrentqueue/benchmarks/dlib/queue/queue_kernel_1.h

555 lines
13 KiB
C++

// 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_