EffectiveModernCppChinese/3.MovingToModernCpp/item13.html
2022-07-02 13:21:48 +08:00

247 lines
26 KiB
HTML
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

<!DOCTYPE HTML>
<html lang="zh" class="sidebar-visible no-js light">
<head>
<!-- Book generated using mdBook -->
<meta charset="UTF-8">
<title>Item 13:优先考虑const_iterator而非iterator - Effective Modern C++</title>
<!-- Custom HTML head -->
<meta content="text/html; charset=utf-8" http-equiv="Content-Type">
<meta name="description" content="">
<meta name="viewport" content="width=device-width, initial-scale=1">
<meta name="theme-color" content="#ffffff" />
<link rel="icon" href="../favicon.svg">
<link rel="shortcut icon" href="../favicon.png">
<link rel="stylesheet" href="../css/variables.css">
<link rel="stylesheet" href="../css/general.css">
<link rel="stylesheet" href="../css/chrome.css">
<link rel="stylesheet" href="../css/print.css" media="print">
<!-- Fonts -->
<link rel="stylesheet" href="../FontAwesome/css/font-awesome.css">
<link rel="stylesheet" href="../fonts/fonts.css">
<!-- Highlight.js Stylesheets -->
<link rel="stylesheet" href="../highlight.css">
<link rel="stylesheet" href="../tomorrow-night.css">
<link rel="stylesheet" href="../ayu-highlight.css">
<!-- Custom theme stylesheets -->
<!-- MathJax -->
<script async type="text/javascript" src="https://cdnjs.cloudflare.com/ajax/libs/mathjax/2.7.1/MathJax.js?config=TeX-AMS-MML_HTMLorMML"></script>
</head>
<body>
<!-- Provide site root to javascript -->
<script type="text/javascript">
var path_to_root = "../";
var default_theme = window.matchMedia("(prefers-color-scheme: dark)").matches ? "navy" : "light";
</script>
<!-- Work around some values being stored in localStorage wrapped in quotes -->
<script type="text/javascript">
try {
var theme = localStorage.getItem('mdbook-theme');
var sidebar = localStorage.getItem('mdbook-sidebar');
if (theme.startsWith('"') && theme.endsWith('"')) {
localStorage.setItem('mdbook-theme', theme.slice(1, theme.length - 1));
}
if (sidebar.startsWith('"') && sidebar.endsWith('"')) {
localStorage.setItem('mdbook-sidebar', sidebar.slice(1, sidebar.length - 1));
}
} catch (e) { }
</script>
<!-- Set the theme before any content is loaded, prevents flash -->
<script type="text/javascript">
var theme;
try { theme = localStorage.getItem('mdbook-theme'); } catch(e) { }
if (theme === null || theme === undefined) { theme = default_theme; }
var html = document.querySelector('html');
html.classList.remove('no-js')
html.classList.remove('light')
html.classList.add(theme);
html.classList.add('js');
</script>
<!-- Hide / unhide sidebar before it is displayed -->
<script type="text/javascript">
var html = document.querySelector('html');
var sidebar = 'hidden';
if (document.body.clientWidth >= 1080) {
try { sidebar = localStorage.getItem('mdbook-sidebar'); } catch(e) { }
sidebar = sidebar || 'visible';
}
html.classList.remove('sidebar-visible');
html.classList.add("sidebar-" + sidebar);
</script>
<nav id="sidebar" class="sidebar" aria-label="Table of contents">
<div class="sidebar-scrollbox">
<ol class="chapter"><li class="chapter-item expanded "><a href="../Introduction.html">简介</a></li><li class="chapter-item expanded "><div>第一章 类型推导</div></li><li><ol class="section"><li class="chapter-item expanded "><a href="../1.DeducingTypes/item1.html">Item 1:理解模板类型推导</a></li><li class="chapter-item expanded "><a href="../1.DeducingTypes/item2.html">Item 2:理解auto类型推导</a></li><li class="chapter-item expanded "><a href="../1.DeducingTypes/item3.html">Item 3:理解decltype</a></li><li class="chapter-item expanded "><a href="../1.DeducingTypes/item4.html">Item 4:学会查看类型推导结果</a></li></ol></li><li class="chapter-item expanded "><div>第二章 auto</div></li><li><ol class="section"><li class="chapter-item expanded "><a href="../2.Auto/item5.html">Item 5:优先考虑auto而非显式类型声明</a></li><li class="chapter-item expanded "><a href="../2.Auto/item6.html">Item 6:auto推导若非己愿使用显式类型初始化惯用法</a></li></ol></li><li class="chapter-item expanded "><div>第三章 移步现代C++</div></li><li><ol class="section"><li class="chapter-item expanded "><a href="../3.MovingToModernCpp/item7.html">Item 7:区别使用()和{}创建对象</a></li><li class="chapter-item expanded "><a href="../3.MovingToModernCpp/item8.html">Item 8:优先考虑nullptr而非0和NULL</a></li><li class="chapter-item expanded "><a href="../3.MovingToModernCpp/item9.html">Item 9:优先考虑别名声明而非typedefs</a></li><li class="chapter-item expanded "><a href="../3.MovingToModernCpp/item10.html">Item 10:优先考虑限域枚举而非未限域枚举</a></li><li class="chapter-item expanded "><a href="../3.MovingToModernCpp/item11.html">Item 11:优先考虑使用deleted函数而非使用未定义的私有声明</a></li><li class="chapter-item expanded "><a href="../3.MovingToModernCpp/item12.html">Item 12:使用override声明重载函数</a></li><li class="chapter-item expanded "><a href="../3.MovingToModernCpp/item13.html" class="active">Item 13:优先考虑const_iterator而非iterator</a></li><li class="chapter-item expanded "><a href="../3.MovingToModernCpp/item14.html">Item 14:如果函数不抛出异常请使用noexcept</a></li><li class="chapter-item expanded "><a href="../3.MovingToModernCpp/item15.html">Item 15:尽可能的使用constexpr</a></li><li class="chapter-item expanded "><a href="../3.MovingToModernCpp/item16.html">Item 16:让const成员函数线程安全</a></li><li class="chapter-item expanded "><a href="../3.MovingToModernCpp/item17.html">Item 17:理解特殊成员函数函数的生成</a></li></ol></li><li class="chapter-item expanded "><div>第四章 智能指针</div></li><li><ol class="section"><li class="chapter-item expanded "><a href="../4.SmartPointers/item18.html">Item 18:对于独占资源使用std::unique_ptr</a></li><li class="chapter-item expanded "><a href="../4.SmartPointers/item19.html">Item 19:对于共享资源使用std::shared_ptr</a></li><li class="chapter-item expanded "><a href="../4.SmartPointers/item20.html">Item 20:当std::shard_ptr可能悬空时使用std::weak_ptr</a></li><li class="chapter-item expanded "><a href="../4.SmartPointers/item21.html">Item 21:优先考虑使用std::make_unique和std::make_shared而非new</a></li><li class="chapter-item expanded "><a href="../4.SmartPointers/item22.html">Item 22:当使用Pimpl惯用法请在实现文件中定义特殊成员函数</a></li></ol></li><li class="chapter-item expanded "><div>第五章 右值引用,移动语义,完美转发</div></li><li><ol class="section"><li class="chapter-item expanded "><a href="../5.RRefMovSemPerfForw/item23.html">Item 23:理解std::move和std::forward</a></li><li class="chapter-item expanded "><a href="../5.RRefMovSemPerfForw/item24.html">Item 24:区别通用引用和右值引用</a></li><li class="chapter-item expanded "><a href="../5.RRefMovSemPerfForw/item25.html">Item 25:对于右值引用使用std::move对于通用引用使用std::forward</a></li><li class="chapter-item expanded "><a href="../5.RRefMovSemPerfForw/item26.html">Item 26:避免重载通用引用</a></li><li class="chapter-item expanded "><a href="../5.RRefMovSemPerfForw/item27.html">Item 27:熟悉重载通用引用的替代品</a></li><li class="chapter-item expanded "><a href="../5.RRefMovSemPerfForw/item28.html">Item 28:理解引用折叠</a></li><li class="chapter-item expanded "><a href="../5.RRefMovSemPerfForw/item29.html">Item 29:认识移动操作的缺点</a></li><li class="chapter-item expanded "><a href="../5.RRefMovSemPerfForw/item30.html">Item 30:熟悉完美转发失败的情况</a></li></ol></li><li class="chapter-item expanded "><div>第六章 Lambda表达式</div></li><li><ol class="section"><li class="chapter-item expanded "><a href="../6.LambdaExpressions/item31.html">Item 31:避免使用默认捕获模式</a></li><li class="chapter-item expanded "><a href="../6.LambdaExpressions/item32.html">Item 32:使用初始化捕获来移动对象到闭包中</a></li><li class="chapter-item expanded "><a href="../6.LambdaExpressions/item33.html">Item 33:对于std::forward的auto&&形参使用decltype</a></li><li class="chapter-item expanded "><a href="../6.LambdaExpressions/item34.html">Item 34:优先考虑lambda表达式而非std::bind</a></li></ol></li><li class="chapter-item expanded "><div>第七章 并发API</div></li><li><ol class="section"><li class="chapter-item expanded "><a href="../7.TheConcurrencyAPI/Item35.html">Item 35:优先考虑基于任务的编程而非基于线程的编程</a></li><li class="chapter-item expanded "><a href="../7.TheConcurrencyAPI/item36.html">Item 36:如果有异步的必要请指定std::launch::threads</a></li><li class="chapter-item expanded "><a href="../7.TheConcurrencyAPI/item37.html">Item 37:从各个方面使得std::threads unjoinable</a></li><li class="chapter-item expanded "><a href="../7.TheConcurrencyAPI/item38.html">Item 38:关注不同线程句柄析构行为</a></li><li class="chapter-item expanded "><a href="../7.TheConcurrencyAPI/item39.html">Item 39:考虑对于单次事件通信使用void</a></li><li class="chapter-item expanded "><a href="../7.TheConcurrencyAPI/item40.html">Item 40:对于并发使用std::atomicvolatile用于特殊内存区</a></li></ol></li><li class="chapter-item expanded "><div>第八章 微调</div></li><li><ol class="section"><li class="chapter-item expanded "><a href="../8.Tweaks/item41.html">Item 41:对于那些可移动总是被拷贝的形参使用传值方式</a></li><li class="chapter-item expanded "><a href="../8.Tweaks/item42.html">Item 42:考虑就地创建而非插入</a></li></ol></li></ol>
</div>
<div id="sidebar-resize-handle" class="sidebar-resize-handle"></div>
</nav>
<div id="page-wrapper" class="page-wrapper">
<div class="page">
<div id="menu-bar-hover-placeholder"></div>
<div id="menu-bar" class="menu-bar sticky bordered">
<div class="left-buttons">
<button id="sidebar-toggle" class="icon-button" type="button" title="Toggle Table of Contents" aria-label="Toggle Table of Contents" aria-controls="sidebar">
<i class="fa fa-bars"></i>
</button>
<button id="theme-toggle" class="icon-button" type="button" title="Change theme" aria-label="Change theme" aria-haspopup="true" aria-expanded="false" aria-controls="theme-list">
<i class="fa fa-paint-brush"></i>
</button>
<ul id="theme-list" class="theme-popup" aria-label="Themes" role="menu">
<li role="none"><button role="menuitem" class="theme" id="light">Light (default)</button></li>
<li role="none"><button role="menuitem" class="theme" id="rust">Rust</button></li>
<li role="none"><button role="menuitem" class="theme" id="coal">Coal</button></li>
<li role="none"><button role="menuitem" class="theme" id="navy">Navy</button></li>
<li role="none"><button role="menuitem" class="theme" id="ayu">Ayu</button></li>
</ul>
<button id="search-toggle" class="icon-button" type="button" title="Search. (Shortkey: s)" aria-label="Toggle Searchbar" aria-expanded="false" aria-keyshortcuts="S" aria-controls="searchbar">
<i class="fa fa-search"></i>
</button>
</div>
<h1 class="menu-title">Effective Modern C++</h1>
<div class="right-buttons">
<a href="../print.html" title="Print this book" aria-label="Print this book">
<i id="print-button" class="fa fa-print"></i>
</a>
</div>
</div>
<div id="search-wrapper" class="hidden">
<form id="searchbar-outer" class="searchbar-outer">
<input type="search" id="searchbar" name="searchbar" placeholder="Search this book ..." aria-controls="searchresults-outer" aria-describedby="searchresults-header">
</form>
<div id="searchresults-outer" class="searchresults-outer hidden">
<div id="searchresults-header" class="searchresults-header"></div>
<ul id="searchresults">
</ul>
</div>
</div>
<!-- Apply ARIA attributes after the sidebar and the sidebar toggle button are added to the DOM -->
<script type="text/javascript">
document.getElementById('sidebar-toggle').setAttribute('aria-expanded', sidebar === 'visible');
document.getElementById('sidebar').setAttribute('aria-hidden', sidebar !== 'visible');
Array.from(document.querySelectorAll('#sidebar a')).forEach(function(link) {
link.setAttribute('tabIndex', sidebar === 'visible' ? 0 : -1);
});
</script>
<div id="content" class="content">
<main>
<h2 id="条款十三优先考虑const_iterator而非iterator"><a class="header" href="#条款十三优先考虑const_iterator而非iterator">条款十三:优先考虑<code>const_iterator</code>而非<code>iterator</code></a></h2>
<p><strong>Item 13: Prefer <code>const_iterators</code> to <code>iterators</code></strong></p>
<p>STL <code>const_iterator</code>等价于指向常量的指针pointer-to-<code>const</code>)。它们都指向不能被修改的值。标准实践是能加上<code>const</code>就加上,这也指示我们需要一个迭代器时只要没必要修改迭代器指向的值,就应当使用<code>const_iterator</code></p>
<p>上面的说法对C++11和C++98都是正确的但是在C++98中标准库对<code>const_iterator</code>的支持不是很完整。首先不容易创建它们,其次就算你有了它,它的使用也是受限的。假如你想在<code>std::vector&lt;int&gt;</code>中查找第一次出现1983C++代替C with classes的那一年的位置然后插入1998第一个ISO C++标准被接纳的那一年)。如果<em>vector</em>中没有1983那么就在<em>vector</em>尾部插入。在C++98中使用<code>iterator</code>可以很容易做到:</p>
<pre><code class="language-cpp">std::vector&lt;int&gt; values;
std::vector&lt;int&gt;::iterator it =
std::find(values.begin(), values.end(), 1983);
values.insert(it, 1998);
</code></pre>
<p>但是这里<code>iterator</code>真的不是一个好的选择,因为这段代码不修改<code>iterator</code>指向的内容。用<code>const_iterator</code>重写这段代码是很平常的但是在C++98中就不是了。下面是一种概念上可行但是不正确的方法</p>
<pre><code class="language-cpp">typedef std::vector&lt;int&gt;::iterator IterT; //typedef
typedef std::vector&lt;int&gt;::const_iterator ConstIterT;
std::vector&lt;int&gt; values;
ConstIterT ci =
std::find(static_cast&lt;ConstIterT&gt;(values.begin()), //cast
static_cast&lt;ConstIterT&gt;(values.end()), //cast
1983);
values.insert(static_cast&lt;IterT&gt;(ci), 1998); //可能无法通过编译,
//原因见下
</code></pre>
<p><code>typedef</code>不是强制的,但是可以让代码中的<em>cast</em>更好写。(你可能想知道为什么我使用<code>typedef</code>而不是<a href="https://github.com/kelthuzadx/EffectiveModernCppChinese/blob/master/3.MovingToModernCpp/item9.md">Item9</a>提到的别名声明因为这段代码在演示C++98做法别名声明是C++11加入的特性</p>
<p>之所以<code>std::find</code>的调用会出现类型转换是因为在C++98中<code>values</code>是non-<code>const</code>容器没办法简简单单的从non-<code>const</code>容器中获取<code>const_iterator</code>。严格来说类型转换不是必须的,因为用其他方法获取<code>const_iterator</code>也是可以的(比如你可以把<code>values</code>绑定到reference-to-<code>const</code>变量上,然后再用这个变量代替<code>values</code>但不管怎么说从non-<code>const</code>容器中获取<code>const_iterator</code>的做法都有点别扭。</p>
<p>当你费劲地获得了<code>const_iterator</code>事情可能会变得更糟因为C++98中插入操作以及删除操作的位置只能由<code>iterator</code>指定,<code>const_iterator</code>是不被接受的。这也是我在上面的代码中,将<code>const_iterator</code>(我那么小心地从<code>std::find</code>搞出来的东西)转换为<code>iterator</code>的原因,因为向<code>insert</code>传入<code>const_iterator</code>不能通过编译。</p>
<p>老实说,上面的代码也可能无法编译,因为没有一个可移植的从<code>const_iterator</code><code>iterator</code>的方法,即使使用<code>static_cast</code>也不行。甚至传说中的牛刀<code>reinterpret_cast</code>也杀不了这条鸡。它不是C++98的限制也不是C++11的限制只是<code>const_iterator</code>就是不能转换为<code>iterator</code>,不管看起来对它们施以转换是有多么合理。)不过有办法生成一个<code>iterator</code>,使其指向和<code>const_iterator</code>指向相同,但是看起来不明显,也没有广泛应用,在这本书也不值得讨论。除此之外,我希望目前我陈述的观点是清晰的:<code>const_iterator</code>在C++98中会有很多问题不如它的兄弟译注<code>iterator</code>)有用。最终,开发者们不再相信能加<code>const</code>就加它的教条而是只在实用的地方加它C++98的<code>const_iterator</code>不是那么实用。</p>
<p>所有的这些都在C++11中改变了现在<code>const_iterator</code>既容易获取又容易使用。容器的成员函数<code>cbegin</code><code>cend</code>产出<code>const_iterator</code>甚至对于non-<code>const</code>容器也可用,那些之前使用<em>iterator</em>指示位置(如<code>insert</code><code>erase</code>的STL成员函数也可以使用<code>const_iterator</code>了。使用C++11 <code>const_iterator</code>重写C++98使用<code>iterator</code>的代码也稀松平常:</p>
<pre><code class="language-cpp">std::vector&lt;int&gt; values; //和之前一样
auto it = //使用cbegin
std::find(values.cbegin(), values.cend(), 1983);//和cend
values.insert(it, 1998);
</code></pre>
<p>现在使用<code>const_iterator</code>的代码就很实用了!</p>
<p>唯一一个C++11对于<code>const_iterator</code>支持不足译注C++14支持但是C++11的时候还没的情况是当你想写最大程度通用的库并且这些库代码为一些容器和类似容器的数据结构提供<code>begin</code><code>end</code>(以及<code>cbegin</code><code>cend</code><code>rbegin</code><code>rend</code>等)作为<strong>非成员函数</strong>而不是成员函数时。其中一种情况就是原生数组,还有一种情况是一些只由自由函数组成接口的第三方库。(译注:自由函数<em>free function</em>,指的是非成员函数,即一个函数,只要不是成员函数就可被称作<em>free function</em>)最大程度通用的库会考虑使用非成员函数而不是假设成员函数版本存在。</p>
<p>举个例子,我们可以泛化下面的<code>findAndInsert</code></p>
<pre><code class="language-cpp">template&lt;typename C, typename V&gt;
void findAndInsert(C&amp; container, //在容器中查找第一次
const V&amp; targetVal, //出现targetVal的位置
const V&amp; insertVal) //然后在那插入insertVal
{
using std::cbegin;
using std::cend;
auto it = std::find(cbegin(container), //非成员函数cbegin
cend(container), //非成员函数cend
targetVal);
container.insert(it, insertVal);
}
</code></pre>
<p>它可以在C++14工作良好但是很遗憾C++11不在良好之列。由于标准化的疏漏C++11只添加了非成员函数<code>begin</code><code>end</code>,但是没有添加<code>cbegin</code><code>cend</code><code>rbegin</code><code>rend</code><code>crbegin</code><code>crend</code>。C++14修订了这个疏漏。</p>
<p>如果你使用C++11并且想写一个最大程度通用的代码而你使用的STL没有提供缺失的非成员函数<code>cbegin</code>和它的朋友们,你可以简单的写下你自己的实现。比如,下面就是非成员函数<code>cbegin</code>的实现:</p>
<pre><code class="language-cpp">template &lt;class C&gt;
auto cbegin(const C&amp; container)-&gt;decltype(std::begin(container))
{
return std::begin(container); //解释见下
}
</code></pre>
<p>你可能很惊讶非成员函数<code>cbegin</code>没有调用成员函数<code>cbegin</code>吧?我也是。但是请跟逻辑走。这个<code>cbegin</code>模板接受任何代表类似容器的数据结构的实参类型<code>C</code>并且通过reference-to-<code>const</code>形参<code>container</code>访问这个实参。如果<code>C</code>是一个普通的容器类型(如<code>std::vector&lt;int&gt;</code><code>container</code>将会引用一个<code>const</code>版本的容器(如<code>const std::vector&lt;int&gt;&amp;</code>)。对<code>const</code>容器调用非成员函数<code>begin</code>由C++11提供将产出<code>const_iterator</code>,这个迭代器也是模板要返回的。用这种方法实现的好处是就算容器只提供<code>begin</code>成员函数对于容器来说C++11的非成员函数<code>begin</code>调用这些成员函数)不提供<code>cbegin</code>成员函数也没问题。那么现在你可以将这个非成员函数<code>cbegin</code>施于只直接支持<code>begin</code>的容器。</p>
<p>如果<code>C</code>是原生数组,这个模板也能工作。这时,<code>container</code>成为一个<code>const</code>数组的引用。C++11为数组提供特化版本的非成员函数<code>begin</code>,它返回指向数组第一个元素的指针。一个<code>const</code>数组的元素也是<code>const</code>,所以对于<code>const</code>数组,非成员函数<code>begin</code>返回指向<code>const</code>的指针pointer-to-<code>const</code>)。在数组的上下文中,所谓指向<code>const</code>的指针pointer-to-<code>const</code>),也就是<code>const_iterator</code>了。</p>
<p>回到最开始,本条款的中心是鼓励你只要能就使用<code>const_iterator</code>。最原始的动机——只要它有意义就加上<code>const</code>——是C++98就有的思想。但是在C++98译注<code>const_iterator</code>只是一般有用到了C++11它就是极其有用了C++14在其基础上做了些修补工作。</p>
<p><strong>请记住:</strong></p>
<ul>
<li>优先考虑<code>const_iterator</code>而非<code>iterator</code></li>
<li>在最大程度通用的代码中,优先考虑非成员函数版本的<code>begin</code><code>end</code><code>rbegin</code>等,而非同名成员函数</li>
</ul>
</main>
<nav class="nav-wrapper" aria-label="Page navigation">
<!-- Mobile navigation buttons -->
<a rel="prev" href="../3.MovingToModernCpp/item12.html" class="mobile-nav-chapters previous" title="Previous chapter" aria-label="Previous chapter" aria-keyshortcuts="Left">
<i class="fa fa-angle-left"></i>
</a>
<a rel="next" href="../3.MovingToModernCpp/item14.html" class="mobile-nav-chapters next" title="Next chapter" aria-label="Next chapter" aria-keyshortcuts="Right">
<i class="fa fa-angle-right"></i>
</a>
<div style="clear: both"></div>
</nav>
</div>
</div>
<nav class="nav-wide-wrapper" aria-label="Page navigation">
<a rel="prev" href="../3.MovingToModernCpp/item12.html" class="nav-chapters previous" title="Previous chapter" aria-label="Previous chapter" aria-keyshortcuts="Left">
<i class="fa fa-angle-left"></i>
</a>
<a rel="next" href="../3.MovingToModernCpp/item14.html" class="nav-chapters next" title="Next chapter" aria-label="Next chapter" aria-keyshortcuts="Right">
<i class="fa fa-angle-right"></i>
</a>
</nav>
</div>
<script type="text/javascript">
window.playground_copyable = true;
</script>
<script src="../elasticlunr.min.js" type="text/javascript" charset="utf-8"></script>
<script src="../mark.min.js" type="text/javascript" charset="utf-8"></script>
<script src="../searcher.js" type="text/javascript" charset="utf-8"></script>
<script src="../clipboard.min.js" type="text/javascript" charset="utf-8"></script>
<script src="../highlight.js" type="text/javascript" charset="utf-8"></script>
<script src="../book.js" type="text/javascript" charset="utf-8"></script>
<!-- Custom JS scripts -->
</body>
</html>