MobinMQL/Include/Generic/Queue.mqh
2025-07-22 14:47:41 +03:00

419 lines
29 KiB
MQL5

//+------------------------------------------------------------------+
//| Queue.mqh |
//| Copyright 2000-2025, MetaQuotes Ltd. |
//| https://www.mql5.com |
//+------------------------------------------------------------------+
#include <Generic\Interfaces\ICollection.mqh>
#include <Generic\Internal\ArrayFunction.mqh>
#include <Generic\Internal\EqualFunction.mqh>
//+------------------------------------------------------------------+
//| Class CQueue<T> |
//| Usage: Represents a first-in, first-out collection of objects |
//+------------------------------------------------------------------+
template<typename T>
class CQueue: public ICollection<T>
{
protected:
T m_array[];
int m_size; // items in queue
int m_head; // first valid element in the queue
int m_tail; // last valid element in the queue
public:
CQueue(void);
CQueue(const int capacity);
CQueue(ICollection<T>&collection[]);
CQueue(const T &array[]);
CQueue(const CQueue &queue);
~CQueue(void);
bool operator=(const CQueue &queue);
//--- methods of filling data
bool Add(T value) override;
bool Enqueue(T value);
//--- methods of access to protected data
int Count(void) override;
int Capacity(void) const;
bool Contains(T item) override;
bool SetCapacity(const int capacity);
void TrimExcess(double threshold_k=0.9);
//--- methods of copy data from collection
int CopyTo(T &dst_array[],const int dst_start=0) override;
//--- methods of cleaning and removing
void Clear(void) override;
bool Remove(T item) override;
bool Dequeue(T &item);
T Dequeue(void);
bool Peek(T &item) const;
T Peek(void) const;
};
//+------------------------------------------------------------------+
//| Initializes a new instance of the CQueue<T> class that is empty |
//| and has the default initial capacity |
//+------------------------------------------------------------------+
template<typename T> CQueue::CQueue(void): m_size(0),m_head(0),m_tail(0)
{
}
//+------------------------------------------------------------------+
//| Initializes a new instance of the CQueue<T> class that is empty |
//| and has the specified initial capacity or the default initial |
//| capacity, whichever is greater |
//+------------------------------------------------------------------+
template<typename T> CQueue::CQueue(const int capacity): m_size(0),m_head(0),m_tail(0)
{
ArrayResize(m_array,capacity);
}
//+------------------------------------------------------------------+
//| Initializes a new instance of the CQueue<T> class that contains |
//| elements copied from the specified array and has sufficient |
//| capacity to accommodate the number of elements copied |
//+------------------------------------------------------------------+
template<typename T> CQueue::CQueue(const T &array[]): m_head(0)
{
m_size=ArrayCopy(m_array,array);
m_tail=m_size;
}
//+------------------------------------------------------------------+
//| Initializes a new instance of the CQueue<T> class that contains |
//| elements copied from the specified collection and has sufficient |
//| capacity to accommodate the number of elements copied |
//+------------------------------------------------------------------+
template<typename T> CQueue::CQueue(ICollection<T>*collection): m_size(0),m_head(0)
{
//--- check collection
if(CheckPointer(collection)!=POINTER_INVALID)
m_size=collection.CopyTo(m_array);
//--- set tail
m_tail=m_size;
}
//+------------------------------------------------------------------+
//| Copy constructor |
//+------------------------------------------------------------------+
template<typename T> CQueue::CQueue(const CQueue &queue)
{
this=queue;
}
//+------------------------------------------------------------------+
//| Destructor |
//+------------------------------------------------------------------+
template<typename T> CQueue::~CQueue(void)
{
}
//+------------------------------------------------------------------+
//| Copy operator |
//+------------------------------------------------------------------+
template<typename T> bool CQueue::operator=(const CQueue &queue)
{
if(&this!=&queue)
{
m_size=0;
m_head=0;
m_tail=0;
if(!SetCapacity(queue.m_size))
return(false);
m_size=queue.m_size;
int size=ArraySize(queue.m_array);
for(int i=0; i<queue.m_size; i++)
m_array[i]=queue.m_array[(queue.m_head+i)%size];
}
return(true);
}
//+------------------------------------------------------------------+
//| Inserts an value at the top of the CQueue<T> |
//+------------------------------------------------------------------+
template<typename T>
bool CQueue::Add(T value)
{
return(Enqueue(value));
}
//+------------------------------------------------------------------+
//| Gets the number of elements |
//+------------------------------------------------------------------+
template<typename T>
int CQueue::Count(void)
{
return(m_size);
}
//+------------------------------------------------------------------+
//| Removes all values from the CQueue<T> |
//+------------------------------------------------------------------+
template<typename T>
bool CQueue::Contains(T item)
{
int size=ArraySize(m_array);
//--- try to find item in array
for(int i=m_head,const end=m_head+m_size; i<end; i++)
{
//--- use default equality function
if(::Equals(m_array[i%size],item))
return(true);
}
return(false);
}
//+------------------------------------------------------------------+
//| Sets the capacity to the actual number of elements in the queue, |
//| if that number is less than a threshold value |
//+------------------------------------------------------------------+
template<typename T>
void CQueue::TrimExcess(double threshold_k=0.9)
{
if(MathIsValidNumber(threshold_k) && threshold_k>0.0)
{
//--- calculate threshold value
int threshold=(int)(ArraySize(m_array)*MathMin(threshold_k,1.0));
//--- set a сapacity equal to the size
if(m_size<threshold)
SetCapacity(m_size);
}
}
//+------------------------------------------------------------------+
//| Copies a range of elements from the queue to a compatible |
//| one-dimensional array |
//+------------------------------------------------------------------+
template<typename T>
int CQueue::CopyTo(T &dst_array[],const int dst_start=0)
{
//--- resize array
if((dst_start+m_size)>ArraySize(dst_array))
if(ArrayResize(dst_array,dst_start+m_size)<0)
return(0);
//--- copy queue elements from head to tail
if(m_tail>m_head)
return(ArrayCopy(dst_array,m_array,dst_start,m_head,m_size));
//--- copy queue elements from head to end
int num_copied=ArrayCopy(dst_array,m_array,dst_start,m_head,m_array.Size()-m_head);
//--- copy queue elements from beginning to tail
num_copied+=ArrayCopy(dst_array,m_array,dst_start+num_copied,0,m_tail);
//--- return number of copied elements
return(num_copied);
}
//+------------------------------------------------------------------+
//| Removes all values from the CQueue<T> |
//+------------------------------------------------------------------+
template<typename T>
void CQueue::Clear(void)
{
//--- check current size
if(m_size>0)
{
ArrayFree(m_array);
m_size=0;
m_head=0;
m_tail=0;
}
}
//+------------------------------------------------------------------+
//| Removes the first occurrence of a specific value from the stack |
//+------------------------------------------------------------------+
template<typename T>
bool CQueue::Remove(T item)
{
//--- check for queue size
if(!m_size)
return(false);
//--- find from head to tail
if(m_tail>m_head)
{
//--- try to find index of item
int index=ArrayIndexOf(m_array,item,m_head,m_size);
if(index!=-1)
{
m_size--;
//--- remove from queue head
if(index==m_head)
m_head++;
else
{
//--- correct tail
m_tail--;
if((m_size-index)!=0)
ArrayCopy(m_array,m_array,index,index+1,m_size-index);
}
return(true);
}
}
else
{
//--- try to find in head part
int head_size=m_size-m_tail;
int index =ArrayIndexOf(m_array,item,m_head,head_size);
if(index!=-1)
{
m_size--;
//--- remove from queue head
if(index==m_head)
m_head++;
else
{
//--- remove from head part
head_size--;
if((m_head+head_size-index)!=0)
ArrayCopy(m_array,m_array,index,index+1,m_head+head_size-index);
//--- correct tail
if(m_tail)
{
m_tail--;
m_array[m_head+head_size]=m_array[0];
if(m_tail!=0)
ArrayCopy(m_array,m_array,0,1,m_tail);
}
else
m_tail=m_head+head_size;
}
return(true);
}
//--- try to find in tail part
index=ArrayIndexOf(m_array,item,0,m_tail);
if(index!=-1)
{
m_tail--;
m_size--;
if((m_tail-index)!=0)
ArrayCopy(m_array,m_array,index,index+1,m_tail-index);
return(true);
}
}
//---
return(false);
}
//+------------------------------------------------------------------+
//| Adds an value to the end of the CQueue<T> |
//+------------------------------------------------------------------+
template<typename T>
bool CQueue::Enqueue(T value)
{
int size=ArraySize(m_array);
//--- check queue capacity
if(m_size==size)
{
//--- check for max size
if(size==INT_MAX)
return(false);
//--- calculate and set new capacity
int new_capacity=MathMax((int)MathMin(size/2ull + size,INT_MAX),16);
if(!SetCapacity(new_capacity))
return(false);
size=ArraySize(m_array);
}
//--- add value to the end
m_array[m_tail++]=value;
//--- increase size and recalculate tail
m_size++;
m_tail%=size;
return(true);
}
//+------------------------------------------------------------------+
//| Removes and returns the value at the beginning of the CQueue<T> |
//+------------------------------------------------------------------+
template<typename T>
bool CQueue::Dequeue(T &item)
{
//--- check queue is not empty
if(!m_size)
return(false);
//--- get value from the end
item=m_array[m_head++];
//--- decrement size and recalculate head
m_size--;
m_head%=ArraySize(m_array);
return(true);
}
//+------------------------------------------------------------------+
//| Removes and returns the value at the beginning of the CQueue<T> |
//+------------------------------------------------------------------+
template<typename T>
T CQueue::Dequeue(void)
{
int index=m_head;
//--- decrement size and recalculate head
m_size--;
m_head=(m_head+1)%ArraySize(m_array);
return(m_array[index]);
}
//+------------------------------------------------------------------+
//| Returns the value at the beginning of the CQueue<T> without |
//| removing |
//+------------------------------------------------------------------+
template<typename T>
bool CQueue::Peek(T &item) const
{
//--- check queue is not empty
if(!m_size)
return(false);
//--- get value from the end
item=m_array[m_head];
return(true);
}
//+------------------------------------------------------------------+
//| Returns the value at the beginning of the CQueue<T> without |
//| removing |
//+------------------------------------------------------------------+
template<typename T>
T CQueue::Peek(void) const
{
//--- get value from the end
return(m_array[m_head]);
}
//+------------------------------------------------------------------+
//| Grows or shrinks the buffer to hold capacity values |
//+------------------------------------------------------------------+
template<typename T>
bool CQueue::SetCapacity(const int capacity)
{
//--- create a new array array for temporary storage
T new_array[];
if(ArrayResize(new_array,capacity)!=capacity)
return(false);
//--- any items in queue
if(m_size>0)
{
if(m_head<m_tail)
{
ArrayCopy(new_array,m_array,0,m_head,m_size);
}
else
{
const int head_size=m_size-m_tail;
//--- copy values from head part
ArrayCopy(new_array,m_array,0,m_head,head_size);
//--- copy values from tail part
ArrayCopy(new_array,m_array,head_size,0,m_tail);
}
}
//--- set temporary buffer as queue buffer
ArraySwap(m_array,new_array);
//--- set new tail and head
m_head=0;
m_tail=m_size;
return(true);
}
//+------------------------------------------------------------------+
//| Returns queue capacity |
//+------------------------------------------------------------------+
template<typename T>
int CQueue::Capacity(void) const
{
return(ArraySize(m_array)-m_size);
}
//+------------------------------------------------------------------+