<!--
    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;deque&gt;</TITLE>
<LINK REL=StyleSheet HREF="../rw.css" TYPE="text/css" TITLE="Apache stdcxx Stylesheet"></HEAD>
<BODY BGCOLOR=#FFFFFF>
<A HREF="ctype-byname.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="deque.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;deque&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;deque&gt;</SAMP> is part of the Containers library of the C++ Standard Library. It defines the sequence class template <B><I><A HREF="deque.html">deque</A></I></B>, a set of relational operators that C++ programs may use to perform equality and inequality tests on specializations of <B><I>deque</I></B>, and the specialization of the <SAMP><A HREF="swap.html">swap()</A></SAMP> algorithm for <B><I>deque</I></B>.</P>
<A NAME="sec3"><H3>Synopsis</H3></A>

<PRE>namespace std {
  template &lt;class T, class Allocator = allocator&lt;T&gt; &gt; 
  class <SAMP><A HREF="deque.html">deque</A></SAMP>;
  template &lt;class T, class Allocator&gt;
  bool operator==(const deque&lt;T, Allocator&gt;&amp;, 
                  const deque&lt;T, Allocator&gt;&amp;);
  template &lt;class T, class Allocator&gt;
  bool operator&lt;(const deque&lt;T, Allocator&gt;&amp;, 
                 const deque&lt;T, Allocator&gt;&amp;);
  template &lt;class T, class Allocator&gt;
  bool operator!=(const deque&lt;T, Allocator&gt;&amp;, 
                  const deque&lt;T, Allocator&gt;&amp;);
  template &lt;class T, class Allocator&gt;
  bool operator&gt;(const deque&lt;T, Allocator&gt;&amp;, 
                 const deque&lt;T, Allocator&gt;&amp;);
  template &lt;class T, class Allocator&gt;
  bool operator&gt;=(const deque&lt;T, Allocator&gt;&amp;, 
                  const deque&lt;T, Allocator&gt;&amp;);
  template &lt;class T, class Allocator&gt;
  bool operator&lt;=(const deque&lt;T, Allocator&gt;&amp;, 
                  const deque&lt;T, Allocator&gt;&amp;);
  template &lt;class T, class Allocator&gt;
  void <SAMP><A HREF="swap.html">swap</A></SAMP>(deque&lt;T, Allocator&gt;&amp;, deque&lt;T, Allocator&gt;&amp;);
}
</PRE>
<A NAME="sec4"><H3>See Also</H3></A>
<P><B><I><A HREF="deque.html">deque</A></I></B>, <A HREF="containers.html">Containers</A>, <A HREF="sequences.html">Sequences</A>, <SAMP><A HREF="swap.html">swap()</A></SAMP></P>
<A NAME="sec5"><H3>Standards Conformance</H3></A>
<P><I>ISO/IEC 14882:1998 -- International Standard for Information Systems --Programming Language C++Section 23.2.1</I></P>

<BR>
<HR>
<A HREF="ctype-byname.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="deque.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>
