<!--
    Licensed to the Apache Software  Foundation (ASF) under one or more
    contributor  license agreements.  See  the NOTICE  file distributed
    with  this  work  for  additional information  regarding  copyright
    ownership.   The ASF  licenses this  file to  you under  the Apache
    License, Version  2.0 (the  License); you may  not use  this file
    except in  compliance with the License.   You may obtain  a copy of
    the License at

    http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the  License is distributed on an  "AS IS" BASIS,
    WITHOUT  WARRANTIES OR CONDITIONS  OF ANY  KIND, either  express or
    implied.   See  the License  for  the  specific language  governing
    permissions and limitations under the License.

    Copyright 1999-2007 Rogue Wave Software, Inc.
-->

<HTML>
<HEAD>
<TITLE>&lt;queue&gt;</TITLE>
<LINK REL=StyleSheet HREF="../rw.css" TYPE="text/css" TITLE="Apache stdcxx Stylesheet"></HEAD>
<BODY BGCOLOR=#FFFFFF>
<A HREF="push-heap.html"><IMG SRC="images/bprev.gif" WIDTH=20 HEIGHT=21 ALT="Previous file" BORDER=O></A><A HREF="noframes.html"><IMG SRC="images/btop.gif" WIDTH=56 HEIGHT=21 ALT="Top of Document" BORDER=O></A><A HREF="booktoc.html"><IMG SRC="images/btoc.gif" WIDTH=56 HEIGHT=21 ALT="Contents" BORDER=O></A><A HREF="tindex.html"><IMG SRC="images/bindex.gif" WIDTH=56 HEIGHT=21 ALT="Index page" BORDER=O></A><A HREF="queue.html"><IMG SRC="images/bnext.gif" WIDTH=25 HEIGHT=21 ALT="Next file" BORDER=O></A><DIV CLASS="DOCUMENTNAME"><B>Apache C++ Standard Library Reference Guide</B></DIV>
<H2>&lt;queue&gt;</H2>
<P><B>Library:</B>&nbsp;&nbsp;<A HREF="2-7.html">Containers</A></P>

<PRE><HR><B><I>Header</I></B><HR></PRE>

<UL>
<LI><A HREF="#sec1">Local Index</A></LI>
<LI><A HREF="#sec2">Summary</A></LI>
<LI><A HREF="#sec3">Synopsis</A></LI>
<LI><A HREF="#sec4">See Also</A></LI>
<LI><A HREF="#sec5">Standards Conformance</A></LI>
</UL>
<A NAME="sec1"><H3>Local Index</H3></A>
No Entries
<A NAME="sec2"><H3>Summary</H3></A>
<P>The header <SAMP>&lt;queue&gt;</SAMP> is part of the Containers library of the C++ Standard Library. It defines the container adaptors, the class templates <B><I><A HREF="queue.html">queue</A></I></B> and <B><I><A HREF="priority-queue.html">priority_queue</A></I></B>, and a set of relational operators that C++ programs may use to perform equality and inequality tests on specializations of <B><I>queue</I></B>.</P>
<A NAME="sec3"><H3>Synopsis</H3></A>

<PRE>namespace std {
  template &lt;class T, class Container = deque&lt;T&gt; &gt; class <SAMP><A HREF="queue.html">queue</A></SAMP>;
  template &lt;class T, class Container&gt;
  bool operator==(const queue&lt;T, Container&gt;&amp;, 
                  const queue&lt;T, Container&gt;&amp;);
  template &lt;class T, class Container&gt;
  bool operator&lt;(const queue&lt;T, Container&gt;&amp;, 
                 const queue&lt;T, Container&gt;&amp;);
  template &lt;class T, class Container&gt;
  bool operator!=(const queue&lt;T, Container&gt;&amp;, 
                  const queue&lt;T, Container&gt;&amp;);
  template &lt;class T, class Container&gt;
  bool operator&gt;(const queue&lt;T, Container&gt;&amp;, 
                 const queue&lt;T, Container&gt;&amp;);
  template &lt;class T, class Container&gt;
  bool operator&gt;=(const queue&lt;T, Container&gt;&amp;, 
                  const queue&lt;T, Container&gt;&amp;);
  template &lt;class T, class Container&gt;
  bool operator&lt;=(const queue&lt;T, Container&gt;&amp;, 
                  const queue&lt;T, Container&gt;&amp;);

  template &lt;class T, class Container = vector&lt;T&gt;, 
          class Compare = less&lt;typename Container::value_type&gt; &gt;
  class <SAMP><A HREF="priority-queue.html">priority_queue</A></SAMP>;
}
</PRE>
<A NAME="sec4"><H3>See Also</H3></A>
<P><B><I><A HREF="queue.html">queue</A></I></B>, <B><I><A HREF="priority-queue.html">priority_queue</A></I></B>, <A HREF="containers.html">Containers</A></P>
<A NAME="sec5"><H3>Standards Conformance</H3></A>
<P><I>ISO/IEC 14882:1998 -- International Standard for Information Systems --Programming Language C++ 23.2</I></P>

<BR>
<HR>
<A HREF="push-heap.html"><IMG SRC="images/bprev.gif" WIDTH=20 HEIGHT=21 ALT="Previous file" BORDER=O></A><A HREF="noframes.html"><IMG SRC="images/btop.gif" WIDTH=56 HEIGHT=21 ALT="Top of Document" BORDER=O></A><A HREF="booktoc.html"><IMG SRC="images/btoc.gif" WIDTH=56 HEIGHT=21 ALT="Contents" BORDER=O></A><A HREF="tindex.html"><IMG SRC="images/bindex.gif" WIDTH=56 HEIGHT=21 ALT="Index page" BORDER=O></A><A HREF="queue.html"><IMG SRC="images/bnext.gif" WIDTH=20 HEIGHT=21 ALT="Next file" BORDER=O></A>

<!-- Google Analytics tracking code -->
<script src="http://www.google-analytics.com/urchin.js" type="text/javascript">
</script>
<script type="text/javascript">
    _uacct = "UA-1775151-1";
    urchinTracker();
</script>
<!-- end of Google Analytics tracking code -->

</BODY>
</HTML>
