<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml">
<head>
<meta http-equiv="Content-Type" content="text/xhtml;charset=UTF-8"/>
<meta http-equiv="X-UA-Compatible" content="IE=9"/>
<meta name="generator" content="Doxygen 1.8.14"/>
<meta name="viewport" content="width=device-width, initial-scale=1"/>
<title>Mrpt: cpp/Mrpt.h Source File</title>
<link href="tabs.css" rel="stylesheet" type="text/css"/>
<script type="text/javascript" src="jquery.js"></script>
<script type="text/javascript" src="dynsections.js"></script>
<link href="search/search.css" rel="stylesheet" type="text/css"/>
<script type="text/javascript" src="search/searchdata.js"></script>
<script type="text/javascript" src="search/search.js"></script>
<link href="doxygen.css" rel="stylesheet" type="text/css" />
</head>
<body>
<div id="top"><!-- do not remove this div, it is closed by doxygen! -->
<div id="titlearea">
<table cellspacing="0" cellpadding="0">
 <tbody>
 <tr style="height: 56px;">
  <td id="projectlogo"><img alt="Logo" src="extree3.png"/></td>
  <td id="projectalign" style="padding-left: 0.5em;">
   <div id="projectname">Mrpt
   &#160;<span id="projectnumber">1.1.1</span>
   </div>
  </td>
 </tr>
 </tbody>
</table>
</div>
<!-- end header part -->
<!-- Generated by Doxygen 1.8.14 -->
<script type="text/javascript">
/* @license magnet:?xt=urn:btih:cf05388f2679ee054f2beb29a391d25f4e673ac3&amp;dn=gpl-2.0.txt GPL-v2 */
var searchBox = new SearchBox("searchBox", "search",false,'Search');
/* @license-end */
</script>
<script type="text/javascript" src="menudata.js"></script>
<script type="text/javascript" src="menu.js"></script>
<script type="text/javascript">
/* @license magnet:?xt=urn:btih:cf05388f2679ee054f2beb29a391d25f4e673ac3&amp;dn=gpl-2.0.txt GPL-v2 */
$(function() {
  initMenu('',true,false,'search.php','Search');
  $(document).ready(function() { init_search(); });
});
/* @license-end */</script>
<div id="main-nav"></div>
<!-- window showing the filter options -->
<div id="MSearchSelectWindow"
     onmouseover="return searchBox.OnSearchSelectShow()"
     onmouseout="return searchBox.OnSearchSelectHide()"
     onkeydown="return searchBox.OnSearchSelectKey(event)">
</div>

<!-- iframe showing the search results (closed by default) -->
<div id="MSearchResultsWindow">
<iframe src="javascript:void(0)" frameborder="0" 
        name="MSearchResults" id="MSearchResults">
</iframe>
</div>

<div id="nav-path" class="navpath">
  <ul>
<li class="navelem"><a class="el" href="dir_df511e5bd85cec96854b39d5e1c27aa8.html">cpp</a></li>  </ul>
</div>
</div><!-- top -->
<div class="header">
  <div class="headertitle">
<div class="title">Mrpt.h</div>  </div>
</div><!--header-->
<div class="contents">
<div class="fragment"><div class="line"><a name="l00001"></a><span class="lineno">    1</span>&#160;<span class="preprocessor">#ifndef CPP_MRPT_H_</span></div><div class="line"><a name="l00002"></a><span class="lineno">    2</span>&#160;<span class="preprocessor">#define CPP_MRPT_H_</span></div><div class="line"><a name="l00003"></a><span class="lineno">    3</span>&#160;</div><div class="line"><a name="l00004"></a><span class="lineno">    4</span>&#160;<span class="preprocessor">#include &lt;algorithm&gt;</span></div><div class="line"><a name="l00005"></a><span class="lineno">    5</span>&#160;<span class="preprocessor">#include &lt;cmath&gt;</span></div><div class="line"><a name="l00006"></a><span class="lineno">    6</span>&#160;<span class="preprocessor">#include &lt;functional&gt;</span></div><div class="line"><a name="l00007"></a><span class="lineno">    7</span>&#160;<span class="preprocessor">#include &lt;map&gt;</span></div><div class="line"><a name="l00008"></a><span class="lineno">    8</span>&#160;<span class="preprocessor">#include &lt;numeric&gt;</span></div><div class="line"><a name="l00009"></a><span class="lineno">    9</span>&#160;<span class="preprocessor">#include &lt;random&gt;</span></div><div class="line"><a name="l00010"></a><span class="lineno">   10</span>&#160;<span class="preprocessor">#include &lt;set&gt;</span></div><div class="line"><a name="l00011"></a><span class="lineno">   11</span>&#160;<span class="preprocessor">#include &lt;stdexcept&gt;</span></div><div class="line"><a name="l00012"></a><span class="lineno">   12</span>&#160;<span class="preprocessor">#include &lt;string&gt;</span></div><div class="line"><a name="l00013"></a><span class="lineno">   13</span>&#160;<span class="preprocessor">#include &lt;utility&gt;</span></div><div class="line"><a name="l00014"></a><span class="lineno">   14</span>&#160;<span class="preprocessor">#include &lt;vector&gt;</span></div><div class="line"><a name="l00015"></a><span class="lineno">   15</span>&#160;</div><div class="line"><a name="l00016"></a><span class="lineno">   16</span>&#160;<span class="preprocessor">#include &lt;Eigen/Dense&gt;</span></div><div class="line"><a name="l00017"></a><span class="lineno">   17</span>&#160;<span class="preprocessor">#include &lt;Eigen/SparseCore&gt;</span></div><div class="line"><a name="l00018"></a><span class="lineno">   18</span>&#160;</div><div class="line"><a name="l00019"></a><span class="lineno"><a class="line" href="struct_mrpt___parameters.html">   19</a></span>&#160;<span class="keyword">struct </span><a class="code" href="struct_mrpt___parameters.html">Mrpt_Parameters</a> {</div><div class="line"><a name="l00020"></a><span class="lineno"><a class="line" href="struct_mrpt___parameters.html#a4a0ab518bfa3786b59e58481bb7de814">   20</a></span>&#160;  <span class="keywordtype">int</span> <a class="code" href="struct_mrpt___parameters.html#a4a0ab518bfa3786b59e58481bb7de814">n_trees</a> = 0; </div><div class="line"><a name="l00021"></a><span class="lineno"><a class="line" href="struct_mrpt___parameters.html#a74ecd1a5f8320ec0cb30d2bfb7979fd0">   21</a></span>&#160;  <span class="keywordtype">int</span> <a class="code" href="struct_mrpt___parameters.html#a74ecd1a5f8320ec0cb30d2bfb7979fd0">depth</a> = 0; </div><div class="line"><a name="l00022"></a><span class="lineno"><a class="line" href="struct_mrpt___parameters.html#ad8b374d6671223a498e67cb98d8ea191">   22</a></span>&#160;  <span class="keywordtype">int</span> <a class="code" href="struct_mrpt___parameters.html#ad8b374d6671223a498e67cb98d8ea191">k</a> = 0; </div><div class="line"><a name="l00023"></a><span class="lineno"><a class="line" href="struct_mrpt___parameters.html#ab34a16df707e0abd7b8b0bf4959f36b6">   23</a></span>&#160;  <span class="keywordtype">int</span> <a class="code" href="struct_mrpt___parameters.html#ab34a16df707e0abd7b8b0bf4959f36b6">votes</a> = 0; </div><div class="line"><a name="l00024"></a><span class="lineno"><a class="line" href="struct_mrpt___parameters.html#a35ec37c03c3ff33e3dc6a9944d581327">   24</a></span>&#160;  <span class="keywordtype">double</span> <a class="code" href="struct_mrpt___parameters.html#a35ec37c03c3ff33e3dc6a9944d581327">estimated_qtime</a> = 0.0; </div><div class="line"><a name="l00025"></a><span class="lineno"><a class="line" href="struct_mrpt___parameters.html#ab2fd9db6c27b3024a4e2990c2b4b52ca">   25</a></span>&#160;  <span class="keywordtype">double</span> <a class="code" href="struct_mrpt___parameters.html#ab2fd9db6c27b3024a4e2990c2b4b52ca">estimated_recall</a> = 0.0; </div><div class="line"><a name="l00026"></a><span class="lineno">   26</span>&#160;};</div><div class="line"><a name="l00027"></a><span class="lineno">   27</span>&#160;</div><div class="line"><a name="l00028"></a><span class="lineno"><a class="line" href="class_mrpt.html">   28</a></span>&#160;<span class="keyword">class </span><a class="code" href="class_mrpt.html">Mrpt</a> {</div><div class="line"><a name="l00029"></a><span class="lineno">   29</span>&#160; <span class="keyword">public</span>:</div><div class="line"><a name="l00049"></a><span class="lineno"><a class="line" href="class_mrpt.html#a1e77a5c97c6ae94d17162f65041821bc">   49</a></span>&#160;    <a class="code" href="class_mrpt.html#a1e77a5c97c6ae94d17162f65041821bc">Mrpt</a>(<span class="keyword">const</span> Eigen::Ref&lt;const Eigen::MatrixXf&gt; &amp;X_) :</div><div class="line"><a name="l00050"></a><span class="lineno">   50</span>&#160;        X(Eigen::Map&lt;const Eigen::MatrixXf&gt;(X_.data(), X_.rows(), X_.cols())),</div><div class="line"><a name="l00051"></a><span class="lineno">   51</span>&#160;        n_samples(X_.cols()),</div><div class="line"><a name="l00052"></a><span class="lineno">   52</span>&#160;        dim(X_.rows()) {}</div><div class="line"><a name="l00053"></a><span class="lineno">   53</span>&#160;</div><div class="line"><a name="l00060"></a><span class="lineno"><a class="line" href="class_mrpt.html#a4e4195e845c91014df816f4e27a38b48">   60</a></span>&#160;    <a class="code" href="class_mrpt.html#a4e4195e845c91014df816f4e27a38b48">Mrpt</a>(<span class="keyword">const</span> <span class="keywordtype">float</span> *X_, <span class="keywordtype">int</span> dim_, <span class="keywordtype">int</span> n_samples_) :</div><div class="line"><a name="l00061"></a><span class="lineno">   61</span>&#160;        X(Eigen::Map&lt;const Eigen::MatrixXf&gt;(X_, dim_, n_samples_)),</div><div class="line"><a name="l00062"></a><span class="lineno">   62</span>&#160;        n_samples(n_samples_),</div><div class="line"><a name="l00063"></a><span class="lineno">   63</span>&#160;        dim(dim_) {}</div><div class="line"><a name="l00064"></a><span class="lineno">   64</span>&#160;</div><div class="line"><a name="l00084"></a><span class="lineno"><a class="line" href="class_mrpt.html#a7d789c72d12dd668fc5ffb157a9a52f4">   84</a></span>&#160;    <span class="keywordtype">void</span> <a class="code" href="class_mrpt.html#a7d789c72d12dd668fc5ffb157a9a52f4">grow</a>(<span class="keywordtype">int</span> n_trees_, <span class="keywordtype">int</span> depth_, <span class="keywordtype">float</span> density_ = -1.0, <span class="keywordtype">int</span> seed = 0) {</div><div class="line"><a name="l00085"></a><span class="lineno">   85</span>&#160;</div><div class="line"><a name="l00086"></a><span class="lineno">   86</span>&#160;      <span class="keywordflow">if</span> (!<a class="code" href="class_mrpt.html#a960f1af81f2b56db7eb8cd0d5a352c14">empty</a>()) {</div><div class="line"><a name="l00087"></a><span class="lineno">   87</span>&#160;        <span class="keywordflow">throw</span> std::logic_error(<span class="stringliteral">&quot;The index has already been grown.&quot;</span>);</div><div class="line"><a name="l00088"></a><span class="lineno">   88</span>&#160;      }</div><div class="line"><a name="l00089"></a><span class="lineno">   89</span>&#160;</div><div class="line"><a name="l00090"></a><span class="lineno">   90</span>&#160;      <span class="keywordflow">if</span> (n_trees_ &lt;= 0) {</div><div class="line"><a name="l00091"></a><span class="lineno">   91</span>&#160;        <span class="keywordflow">throw</span> std::out_of_range(<span class="stringliteral">&quot;The number of trees must be positive.&quot;</span>);</div><div class="line"><a name="l00092"></a><span class="lineno">   92</span>&#160;      }</div><div class="line"><a name="l00093"></a><span class="lineno">   93</span>&#160;</div><div class="line"><a name="l00094"></a><span class="lineno">   94</span>&#160;      <span class="keywordflow">if</span> (depth_ &lt;= 0 || depth_ &gt; std::log2(n_samples)) {</div><div class="line"><a name="l00095"></a><span class="lineno">   95</span>&#160;        <span class="keywordflow">throw</span> std::out_of_range(<span class="stringliteral">&quot;The depth must belong to the set {1, ... , log2(n)}.&quot;</span>);</div><div class="line"><a name="l00096"></a><span class="lineno">   96</span>&#160;      }</div><div class="line"><a name="l00097"></a><span class="lineno">   97</span>&#160;</div><div class="line"><a name="l00098"></a><span class="lineno">   98</span>&#160;      <span class="keywordflow">if</span> (density_ &lt; -1.0001 || density_ &gt; 1.0001 || (density_ &gt; -0.9999 &amp;&amp; density_ &lt; -0.0001)) {</div><div class="line"><a name="l00099"></a><span class="lineno">   99</span>&#160;        <span class="keywordflow">throw</span> std::out_of_range(<span class="stringliteral">&quot;The density must be on the interval (0,1].&quot;</span>);</div><div class="line"><a name="l00100"></a><span class="lineno">  100</span>&#160;      }</div><div class="line"><a name="l00101"></a><span class="lineno">  101</span>&#160;</div><div class="line"><a name="l00102"></a><span class="lineno">  102</span>&#160;      n_trees = n_trees_;</div><div class="line"><a name="l00103"></a><span class="lineno">  103</span>&#160;      depth = depth_;</div><div class="line"><a name="l00104"></a><span class="lineno">  104</span>&#160;      n_pool = n_trees_ * depth_;</div><div class="line"><a name="l00105"></a><span class="lineno">  105</span>&#160;      n_array = 1 &lt;&lt; (depth_ + 1);</div><div class="line"><a name="l00106"></a><span class="lineno">  106</span>&#160;</div><div class="line"><a name="l00107"></a><span class="lineno">  107</span>&#160;      <span class="keywordflow">if</span> (density_ &lt; 0) {</div><div class="line"><a name="l00108"></a><span class="lineno">  108</span>&#160;        density = 1.0 / std::sqrt(dim);</div><div class="line"><a name="l00109"></a><span class="lineno">  109</span>&#160;      } <span class="keywordflow">else</span> {</div><div class="line"><a name="l00110"></a><span class="lineno">  110</span>&#160;        density = density_;</div><div class="line"><a name="l00111"></a><span class="lineno">  111</span>&#160;      }</div><div class="line"><a name="l00112"></a><span class="lineno">  112</span>&#160;</div><div class="line"><a name="l00113"></a><span class="lineno">  113</span>&#160;      density &lt; 1 ? build_sparse_random_matrix(sparse_random_matrix, n_pool, dim, density, seed) :</div><div class="line"><a name="l00114"></a><span class="lineno">  114</span>&#160;                    build_dense_random_matrix(dense_random_matrix, n_pool, dim, seed);</div><div class="line"><a name="l00115"></a><span class="lineno">  115</span>&#160;</div><div class="line"><a name="l00116"></a><span class="lineno">  116</span>&#160;      split_points = Eigen::MatrixXf(n_array, n_trees);</div><div class="line"><a name="l00117"></a><span class="lineno">  117</span>&#160;      tree_leaves = std::vector&lt;std::vector&lt;int&gt;&gt;(n_trees);</div><div class="line"><a name="l00118"></a><span class="lineno">  118</span>&#160;</div><div class="line"><a name="l00119"></a><span class="lineno">  119</span>&#160;      count_first_leaf_indices_all(leaf_first_indices_all, n_samples, depth);</div><div class="line"><a name="l00120"></a><span class="lineno">  120</span>&#160;      leaf_first_indices = leaf_first_indices_all[depth];</div><div class="line"><a name="l00121"></a><span class="lineno">  121</span>&#160;</div><div class="line"><a name="l00122"></a><span class="lineno">  122</span>&#160;<span class="preprocessor">      #pragma omp parallel for</span></div><div class="line"><a name="l00123"></a><span class="lineno">  123</span>&#160;      <span class="keywordflow">for</span> (<span class="keywordtype">int</span> n_tree = 0; n_tree &lt; n_trees; ++n_tree) {</div><div class="line"><a name="l00124"></a><span class="lineno">  124</span>&#160;        Eigen::MatrixXf tree_projections;</div><div class="line"><a name="l00125"></a><span class="lineno">  125</span>&#160;</div><div class="line"><a name="l00126"></a><span class="lineno">  126</span>&#160;        <span class="keywordflow">if</span> (density &lt; 1)</div><div class="line"><a name="l00127"></a><span class="lineno">  127</span>&#160;          tree_projections.noalias() = sparse_random_matrix.middleRows(n_tree * depth, depth) * X;</div><div class="line"><a name="l00128"></a><span class="lineno">  128</span>&#160;        <span class="keywordflow">else</span></div><div class="line"><a name="l00129"></a><span class="lineno">  129</span>&#160;          tree_projections.noalias() = dense_random_matrix.middleRows(n_tree * depth, depth) * X;</div><div class="line"><a name="l00130"></a><span class="lineno">  130</span>&#160;</div><div class="line"><a name="l00131"></a><span class="lineno">  131</span>&#160;        tree_leaves[n_tree] = std::vector&lt;int&gt;(n_samples);</div><div class="line"><a name="l00132"></a><span class="lineno">  132</span>&#160;        std::vector&lt;int&gt; &amp;indices = tree_leaves[n_tree];</div><div class="line"><a name="l00133"></a><span class="lineno">  133</span>&#160;        std::iota(indices.begin(), indices.end(), 0);</div><div class="line"><a name="l00134"></a><span class="lineno">  134</span>&#160;</div><div class="line"><a name="l00135"></a><span class="lineno">  135</span>&#160;        grow_subtree(indices.begin(), indices.end(), 0, 0, n_tree, tree_projections);</div><div class="line"><a name="l00136"></a><span class="lineno">  136</span>&#160;      }</div><div class="line"><a name="l00137"></a><span class="lineno">  137</span>&#160;    }</div><div class="line"><a name="l00138"></a><span class="lineno">  138</span>&#160;</div><div class="line"><a name="l00178"></a><span class="lineno"><a class="line" href="class_mrpt.html#a7360964458339aab0ddedde324a03e47">  178</a></span>&#160;    <span class="keywordtype">void</span> <a class="code" href="class_mrpt.html#a7360964458339aab0ddedde324a03e47">grow</a>(<span class="keywordtype">double</span> target_recall, <span class="keyword">const</span> Eigen::Ref&lt;const Eigen::MatrixXf&gt; &amp;Q, <span class="keywordtype">int</span> k_, <span class="keywordtype">int</span> trees_max = -1,</div><div class="line"><a name="l00179"></a><span class="lineno">  179</span>&#160;              <span class="keywordtype">int</span> depth_max = -1, <span class="keywordtype">int</span> depth_min_ = -1, <span class="keywordtype">int</span> votes_max_ = -1,</div><div class="line"><a name="l00180"></a><span class="lineno">  180</span>&#160;              <span class="keywordtype">float</span> density = -1.0, <span class="keywordtype">int</span> seed = 0) {</div><div class="line"><a name="l00181"></a><span class="lineno">  181</span>&#160;      <span class="keywordflow">if</span> (target_recall &lt; 0.0 - epsilon || target_recall &gt; 1.0 + epsilon) {</div><div class="line"><a name="l00182"></a><span class="lineno">  182</span>&#160;        <span class="keywordflow">throw</span> std::out_of_range(<span class="stringliteral">&quot;Target recall must be on the interval [0,1].&quot;</span>);</div><div class="line"><a name="l00183"></a><span class="lineno">  183</span>&#160;      }</div><div class="line"><a name="l00184"></a><span class="lineno">  184</span>&#160;</div><div class="line"><a name="l00185"></a><span class="lineno">  185</span>&#160;      <a class="code" href="class_mrpt.html#a7d789c72d12dd668fc5ffb157a9a52f4">grow</a>(Q, k_, trees_max, depth_max, depth_min_, votes_max_, density, seed);</div><div class="line"><a name="l00186"></a><span class="lineno">  186</span>&#160;      prune(target_recall);</div><div class="line"><a name="l00187"></a><span class="lineno">  187</span>&#160;    }</div><div class="line"><a name="l00188"></a><span class="lineno">  188</span>&#160;</div><div class="line"><a name="l00218"></a><span class="lineno"><a class="line" href="class_mrpt.html#a2bf7a8162fc7f6bd56d10097aeedffc8">  218</a></span>&#160;    <span class="keywordtype">void</span> <a class="code" href="class_mrpt.html#a2bf7a8162fc7f6bd56d10097aeedffc8">grow</a>(<span class="keywordtype">double</span> target_recall, <span class="keyword">const</span> <span class="keywordtype">float</span> *Q, <span class="keywordtype">int</span> n_test, <span class="keywordtype">int</span> k_, <span class="keywordtype">int</span> trees_max = -1,</div><div class="line"><a name="l00219"></a><span class="lineno">  219</span>&#160;              <span class="keywordtype">int</span> depth_max = -1, <span class="keywordtype">int</span> depth_min_ = -1, <span class="keywordtype">int</span> votes_max_ = -1,</div><div class="line"><a name="l00220"></a><span class="lineno">  220</span>&#160;              <span class="keywordtype">float</span> density = -1.0, <span class="keywordtype">int</span> seed = 0, <span class="keyword">const</span> std::vector&lt;int&gt; &amp;indices_test = {}) {</div><div class="line"><a name="l00221"></a><span class="lineno">  221</span>&#160;      <span class="keywordflow">if</span> (target_recall &lt; 0.0 - epsilon || target_recall &gt; 1.0 + epsilon) {</div><div class="line"><a name="l00222"></a><span class="lineno">  222</span>&#160;        <span class="keywordflow">throw</span> std::out_of_range(<span class="stringliteral">&quot;Target recall must be on the interval [0,1].&quot;</span>);</div><div class="line"><a name="l00223"></a><span class="lineno">  223</span>&#160;      }</div><div class="line"><a name="l00224"></a><span class="lineno">  224</span>&#160;</div><div class="line"><a name="l00225"></a><span class="lineno">  225</span>&#160;      <a class="code" href="class_mrpt.html#a7d789c72d12dd668fc5ffb157a9a52f4">grow</a>(Q, n_test, k_, trees_max, depth_max, depth_min_, votes_max_, density, seed, indices_test);</div><div class="line"><a name="l00226"></a><span class="lineno">  226</span>&#160;      prune(target_recall);</div><div class="line"><a name="l00227"></a><span class="lineno">  227</span>&#160;    }</div><div class="line"><a name="l00228"></a><span class="lineno">  228</span>&#160;</div><div class="line"><a name="l00256"></a><span class="lineno"><a class="line" href="class_mrpt.html#a6911b63fb6fb87129f893831391f42dd">  256</a></span>&#160;    <span class="keywordtype">void</span> <a class="code" href="class_mrpt.html#a6911b63fb6fb87129f893831391f42dd">grow_autotune</a>(<span class="keywordtype">double</span> target_recall, <span class="keywordtype">int</span> k_, <span class="keywordtype">int</span> trees_max = -1, <span class="keywordtype">int</span> depth_max = -1, <span class="keywordtype">int</span> depth_min_ = -1,</div><div class="line"><a name="l00257"></a><span class="lineno">  257</span>&#160;                       <span class="keywordtype">int</span> votes_max_ = -1, <span class="keywordtype">float</span> density_ = -1.0, <span class="keywordtype">int</span> seed = 0, <span class="keywordtype">int</span> n_test = 100) {</div><div class="line"><a name="l00258"></a><span class="lineno">  258</span>&#160;      <span class="keywordflow">if</span> (n_test &lt; 1) {</div><div class="line"><a name="l00259"></a><span class="lineno">  259</span>&#160;        <span class="keywordflow">throw</span> std::out_of_range(<span class="stringliteral">&quot;Test set size must be &gt; 0.&quot;</span>);</div><div class="line"><a name="l00260"></a><span class="lineno">  260</span>&#160;      }</div><div class="line"><a name="l00261"></a><span class="lineno">  261</span>&#160;</div><div class="line"><a name="l00262"></a><span class="lineno">  262</span>&#160;      n_test = n_test &gt; n_samples ? n_samples : n_test;</div><div class="line"><a name="l00263"></a><span class="lineno">  263</span>&#160;      std::vector&lt;int&gt; indices_test(sample_indices(n_test, seed));</div><div class="line"><a name="l00264"></a><span class="lineno">  264</span>&#160;      <span class="keyword">const</span> Eigen::MatrixXf Q(<a class="code" href="class_mrpt.html#add8686a73cfae8d8391e85314acbe81a">subset</a>(indices_test));</div><div class="line"><a name="l00265"></a><span class="lineno">  265</span>&#160;</div><div class="line"><a name="l00266"></a><span class="lineno">  266</span>&#160;      <a class="code" href="class_mrpt.html#a7d789c72d12dd668fc5ffb157a9a52f4">grow</a>(target_recall, Q.data(), Q.cols(), k_, trees_max,</div><div class="line"><a name="l00267"></a><span class="lineno">  267</span>&#160;        depth_max, depth_min_, votes_max_, density_, seed, indices_test);</div><div class="line"><a name="l00268"></a><span class="lineno">  268</span>&#160;    }</div><div class="line"><a name="l00269"></a><span class="lineno">  269</span>&#160;</div><div class="line"><a name="l00281"></a><span class="lineno"><a class="line" href="class_mrpt.html#a82a6cd3ce04ff8307bfd00d99fc5400a">  281</a></span>&#160;    <a class="code" href="struct_mrpt___parameters.html">Mrpt_Parameters</a> <a class="code" href="class_mrpt.html#a82a6cd3ce04ff8307bfd00d99fc5400a">parameters</a>()<span class="keyword"> const </span>{</div><div class="line"><a name="l00282"></a><span class="lineno">  282</span>&#160;      <span class="keywordflow">if</span> (index_type == normal || index_type == autotuned_unpruned) {</div><div class="line"><a name="l00283"></a><span class="lineno">  283</span>&#160;        <a class="code" href="struct_mrpt___parameters.html">Mrpt_Parameters</a> p;</div><div class="line"><a name="l00284"></a><span class="lineno">  284</span>&#160;        p.<a class="code" href="struct_mrpt___parameters.html#a4a0ab518bfa3786b59e58481bb7de814">n_trees</a> = n_trees;</div><div class="line"><a name="l00285"></a><span class="lineno">  285</span>&#160;        p.<a class="code" href="struct_mrpt___parameters.html#a74ecd1a5f8320ec0cb30d2bfb7979fd0">depth</a> = depth;</div><div class="line"><a name="l00286"></a><span class="lineno">  286</span>&#160;        p.<a class="code" href="struct_mrpt___parameters.html#ad8b374d6671223a498e67cb98d8ea191">k</a> = par.<a class="code" href="struct_mrpt___parameters.html#ad8b374d6671223a498e67cb98d8ea191">k</a>;</div><div class="line"><a name="l00287"></a><span class="lineno">  287</span>&#160;        <span class="keywordflow">return</span> p;</div><div class="line"><a name="l00288"></a><span class="lineno">  288</span>&#160;      }</div><div class="line"><a name="l00289"></a><span class="lineno">  289</span>&#160;</div><div class="line"><a name="l00290"></a><span class="lineno">  290</span>&#160;      <span class="keywordflow">return</span> par;</div><div class="line"><a name="l00291"></a><span class="lineno">  291</span>&#160;    }</div><div class="line"><a name="l00292"></a><span class="lineno">  292</span>&#160;</div><div class="line"><a name="l00298"></a><span class="lineno"><a class="line" href="class_mrpt.html#a64adc0d6952d12e4599459f8654e860e">  298</a></span>&#160;    <span class="keywordtype">bool</span> <a class="code" href="class_mrpt.html#a64adc0d6952d12e4599459f8654e860e">is_autotuned</a>()<span class="keyword"> const </span>{</div><div class="line"><a name="l00299"></a><span class="lineno">  299</span>&#160;      <span class="keywordflow">return</span> index_type == autotuned;</div><div class="line"><a name="l00300"></a><span class="lineno">  300</span>&#160;    }</div><div class="line"><a name="l00301"></a><span class="lineno">  301</span>&#160;</div><div class="line"><a name="l00344"></a><span class="lineno"><a class="line" href="class_mrpt.html#a40eae6f75c79b4ff0f5400b14f8d81c2">  344</a></span>&#160;    <span class="keywordtype">void</span> <a class="code" href="class_mrpt.html#a40eae6f75c79b4ff0f5400b14f8d81c2">grow</a>(<span class="keyword">const</span> <span class="keywordtype">float</span> *data, <span class="keywordtype">int</span> n_test, <span class="keywordtype">int</span> k_, <span class="keywordtype">int</span> trees_max = -1, <span class="keywordtype">int</span> depth_max = -1,</div><div class="line"><a name="l00345"></a><span class="lineno">  345</span>&#160;              <span class="keywordtype">int</span> depth_min_ = -1, <span class="keywordtype">int</span> votes_max_ = -1, <span class="keywordtype">float</span> density_ = -1.0, <span class="keywordtype">int</span> seed = 0,</div><div class="line"><a name="l00346"></a><span class="lineno">  346</span>&#160;              <span class="keyword">const</span> std::vector&lt;int&gt; &amp;indices_test = {}) {</div><div class="line"><a name="l00347"></a><span class="lineno">  347</span>&#160;</div><div class="line"><a name="l00348"></a><span class="lineno">  348</span>&#160;      <span class="keywordflow">if</span> (trees_max == - 1) {</div><div class="line"><a name="l00349"></a><span class="lineno">  349</span>&#160;        trees_max = std::min(std::sqrt(n_samples), 1000.0);</div><div class="line"><a name="l00350"></a><span class="lineno">  350</span>&#160;      }</div><div class="line"><a name="l00351"></a><span class="lineno">  351</span>&#160;</div><div class="line"><a name="l00352"></a><span class="lineno">  352</span>&#160;      <span class="keywordflow">if</span> (depth_min_ == -1) {</div><div class="line"><a name="l00353"></a><span class="lineno">  353</span>&#160;        depth_min_ = std::max(static_cast&lt;int&gt;(std::log2(n_samples) - 11), 5);</div><div class="line"><a name="l00354"></a><span class="lineno">  354</span>&#160;      }</div><div class="line"><a name="l00355"></a><span class="lineno">  355</span>&#160;</div><div class="line"><a name="l00356"></a><span class="lineno">  356</span>&#160;      <span class="keywordflow">if</span> (depth_max == -1) {</div><div class="line"><a name="l00357"></a><span class="lineno">  357</span>&#160;        depth_max = std::max(static_cast&lt;int&gt;(std::log2(n_samples) - 4), depth_min_);</div><div class="line"><a name="l00358"></a><span class="lineno">  358</span>&#160;      }</div><div class="line"><a name="l00359"></a><span class="lineno">  359</span>&#160;</div><div class="line"><a name="l00360"></a><span class="lineno">  360</span>&#160;      <span class="keywordflow">if</span> (votes_max_ == -1) {</div><div class="line"><a name="l00361"></a><span class="lineno">  361</span>&#160;        votes_max_ = std::max(trees_max / 10, std::min(trees_max, 10));</div><div class="line"><a name="l00362"></a><span class="lineno">  362</span>&#160;      }</div><div class="line"><a name="l00363"></a><span class="lineno">  363</span>&#160;</div><div class="line"><a name="l00364"></a><span class="lineno">  364</span>&#160;      <span class="keywordflow">if</span> (density_ &gt; -1.0001 &amp;&amp; density_ &lt; -0.9999) {</div><div class="line"><a name="l00365"></a><span class="lineno">  365</span>&#160;        density_ = 1.0 / std::sqrt(dim);</div><div class="line"><a name="l00366"></a><span class="lineno">  366</span>&#160;      }</div><div class="line"><a name="l00367"></a><span class="lineno">  367</span>&#160;</div><div class="line"><a name="l00368"></a><span class="lineno">  368</span>&#160;      <span class="keywordflow">if</span> (!<a class="code" href="class_mrpt.html#a960f1af81f2b56db7eb8cd0d5a352c14">empty</a>()) {</div><div class="line"><a name="l00369"></a><span class="lineno">  369</span>&#160;        <span class="keywordflow">throw</span> std::logic_error(<span class="stringliteral">&quot;The index has already been grown.&quot;</span>);</div><div class="line"><a name="l00370"></a><span class="lineno">  370</span>&#160;      }</div><div class="line"><a name="l00371"></a><span class="lineno">  371</span>&#160;</div><div class="line"><a name="l00372"></a><span class="lineno">  372</span>&#160;      <span class="keywordflow">if</span> (k_ &lt;= 0 || k_ &gt; n_samples) {</div><div class="line"><a name="l00373"></a><span class="lineno">  373</span>&#160;        <span class="keywordflow">throw</span> std::out_of_range(<span class="stringliteral">&quot;k_ must belong to the set {1, ..., n}.&quot;</span>);</div><div class="line"><a name="l00374"></a><span class="lineno">  374</span>&#160;      }</div><div class="line"><a name="l00375"></a><span class="lineno">  375</span>&#160;</div><div class="line"><a name="l00376"></a><span class="lineno">  376</span>&#160;      <span class="keywordflow">if</span> (trees_max &lt;= 0) {</div><div class="line"><a name="l00377"></a><span class="lineno">  377</span>&#160;        <span class="keywordflow">throw</span> std::out_of_range(<span class="stringliteral">&quot;trees_max must be positive.&quot;</span>);</div><div class="line"><a name="l00378"></a><span class="lineno">  378</span>&#160;      }</div><div class="line"><a name="l00379"></a><span class="lineno">  379</span>&#160;</div><div class="line"><a name="l00380"></a><span class="lineno">  380</span>&#160;      <span class="keywordflow">if</span> (depth_max &lt;= 0 || depth_max &gt; std::log2(n_samples)) {</div><div class="line"><a name="l00381"></a><span class="lineno">  381</span>&#160;        <span class="keywordflow">throw</span> std::out_of_range(<span class="stringliteral">&quot;depth_max must belong to the set {1, ... , log2(n)}.&quot;</span>);</div><div class="line"><a name="l00382"></a><span class="lineno">  382</span>&#160;      }</div><div class="line"><a name="l00383"></a><span class="lineno">  383</span>&#160;</div><div class="line"><a name="l00384"></a><span class="lineno">  384</span>&#160;      <span class="keywordflow">if</span> (depth_min_ &lt;= 0 || depth_min_ &gt; depth_max) {</div><div class="line"><a name="l00385"></a><span class="lineno">  385</span>&#160;        <span class="keywordflow">throw</span> std::out_of_range(<span class="stringliteral">&quot;depth_min_ must belong to the set {1, ... , depth_max}&quot;</span>);</div><div class="line"><a name="l00386"></a><span class="lineno">  386</span>&#160;      }</div><div class="line"><a name="l00387"></a><span class="lineno">  387</span>&#160;</div><div class="line"><a name="l00388"></a><span class="lineno">  388</span>&#160;      <span class="keywordflow">if</span> (votes_max_ &lt;= 0 || votes_max_ &gt; trees_max) {</div><div class="line"><a name="l00389"></a><span class="lineno">  389</span>&#160;        <span class="keywordflow">throw</span> std::out_of_range(<span class="stringliteral">&quot;votes_max_ must belong to the set {1, ... , trees_max}.&quot;</span>);</div><div class="line"><a name="l00390"></a><span class="lineno">  390</span>&#160;      }</div><div class="line"><a name="l00391"></a><span class="lineno">  391</span>&#160;</div><div class="line"><a name="l00392"></a><span class="lineno">  392</span>&#160;      <span class="keywordflow">if</span> (density_ &lt; 0.0 || density_ &gt; 1.0001) {</div><div class="line"><a name="l00393"></a><span class="lineno">  393</span>&#160;        <span class="keywordflow">throw</span> std::out_of_range(<span class="stringliteral">&quot;The density must be on the interval (0,1].&quot;</span>);</div><div class="line"><a name="l00394"></a><span class="lineno">  394</span>&#160;      }</div><div class="line"><a name="l00395"></a><span class="lineno">  395</span>&#160;</div><div class="line"><a name="l00396"></a><span class="lineno">  396</span>&#160;      <span class="keywordflow">if</span>(n_samples &lt; 101) {</div><div class="line"><a name="l00397"></a><span class="lineno">  397</span>&#160;        <span class="keywordflow">throw</span> std::out_of_range(<span class="stringliteral">&quot;Sample size must be at least 101 to autotune an index.&quot;</span>);</div><div class="line"><a name="l00398"></a><span class="lineno">  398</span>&#160;      }</div><div class="line"><a name="l00399"></a><span class="lineno">  399</span>&#160;</div><div class="line"><a name="l00400"></a><span class="lineno">  400</span>&#160;      depth_min = depth_min_;</div><div class="line"><a name="l00401"></a><span class="lineno">  401</span>&#160;      votes_max = votes_max_;</div><div class="line"><a name="l00402"></a><span class="lineno">  402</span>&#160;      k = k_;</div><div class="line"><a name="l00403"></a><span class="lineno">  403</span>&#160;</div><div class="line"><a name="l00404"></a><span class="lineno">  404</span>&#160;      <span class="keyword">const</span> Eigen::Map&lt;const Eigen::MatrixXf&gt; Q(data, dim, n_test);</div><div class="line"><a name="l00405"></a><span class="lineno">  405</span>&#160;</div><div class="line"><a name="l00406"></a><span class="lineno">  406</span>&#160;      <a class="code" href="class_mrpt.html#a7d789c72d12dd668fc5ffb157a9a52f4">grow</a>(trees_max, depth_max, density_, seed);</div><div class="line"><a name="l00407"></a><span class="lineno">  407</span>&#160;      Eigen::MatrixXi exact(k, n_test);</div><div class="line"><a name="l00408"></a><span class="lineno">  408</span>&#160;      compute_exact(Q, exact, indices_test);</div><div class="line"><a name="l00409"></a><span class="lineno">  409</span>&#160;</div><div class="line"><a name="l00410"></a><span class="lineno">  410</span>&#160;      std::vector&lt;Eigen::MatrixXd&gt; recalls(depth_max - depth_min + 1);</div><div class="line"><a name="l00411"></a><span class="lineno">  411</span>&#160;      cs_sizes = std::vector&lt;Eigen::MatrixXd&gt;(depth_max - depth_min + 1);</div><div class="line"><a name="l00412"></a><span class="lineno">  412</span>&#160;</div><div class="line"><a name="l00413"></a><span class="lineno">  413</span>&#160;      <span class="keywordflow">for</span> (<span class="keywordtype">int</span> d = depth_min; d &lt;= depth_max; ++d) {</div><div class="line"><a name="l00414"></a><span class="lineno">  414</span>&#160;        recalls[d - depth_min] = Eigen::MatrixXd::Zero(votes_max, trees_max);</div><div class="line"><a name="l00415"></a><span class="lineno">  415</span>&#160;        cs_sizes[d - depth_min] = Eigen::MatrixXd::Zero(votes_max, trees_max);</div><div class="line"><a name="l00416"></a><span class="lineno">  416</span>&#160;      }</div><div class="line"><a name="l00417"></a><span class="lineno">  417</span>&#160;</div><div class="line"><a name="l00418"></a><span class="lineno">  418</span>&#160;      <span class="keywordflow">for</span> (<span class="keywordtype">int</span> i = 0; i &lt; n_test; ++i) {</div><div class="line"><a name="l00419"></a><span class="lineno">  419</span>&#160;        std::vector&lt;Eigen::MatrixXd&gt; recall_tmp(depth_max - depth_min + 1);</div><div class="line"><a name="l00420"></a><span class="lineno">  420</span>&#160;        std::vector&lt;Eigen::MatrixXd&gt; cs_size_tmp(depth_max - depth_min + 1);</div><div class="line"><a name="l00421"></a><span class="lineno">  421</span>&#160;</div><div class="line"><a name="l00422"></a><span class="lineno">  422</span>&#160;        count_elected(Q.col(i), Eigen::Map&lt;Eigen::VectorXi&gt;(exact.data() + i * k, k),</div><div class="line"><a name="l00423"></a><span class="lineno">  423</span>&#160;         votes_max, recall_tmp, cs_size_tmp);</div><div class="line"><a name="l00424"></a><span class="lineno">  424</span>&#160;</div><div class="line"><a name="l00425"></a><span class="lineno">  425</span>&#160;        <span class="keywordflow">for</span> (<span class="keywordtype">int</span> d = depth_min; d &lt;= depth_max; ++d) {</div><div class="line"><a name="l00426"></a><span class="lineno">  426</span>&#160;          recalls[d - depth_min] += recall_tmp[d - depth_min];</div><div class="line"><a name="l00427"></a><span class="lineno">  427</span>&#160;          cs_sizes[d - depth_min] += cs_size_tmp[d - depth_min];</div><div class="line"><a name="l00428"></a><span class="lineno">  428</span>&#160;        }</div><div class="line"><a name="l00429"></a><span class="lineno">  429</span>&#160;      }</div><div class="line"><a name="l00430"></a><span class="lineno">  430</span>&#160;</div><div class="line"><a name="l00431"></a><span class="lineno">  431</span>&#160;      <span class="keywordflow">for</span> (<span class="keywordtype">int</span> d = depth_min; d &lt;= depth_max; ++d) {</div><div class="line"><a name="l00432"></a><span class="lineno">  432</span>&#160;        recalls[d - depth_min] /= (k * n_test);</div><div class="line"><a name="l00433"></a><span class="lineno">  433</span>&#160;        cs_sizes[d - depth_min] /= n_test;</div><div class="line"><a name="l00434"></a><span class="lineno">  434</span>&#160;      }</div><div class="line"><a name="l00435"></a><span class="lineno">  435</span>&#160;</div><div class="line"><a name="l00436"></a><span class="lineno">  436</span>&#160;      fit_times(Q);</div><div class="line"><a name="l00437"></a><span class="lineno">  437</span>&#160;      std::set&lt;Mrpt_Parameters,decltype(is_faster)*&gt; pars = list_parameters(recalls);</div><div class="line"><a name="l00438"></a><span class="lineno">  438</span>&#160;      opt_pars = pareto_frontier(pars);</div><div class="line"><a name="l00439"></a><span class="lineno">  439</span>&#160;</div><div class="line"><a name="l00440"></a><span class="lineno">  440</span>&#160;      index_type = autotuned_unpruned;</div><div class="line"><a name="l00441"></a><span class="lineno">  441</span>&#160;      par.<a class="code" href="struct_mrpt___parameters.html#ad8b374d6671223a498e67cb98d8ea191">k</a> = k_;</div><div class="line"><a name="l00442"></a><span class="lineno">  442</span>&#160;    }</div><div class="line"><a name="l00443"></a><span class="lineno">  443</span>&#160;</div><div class="line"><a name="l00468"></a><span class="lineno"><a class="line" href="class_mrpt.html#a7a6325435fc140ac5394c2f0136ea211">  468</a></span>&#160;    <span class="keywordtype">void</span> <a class="code" href="class_mrpt.html#a7a6325435fc140ac5394c2f0136ea211">grow</a>(<span class="keyword">const</span> Eigen::Ref&lt;const Eigen::MatrixXf&gt; &amp;Q, <span class="keywordtype">int</span> k_, <span class="keywordtype">int</span> trees_max = -1, <span class="keywordtype">int</span> depth_max = -1,</div><div class="line"><a name="l00469"></a><span class="lineno">  469</span>&#160;              <span class="keywordtype">int</span> depth_min_ = -1, <span class="keywordtype">int</span> votes_max_ = -1, <span class="keywordtype">float</span> density_ = -1.0, <span class="keywordtype">int</span> seed = 0) {</div><div class="line"><a name="l00470"></a><span class="lineno">  470</span>&#160;      <span class="keywordflow">if</span> (Q.rows() != dim) {</div><div class="line"><a name="l00471"></a><span class="lineno">  471</span>&#160;        <span class="keywordflow">throw</span> std::invalid_argument(<span class="stringliteral">&quot;Dimensions of the data and the validation set do not match.&quot;</span>);</div><div class="line"><a name="l00472"></a><span class="lineno">  472</span>&#160;      }</div><div class="line"><a name="l00473"></a><span class="lineno">  473</span>&#160;</div><div class="line"><a name="l00474"></a><span class="lineno">  474</span>&#160;      <a class="code" href="class_mrpt.html#a7d789c72d12dd668fc5ffb157a9a52f4">grow</a>(Q.data(), Q.cols(), k_, trees_max,</div><div class="line"><a name="l00475"></a><span class="lineno">  475</span>&#160;        depth_max, depth_min_, votes_max_, density_, seed);</div><div class="line"><a name="l00476"></a><span class="lineno">  476</span>&#160;    }</div><div class="line"><a name="l00477"></a><span class="lineno">  477</span>&#160;</div><div class="line"><a name="l00503"></a><span class="lineno"><a class="line" href="class_mrpt.html#a9ee717a90c0548f31577b033540dda44">  503</a></span>&#160;    <span class="keywordtype">void</span> <a class="code" href="class_mrpt.html#a9ee717a90c0548f31577b033540dda44">grow_autotune</a>(<span class="keywordtype">int</span> k_, <span class="keywordtype">int</span> trees_max = -1, <span class="keywordtype">int</span> depth_max = -1, <span class="keywordtype">int</span> depth_min_ = -1,</div><div class="line"><a name="l00504"></a><span class="lineno">  504</span>&#160;                    <span class="keywordtype">int</span> votes_max_ = -1, <span class="keywordtype">float</span> density_ = -1.0, <span class="keywordtype">int</span> seed = 0, <span class="keywordtype">int</span> n_test = 100) {</div><div class="line"><a name="l00505"></a><span class="lineno">  505</span>&#160;      <span class="keywordflow">if</span> (n_test &lt; 1) {</div><div class="line"><a name="l00506"></a><span class="lineno">  506</span>&#160;        <span class="keywordflow">throw</span> std::out_of_range(<span class="stringliteral">&quot;Test set size must be &gt; 0.&quot;</span>);</div><div class="line"><a name="l00507"></a><span class="lineno">  507</span>&#160;      }</div><div class="line"><a name="l00508"></a><span class="lineno">  508</span>&#160;</div><div class="line"><a name="l00509"></a><span class="lineno">  509</span>&#160;      n_test = n_test &gt; n_samples ? n_samples : n_test;</div><div class="line"><a name="l00510"></a><span class="lineno">  510</span>&#160;      std::vector&lt;int&gt; indices_test(sample_indices(n_test, seed));</div><div class="line"><a name="l00511"></a><span class="lineno">  511</span>&#160;      <span class="keyword">const</span> Eigen::MatrixXf Q(<a class="code" href="class_mrpt.html#add8686a73cfae8d8391e85314acbe81a">subset</a>(indices_test));</div><div class="line"><a name="l00512"></a><span class="lineno">  512</span>&#160;</div><div class="line"><a name="l00513"></a><span class="lineno">  513</span>&#160;      <a class="code" href="class_mrpt.html#a7d789c72d12dd668fc5ffb157a9a52f4">grow</a>(Q.data(), Q.cols(), k_, trees_max,</div><div class="line"><a name="l00514"></a><span class="lineno">  514</span>&#160;        depth_max, depth_min_, votes_max_, density_, seed, indices_test);</div><div class="line"><a name="l00515"></a><span class="lineno">  515</span>&#160;    }</div><div class="line"><a name="l00516"></a><span class="lineno">  516</span>&#160;</div><div class="line"><a name="l00527"></a><span class="lineno"><a class="line" href="class_mrpt.html#add8686a73cfae8d8391e85314acbe81a">  527</a></span>&#160;    <a class="code" href="class_mrpt.html">Mrpt</a> <a class="code" href="class_mrpt.html#add8686a73cfae8d8391e85314acbe81a">subset</a>(<span class="keywordtype">double</span> target_recall)<span class="keyword"> const </span>{</div><div class="line"><a name="l00528"></a><span class="lineno">  528</span>&#160;      <span class="keywordflow">if</span> (target_recall &lt; 0.0 - epsilon || target_recall &gt; 1.0 + epsilon) {</div><div class="line"><a name="l00529"></a><span class="lineno">  529</span>&#160;        <span class="keywordflow">throw</span> std::out_of_range(<span class="stringliteral">&quot;Target recall must be on the interval [0,1].&quot;</span>);</div><div class="line"><a name="l00530"></a><span class="lineno">  530</span>&#160;      }</div><div class="line"><a name="l00531"></a><span class="lineno">  531</span>&#160;</div><div class="line"><a name="l00532"></a><span class="lineno">  532</span>&#160;      <a class="code" href="class_mrpt.html">Mrpt</a> index2(X);</div><div class="line"><a name="l00533"></a><span class="lineno">  533</span>&#160;      index2.par = <a class="code" href="class_mrpt.html#a82a6cd3ce04ff8307bfd00d99fc5400a">parameters</a>(target_recall);</div><div class="line"><a name="l00534"></a><span class="lineno">  534</span>&#160;</div><div class="line"><a name="l00535"></a><span class="lineno">  535</span>&#160;      <span class="keywordtype">int</span> depth_max = depth;</div><div class="line"><a name="l00536"></a><span class="lineno">  536</span>&#160;</div><div class="line"><a name="l00537"></a><span class="lineno">  537</span>&#160;      index2.n_trees = index2.par.<a class="code" href="struct_mrpt___parameters.html#a4a0ab518bfa3786b59e58481bb7de814">n_trees</a>;</div><div class="line"><a name="l00538"></a><span class="lineno">  538</span>&#160;      index2.depth = index2.par.<a class="code" href="struct_mrpt___parameters.html#a74ecd1a5f8320ec0cb30d2bfb7979fd0">depth</a>;</div><div class="line"><a name="l00539"></a><span class="lineno">  539</span>&#160;      index2.votes = index2.par.<a class="code" href="struct_mrpt___parameters.html#ab34a16df707e0abd7b8b0bf4959f36b6">votes</a>;</div><div class="line"><a name="l00540"></a><span class="lineno">  540</span>&#160;      index2.n_pool = index2.depth * index2.n_trees;</div><div class="line"><a name="l00541"></a><span class="lineno">  541</span>&#160;      index2.n_array = 1 &lt;&lt; (index2.depth + 1);</div><div class="line"><a name="l00542"></a><span class="lineno">  542</span>&#160;      index2.tree_leaves.assign(tree_leaves.begin(), tree_leaves.begin() + index2.n_trees);</div><div class="line"><a name="l00543"></a><span class="lineno">  543</span>&#160;      index2.leaf_first_indices_all = leaf_first_indices_all;</div><div class="line"><a name="l00544"></a><span class="lineno">  544</span>&#160;      index2.density = density;</div><div class="line"><a name="l00545"></a><span class="lineno">  545</span>&#160;      index2.k = k;</div><div class="line"><a name="l00546"></a><span class="lineno">  546</span>&#160;</div><div class="line"><a name="l00547"></a><span class="lineno">  547</span>&#160;      index2.split_points = split_points.topLeftCorner(index2.n_array, index2.n_trees);</div><div class="line"><a name="l00548"></a><span class="lineno">  548</span>&#160;      index2.leaf_first_indices = leaf_first_indices_all[index2.depth];</div><div class="line"><a name="l00549"></a><span class="lineno">  549</span>&#160;      <span class="keywordflow">if</span> (index2.density &lt; 1) {</div><div class="line"><a name="l00550"></a><span class="lineno">  550</span>&#160;        index2.sparse_random_matrix = Eigen::SparseMatrix&lt;float, Eigen::RowMajor&gt;(index2.n_pool, index2.dim);</div><div class="line"><a name="l00551"></a><span class="lineno">  551</span>&#160;        <span class="keywordflow">for</span> (<span class="keywordtype">int</span> n_tree = 0; n_tree &lt; index2.n_trees; ++n_tree)</div><div class="line"><a name="l00552"></a><span class="lineno">  552</span>&#160;          index2.sparse_random_matrix.middleRows(n_tree * index2.depth, index2.depth) =</div><div class="line"><a name="l00553"></a><span class="lineno">  553</span>&#160;            sparse_random_matrix.middleRows(n_tree * depth_max, index2.depth);</div><div class="line"><a name="l00554"></a><span class="lineno">  554</span>&#160;      } <span class="keywordflow">else</span> {</div><div class="line"><a name="l00555"></a><span class="lineno">  555</span>&#160;        index2.dense_random_matrix = Eigen::Matrix&lt;float, Eigen::Dynamic, Eigen::Dynamic, Eigen::RowMajor&gt;(index2.n_pool, index2.dim);</div><div class="line"><a name="l00556"></a><span class="lineno">  556</span>&#160;        <span class="keywordflow">for</span> (<span class="keywordtype">int</span> n_tree = 0; n_tree &lt; index2.n_trees; ++n_tree)</div><div class="line"><a name="l00557"></a><span class="lineno">  557</span>&#160;          index2.dense_random_matrix.middleRows(n_tree * index2.depth, index2.depth) =</div><div class="line"><a name="l00558"></a><span class="lineno">  558</span>&#160;            dense_random_matrix.middleRows(n_tree * depth_max, index2.depth);</div><div class="line"><a name="l00559"></a><span class="lineno">  559</span>&#160;      }</div><div class="line"><a name="l00560"></a><span class="lineno">  560</span>&#160;      index2.index_type = autotuned;</div><div class="line"><a name="l00561"></a><span class="lineno">  561</span>&#160;</div><div class="line"><a name="l00562"></a><span class="lineno">  562</span>&#160;      <span class="keywordflow">return</span> index2;</div><div class="line"><a name="l00563"></a><span class="lineno">  563</span>&#160;    }</div><div class="line"><a name="l00564"></a><span class="lineno">  564</span>&#160;</div><div class="line"><a name="l00565"></a><span class="lineno">  565</span>&#160;</div><div class="line"><a name="l00577"></a><span class="lineno"><a class="line" href="class_mrpt.html#a4dc319eab5e2bc9a5a6d7a6007cef34d">  577</a></span>&#160;    <a class="code" href="class_mrpt.html">Mrpt</a> *<a class="code" href="class_mrpt.html#a4dc319eab5e2bc9a5a6d7a6007cef34d">subset_pointer</a>(<span class="keywordtype">double</span> target_recall)<span class="keyword"> const </span>{</div><div class="line"><a name="l00578"></a><span class="lineno">  578</span>&#160;      <span class="keywordflow">if</span> (target_recall &lt; 0.0 - epsilon || target_recall &gt; 1.0 + epsilon) {</div><div class="line"><a name="l00579"></a><span class="lineno">  579</span>&#160;        <span class="keywordflow">throw</span> std::out_of_range(<span class="stringliteral">&quot;Target recall must be on the interval [0,1].&quot;</span>);</div><div class="line"><a name="l00580"></a><span class="lineno">  580</span>&#160;      }</div><div class="line"><a name="l00581"></a><span class="lineno">  581</span>&#160;</div><div class="line"><a name="l00582"></a><span class="lineno">  582</span>&#160;      <a class="code" href="class_mrpt.html">Mrpt</a> *index2 = <span class="keyword">new</span> <a class="code" href="class_mrpt.html#a1e77a5c97c6ae94d17162f65041821bc">Mrpt</a>(X);</div><div class="line"><a name="l00583"></a><span class="lineno">  583</span>&#160;      index2-&gt;par = <a class="code" href="class_mrpt.html#a82a6cd3ce04ff8307bfd00d99fc5400a">parameters</a>(target_recall);</div><div class="line"><a name="l00584"></a><span class="lineno">  584</span>&#160;</div><div class="line"><a name="l00585"></a><span class="lineno">  585</span>&#160;      <span class="keywordtype">int</span> depth_max = depth;</div><div class="line"><a name="l00586"></a><span class="lineno">  586</span>&#160;</div><div class="line"><a name="l00587"></a><span class="lineno">  587</span>&#160;      index2-&gt;n_trees = index2-&gt;par.<a class="code" href="struct_mrpt___parameters.html#a4a0ab518bfa3786b59e58481bb7de814">n_trees</a>;</div><div class="line"><a name="l00588"></a><span class="lineno">  588</span>&#160;      index2-&gt;depth = index2-&gt;par.<a class="code" href="struct_mrpt___parameters.html#a74ecd1a5f8320ec0cb30d2bfb7979fd0">depth</a>;</div><div class="line"><a name="l00589"></a><span class="lineno">  589</span>&#160;      index2-&gt;votes = index2-&gt;par.<a class="code" href="struct_mrpt___parameters.html#ab34a16df707e0abd7b8b0bf4959f36b6">votes</a>;</div><div class="line"><a name="l00590"></a><span class="lineno">  590</span>&#160;      index2-&gt;n_pool = index2-&gt;depth * index2-&gt;n_trees;</div><div class="line"><a name="l00591"></a><span class="lineno">  591</span>&#160;      index2-&gt;n_array = 1 &lt;&lt; (index2-&gt;depth + 1);</div><div class="line"><a name="l00592"></a><span class="lineno">  592</span>&#160;      index2-&gt;tree_leaves.assign(tree_leaves.begin(), tree_leaves.begin() + index2-&gt;n_trees);</div><div class="line"><a name="l00593"></a><span class="lineno">  593</span>&#160;      index2-&gt;leaf_first_indices_all = leaf_first_indices_all;</div><div class="line"><a name="l00594"></a><span class="lineno">  594</span>&#160;      index2-&gt;density = density;</div><div class="line"><a name="l00595"></a><span class="lineno">  595</span>&#160;      index2-&gt;k = k;</div><div class="line"><a name="l00596"></a><span class="lineno">  596</span>&#160;</div><div class="line"><a name="l00597"></a><span class="lineno">  597</span>&#160;      index2-&gt;split_points = split_points.topLeftCorner(index2-&gt;n_array, index2-&gt;n_trees);</div><div class="line"><a name="l00598"></a><span class="lineno">  598</span>&#160;      index2-&gt;leaf_first_indices = leaf_first_indices_all[index2-&gt;depth];</div><div class="line"><a name="l00599"></a><span class="lineno">  599</span>&#160;      <span class="keywordflow">if</span> (index2-&gt;density &lt; 1) {</div><div class="line"><a name="l00600"></a><span class="lineno">  600</span>&#160;        index2-&gt;sparse_random_matrix = Eigen::SparseMatrix&lt;float, Eigen::RowMajor&gt;(index2-&gt;n_pool, index2-&gt;dim);</div><div class="line"><a name="l00601"></a><span class="lineno">  601</span>&#160;        <span class="keywordflow">for</span> (<span class="keywordtype">int</span> n_tree = 0; n_tree &lt; index2-&gt;n_trees; ++n_tree)</div><div class="line"><a name="l00602"></a><span class="lineno">  602</span>&#160;          index2-&gt;sparse_random_matrix.middleRows(n_tree * index2-&gt;depth, index2-&gt;depth) =</div><div class="line"><a name="l00603"></a><span class="lineno">  603</span>&#160;            sparse_random_matrix.middleRows(n_tree * depth_max, index2-&gt;depth);</div><div class="line"><a name="l00604"></a><span class="lineno">  604</span>&#160;      } <span class="keywordflow">else</span> {</div><div class="line"><a name="l00605"></a><span class="lineno">  605</span>&#160;        index2-&gt;dense_random_matrix = Eigen::Matrix&lt;float, Eigen::Dynamic, Eigen::Dynamic, Eigen::RowMajor&gt;(index2-&gt;n_pool, index2-&gt;dim);</div><div class="line"><a name="l00606"></a><span class="lineno">  606</span>&#160;        <span class="keywordflow">for</span> (<span class="keywordtype">int</span> n_tree = 0; n_tree &lt; index2-&gt;n_trees; ++n_tree)</div><div class="line"><a name="l00607"></a><span class="lineno">  607</span>&#160;          index2-&gt;dense_random_matrix.middleRows(n_tree * index2-&gt;depth, index2-&gt;depth) =</div><div class="line"><a name="l00608"></a><span class="lineno">  608</span>&#160;            dense_random_matrix.middleRows(n_tree * depth_max, index2-&gt;depth);</div><div class="line"><a name="l00609"></a><span class="lineno">  609</span>&#160;      }</div><div class="line"><a name="l00610"></a><span class="lineno">  610</span>&#160;      index2-&gt;index_type = autotuned;</div><div class="line"><a name="l00611"></a><span class="lineno">  611</span>&#160;</div><div class="line"><a name="l00612"></a><span class="lineno">  612</span>&#160;      <span class="keywordflow">return</span> index2;</div><div class="line"><a name="l00613"></a><span class="lineno">  613</span>&#160;    }</div><div class="line"><a name="l00614"></a><span class="lineno">  614</span>&#160;</div><div class="line"><a name="l00615"></a><span class="lineno">  615</span>&#160;</div><div class="line"><a name="l00625"></a><span class="lineno"><a class="line" href="class_mrpt.html#a0907893c90a6255d3cf8d11e940c6ce3">  625</a></span>&#160;    std::vector&lt;Mrpt_Parameters&gt; <a class="code" href="class_mrpt.html#a0907893c90a6255d3cf8d11e940c6ce3">optimal_parameters</a>()<span class="keyword"> const </span>{</div><div class="line"><a name="l00626"></a><span class="lineno">  626</span>&#160;      <span class="keywordflow">if</span> (index_type == normal) {</div><div class="line"><a name="l00627"></a><span class="lineno">  627</span>&#160;        <span class="keywordflow">throw</span> std::logic_error(<span class="stringliteral">&quot;The list of optimal parameters cannot be retrieved for the non-autotuned index.&quot;</span>);</div><div class="line"><a name="l00628"></a><span class="lineno">  628</span>&#160;      }</div><div class="line"><a name="l00629"></a><span class="lineno">  629</span>&#160;      <span class="keywordflow">if</span> (index_type == autotuned) {</div><div class="line"><a name="l00630"></a><span class="lineno">  630</span>&#160;        <span class="keywordflow">throw</span> std::logic_error(<span class="stringliteral">&quot;The list of optimal parameters cannot be retrieved for the index which has already been subsetted or deleted to the target recall level.&quot;</span>);</div><div class="line"><a name="l00631"></a><span class="lineno">  631</span>&#160;      }</div><div class="line"><a name="l00632"></a><span class="lineno">  632</span>&#160;</div><div class="line"><a name="l00633"></a><span class="lineno">  633</span>&#160;      std::vector&lt;Mrpt_Parameters&gt; new_pars;</div><div class="line"><a name="l00634"></a><span class="lineno">  634</span>&#160;      std::copy(opt_pars.begin(), opt_pars.end(), std::back_inserter(new_pars));</div><div class="line"><a name="l00635"></a><span class="lineno">  635</span>&#160;      <span class="keywordflow">return</span> new_pars;</div><div class="line"><a name="l00636"></a><span class="lineno">  636</span>&#160;    }</div><div class="line"><a name="l00637"></a><span class="lineno">  637</span>&#160;</div><div class="line"><a name="l00661"></a><span class="lineno"><a class="line" href="class_mrpt.html#a637e67fe6eba9a6711f5eee65ad4a0da">  661</a></span>&#160;    <span class="keywordtype">void</span> <a class="code" href="class_mrpt.html#a637e67fe6eba9a6711f5eee65ad4a0da">query</a>(<span class="keyword">const</span> <span class="keywordtype">float</span> *data, <span class="keywordtype">int</span> k, <span class="keywordtype">int</span> vote_threshold, <span class="keywordtype">int</span> *out,</div><div class="line"><a name="l00662"></a><span class="lineno">  662</span>&#160;               <span class="keywordtype">float</span> *out_distances = <span class="keyword">nullptr</span>, <span class="keywordtype">int</span> *out_n_elected = <span class="keyword">nullptr</span>)<span class="keyword"> const </span>{</div><div class="line"><a name="l00663"></a><span class="lineno">  663</span>&#160;</div><div class="line"><a name="l00664"></a><span class="lineno">  664</span>&#160;        <span class="keywordflow">if</span> (k &lt;= 0 || k &gt; n_samples) {</div><div class="line"><a name="l00665"></a><span class="lineno">  665</span>&#160;          <span class="keywordflow">throw</span> std::out_of_range(<span class="stringliteral">&quot;k must belong to the set {1, ..., n}.&quot;</span>);</div><div class="line"><a name="l00666"></a><span class="lineno">  666</span>&#160;        }</div><div class="line"><a name="l00667"></a><span class="lineno">  667</span>&#160;</div><div class="line"><a name="l00668"></a><span class="lineno">  668</span>&#160;        <span class="keywordflow">if</span> (vote_threshold &lt;= 0 || vote_threshold &gt; n_trees) {</div><div class="line"><a name="l00669"></a><span class="lineno">  669</span>&#160;          <span class="keywordflow">throw</span> std::out_of_range(<span class="stringliteral">&quot;vote_threshold must belong to the set {1, ... , n_trees}.&quot;</span>);</div><div class="line"><a name="l00670"></a><span class="lineno">  670</span>&#160;        }</div><div class="line"><a name="l00671"></a><span class="lineno">  671</span>&#160;</div><div class="line"><a name="l00672"></a><span class="lineno">  672</span>&#160;        <span class="keywordflow">if</span> (<a class="code" href="class_mrpt.html#a960f1af81f2b56db7eb8cd0d5a352c14">empty</a>()) {</div><div class="line"><a name="l00673"></a><span class="lineno">  673</span>&#160;          <span class="keywordflow">throw</span> std::logic_error(<span class="stringliteral">&quot;The index must be built before making queries.&quot;</span>);</div><div class="line"><a name="l00674"></a><span class="lineno">  674</span>&#160;        }</div><div class="line"><a name="l00675"></a><span class="lineno">  675</span>&#160;</div><div class="line"><a name="l00676"></a><span class="lineno">  676</span>&#160;        <span class="keyword">const</span> Eigen::Map&lt;const Eigen::VectorXf&gt; q(data, dim);</div><div class="line"><a name="l00677"></a><span class="lineno">  677</span>&#160;</div><div class="line"><a name="l00678"></a><span class="lineno">  678</span>&#160;        Eigen::VectorXf projected_query(n_pool);</div><div class="line"><a name="l00679"></a><span class="lineno">  679</span>&#160;        <span class="keywordflow">if</span> (density &lt; 1)</div><div class="line"><a name="l00680"></a><span class="lineno">  680</span>&#160;            projected_query.noalias() = sparse_random_matrix * q;</div><div class="line"><a name="l00681"></a><span class="lineno">  681</span>&#160;        <span class="keywordflow">else</span></div><div class="line"><a name="l00682"></a><span class="lineno">  682</span>&#160;            projected_query.noalias() = dense_random_matrix * q;</div><div class="line"><a name="l00683"></a><span class="lineno">  683</span>&#160;</div><div class="line"><a name="l00684"></a><span class="lineno">  684</span>&#160;        std::vector&lt;int&gt; found_leaves(n_trees);</div><div class="line"><a name="l00685"></a><span class="lineno">  685</span>&#160;</div><div class="line"><a name="l00686"></a><span class="lineno">  686</span>&#160;        <span class="comment">/*</span></div><div class="line"><a name="l00687"></a><span class="lineno">  687</span>&#160;<span class="comment">        * The following loops over all trees, and routes the query to exactly one</span></div><div class="line"><a name="l00688"></a><span class="lineno">  688</span>&#160;<span class="comment">        * leaf in each.</span></div><div class="line"><a name="l00689"></a><span class="lineno">  689</span>&#160;<span class="comment">        */</span></div><div class="line"><a name="l00690"></a><span class="lineno">  690</span>&#160;<span class="preprocessor">        #pragma omp parallel for</span></div><div class="line"><a name="l00691"></a><span class="lineno">  691</span>&#160;        <span class="keywordflow">for</span> (<span class="keywordtype">int</span> n_tree = 0; n_tree &lt; n_trees; ++n_tree) {</div><div class="line"><a name="l00692"></a><span class="lineno">  692</span>&#160;          <span class="keywordtype">int</span> idx_tree = 0;</div><div class="line"><a name="l00693"></a><span class="lineno">  693</span>&#160;          <span class="keywordflow">for</span> (<span class="keywordtype">int</span> d = 0; d &lt; depth; ++d) {</div><div class="line"><a name="l00694"></a><span class="lineno">  694</span>&#160;            <span class="keyword">const</span> <span class="keywordtype">int</span> j = n_tree * depth + d;</div><div class="line"><a name="l00695"></a><span class="lineno">  695</span>&#160;            <span class="keyword">const</span> <span class="keywordtype">int</span> idx_left = 2 * idx_tree + 1;</div><div class="line"><a name="l00696"></a><span class="lineno">  696</span>&#160;            <span class="keyword">const</span> <span class="keywordtype">int</span> idx_right = idx_left + 1;</div><div class="line"><a name="l00697"></a><span class="lineno">  697</span>&#160;            <span class="keyword">const</span> <span class="keywordtype">float</span> split_point = split_points(idx_tree, n_tree);</div><div class="line"><a name="l00698"></a><span class="lineno">  698</span>&#160;            <span class="keywordflow">if</span> (projected_query(j) &lt;= split_point) {</div><div class="line"><a name="l00699"></a><span class="lineno">  699</span>&#160;              idx_tree = idx_left;</div><div class="line"><a name="l00700"></a><span class="lineno">  700</span>&#160;            } <span class="keywordflow">else</span> {</div><div class="line"><a name="l00701"></a><span class="lineno">  701</span>&#160;              idx_tree = idx_right;</div><div class="line"><a name="l00702"></a><span class="lineno">  702</span>&#160;            }</div><div class="line"><a name="l00703"></a><span class="lineno">  703</span>&#160;          }</div><div class="line"><a name="l00704"></a><span class="lineno">  704</span>&#160;          found_leaves[n_tree] = idx_tree - (1 &lt;&lt; depth) + 1;</div><div class="line"><a name="l00705"></a><span class="lineno">  705</span>&#160;        }</div><div class="line"><a name="l00706"></a><span class="lineno">  706</span>&#160;</div><div class="line"><a name="l00707"></a><span class="lineno">  707</span>&#160;        <span class="keywordtype">int</span> n_elected = 0, max_leaf_size = n_samples / (1 &lt;&lt; depth) + 1;</div><div class="line"><a name="l00708"></a><span class="lineno">  708</span>&#160;        Eigen::VectorXi elected(n_trees * max_leaf_size);</div><div class="line"><a name="l00709"></a><span class="lineno">  709</span>&#160;        Eigen::VectorXi votes = Eigen::VectorXi::Zero(n_samples);</div><div class="line"><a name="l00710"></a><span class="lineno">  710</span>&#160;</div><div class="line"><a name="l00711"></a><span class="lineno">  711</span>&#160;        <span class="comment">// count votes</span></div><div class="line"><a name="l00712"></a><span class="lineno">  712</span>&#160;        <span class="keywordflow">for</span> (<span class="keywordtype">int</span> n_tree = 0; n_tree &lt; n_trees; ++n_tree) {</div><div class="line"><a name="l00713"></a><span class="lineno">  713</span>&#160;          <span class="keywordtype">int</span> leaf_begin = leaf_first_indices[found_leaves[n_tree]];</div><div class="line"><a name="l00714"></a><span class="lineno">  714</span>&#160;          <span class="keywordtype">int</span> leaf_end = leaf_first_indices[found_leaves[n_tree] + 1];</div><div class="line"><a name="l00715"></a><span class="lineno">  715</span>&#160;          <span class="keyword">const</span> std::vector&lt;int&gt; &amp;indices = tree_leaves[n_tree];</div><div class="line"><a name="l00716"></a><span class="lineno">  716</span>&#160;          <span class="keywordflow">for</span> (<span class="keywordtype">int</span> i = leaf_begin; i &lt; leaf_end; ++i) {</div><div class="line"><a name="l00717"></a><span class="lineno">  717</span>&#160;            <span class="keywordtype">int</span> idx = indices[i];</div><div class="line"><a name="l00718"></a><span class="lineno">  718</span>&#160;            <span class="keywordflow">if</span> (++votes(idx) == vote_threshold)</div><div class="line"><a name="l00719"></a><span class="lineno">  719</span>&#160;              elected(n_elected++) = idx;</div><div class="line"><a name="l00720"></a><span class="lineno">  720</span>&#160;          }</div><div class="line"><a name="l00721"></a><span class="lineno">  721</span>&#160;        }</div><div class="line"><a name="l00722"></a><span class="lineno">  722</span>&#160;</div><div class="line"><a name="l00723"></a><span class="lineno">  723</span>&#160;        <span class="keywordflow">if</span> (out_n_elected) {</div><div class="line"><a name="l00724"></a><span class="lineno">  724</span>&#160;          *out_n_elected = n_elected;</div><div class="line"><a name="l00725"></a><span class="lineno">  725</span>&#160;        }</div><div class="line"><a name="l00726"></a><span class="lineno">  726</span>&#160;</div><div class="line"><a name="l00727"></a><span class="lineno">  727</span>&#160;        <a class="code" href="class_mrpt.html#a19867c3d3f83e50c88e1edde63a81985">exact_knn</a>(q, k, elected, n_elected, out, out_distances);</div><div class="line"><a name="l00728"></a><span class="lineno">  728</span>&#160;    }</div><div class="line"><a name="l00729"></a><span class="lineno">  729</span>&#160;</div><div class="line"><a name="l00740"></a><span class="lineno"><a class="line" href="class_mrpt.html#a35235eb7acf4bff5fbfb12c9cd82f0b8">  740</a></span>&#160;    <span class="keywordtype">void</span> <a class="code" href="class_mrpt.html#a35235eb7acf4bff5fbfb12c9cd82f0b8">query</a>(<span class="keyword">const</span> Eigen::Ref&lt;const Eigen::VectorXf&gt; &amp;q, <span class="keywordtype">int</span> k, <span class="keywordtype">int</span> vote_threshold, <span class="keywordtype">int</span> *out,</div><div class="line"><a name="l00741"></a><span class="lineno">  741</span>&#160;               <span class="keywordtype">float</span> *out_distances = <span class="keyword">nullptr</span>, <span class="keywordtype">int</span> *out_n_elected = <span class="keyword">nullptr</span>)<span class="keyword"> const </span>{</div><div class="line"><a name="l00742"></a><span class="lineno">  742</span>&#160;      <a class="code" href="class_mrpt.html#a637e67fe6eba9a6711f5eee65ad4a0da">query</a>(q.data(), k, vote_threshold, out, out_distances, out_n_elected);</div><div class="line"><a name="l00743"></a><span class="lineno">  743</span>&#160;    }</div><div class="line"><a name="l00744"></a><span class="lineno">  744</span>&#160;</div><div class="line"><a name="l00767"></a><span class="lineno"><a class="line" href="class_mrpt.html#a16fac96c266abcc778694e0b0cdc5f3c">  767</a></span>&#160;    <span class="keywordtype">void</span> <a class="code" href="class_mrpt.html#a16fac96c266abcc778694e0b0cdc5f3c">query</a>(<span class="keyword">const</span> <span class="keywordtype">float</span> *q, <span class="keywordtype">int</span> *out, <span class="keywordtype">float</span> *out_distances = <span class="keyword">nullptr</span>,</div><div class="line"><a name="l00768"></a><span class="lineno">  768</span>&#160;               <span class="keywordtype">int</span> *out_n_elected = <span class="keyword">nullptr</span>)<span class="keyword"> const </span>{</div><div class="line"><a name="l00769"></a><span class="lineno">  769</span>&#160;      <span class="keywordflow">if</span> (index_type == normal) {</div><div class="line"><a name="l00770"></a><span class="lineno">  770</span>&#160;        <span class="keywordflow">throw</span> std::logic_error(<span class="stringliteral">&quot;The index is not autotuned: k and vote threshold has to be specified.&quot;</span>);</div><div class="line"><a name="l00771"></a><span class="lineno">  771</span>&#160;      }</div><div class="line"><a name="l00772"></a><span class="lineno">  772</span>&#160;</div><div class="line"><a name="l00773"></a><span class="lineno">  773</span>&#160;      <span class="keywordflow">if</span> (index_type == autotuned_unpruned) {</div><div class="line"><a name="l00774"></a><span class="lineno">  774</span>&#160;        <span class="keywordflow">throw</span> std::logic_error(<span class="stringliteral">&quot;The target recall level has to be set before making queries.&quot;</span>);</div><div class="line"><a name="l00775"></a><span class="lineno">  775</span>&#160;      }</div><div class="line"><a name="l00776"></a><span class="lineno">  776</span>&#160;</div><div class="line"><a name="l00777"></a><span class="lineno">  777</span>&#160;      <a class="code" href="class_mrpt.html#a637e67fe6eba9a6711f5eee65ad4a0da">query</a>(q, k, votes, out, out_distances, out_n_elected);</div><div class="line"><a name="l00778"></a><span class="lineno">  778</span>&#160;    }</div><div class="line"><a name="l00779"></a><span class="lineno">  779</span>&#160;</div><div class="line"><a name="l00788"></a><span class="lineno"><a class="line" href="class_mrpt.html#a9579b70d0553126363f6112a10ffb0eb">  788</a></span>&#160;    <span class="keywordtype">void</span> <a class="code" href="class_mrpt.html#a9579b70d0553126363f6112a10ffb0eb">query</a>(<span class="keyword">const</span> Eigen::Ref&lt;const Eigen::VectorXf&gt; &amp;q, <span class="keywordtype">int</span> *out, <span class="keywordtype">float</span> *out_distances = <span class="keyword">nullptr</span>,</div><div class="line"><a name="l00789"></a><span class="lineno">  789</span>&#160;               <span class="keywordtype">int</span> *out_n_elected = <span class="keyword">nullptr</span>)<span class="keyword"> const </span>{</div><div class="line"><a name="l00790"></a><span class="lineno">  790</span>&#160;      <a class="code" href="class_mrpt.html#a637e67fe6eba9a6711f5eee65ad4a0da">query</a>(q.data(), out, out_distances, out_n_elected);</div><div class="line"><a name="l00791"></a><span class="lineno">  791</span>&#160;    }</div><div class="line"><a name="l00792"></a><span class="lineno">  792</span>&#160;</div><div class="line"><a name="l00814"></a><span class="lineno"><a class="line" href="class_mrpt.html#a19867c3d3f83e50c88e1edde63a81985">  814</a></span>&#160;    <span class="keyword">static</span> <span class="keywordtype">void</span> <a class="code" href="class_mrpt.html#a19867c3d3f83e50c88e1edde63a81985">exact_knn</a>(<span class="keyword">const</span> <span class="keywordtype">float</span> *q_data, <span class="keyword">const</span> <span class="keywordtype">float</span> *X_data, <span class="keywordtype">int</span> dim, <span class="keywordtype">int</span> n_samples,</div><div class="line"><a name="l00815"></a><span class="lineno">  815</span>&#160;        <span class="keywordtype">int</span> k, <span class="keywordtype">int</span> *out, <span class="keywordtype">float</span> *out_distances = <span class="keyword">nullptr</span>) {</div><div class="line"><a name="l00816"></a><span class="lineno">  816</span>&#160;</div><div class="line"><a name="l00817"></a><span class="lineno">  817</span>&#160;      <span class="keyword">const</span> Eigen::Map&lt;const Eigen::MatrixXf&gt; X(X_data, dim, n_samples);</div><div class="line"><a name="l00818"></a><span class="lineno">  818</span>&#160;      <span class="keyword">const</span> Eigen::Map&lt;const Eigen::VectorXf&gt; q(q_data, dim);</div><div class="line"><a name="l00819"></a><span class="lineno">  819</span>&#160;</div><div class="line"><a name="l00820"></a><span class="lineno">  820</span>&#160;      <span class="keywordflow">if</span> (k &lt; 1 || k &gt; n_samples) {</div><div class="line"><a name="l00821"></a><span class="lineno">  821</span>&#160;        <span class="keywordflow">throw</span> std::out_of_range(<span class="stringliteral">&quot;k must be positive and no greater than the sample size of data X.&quot;</span>);</div><div class="line"><a name="l00822"></a><span class="lineno">  822</span>&#160;      }</div><div class="line"><a name="l00823"></a><span class="lineno">  823</span>&#160;</div><div class="line"><a name="l00824"></a><span class="lineno">  824</span>&#160;      Eigen::VectorXf distances(n_samples);</div><div class="line"><a name="l00825"></a><span class="lineno">  825</span>&#160;</div><div class="line"><a name="l00826"></a><span class="lineno">  826</span>&#160;<span class="preprocessor">      #pragma omp parallel for</span></div><div class="line"><a name="l00827"></a><span class="lineno">  827</span>&#160;      <span class="keywordflow">for</span> (<span class="keywordtype">int</span> i = 0; i &lt; n_samples; ++i)</div><div class="line"><a name="l00828"></a><span class="lineno">  828</span>&#160;        distances(i) = (X.col(i) - q).squaredNorm();</div><div class="line"><a name="l00829"></a><span class="lineno">  829</span>&#160;</div><div class="line"><a name="l00830"></a><span class="lineno">  830</span>&#160;      <span class="keywordflow">if</span> (k == 1) {</div><div class="line"><a name="l00831"></a><span class="lineno">  831</span>&#160;        Eigen::MatrixXf::Index index;</div><div class="line"><a name="l00832"></a><span class="lineno">  832</span>&#160;        distances.minCoeff(&amp;index);</div><div class="line"><a name="l00833"></a><span class="lineno">  833</span>&#160;        out[0] = index;</div><div class="line"><a name="l00834"></a><span class="lineno">  834</span>&#160;</div><div class="line"><a name="l00835"></a><span class="lineno">  835</span>&#160;        <span class="keywordflow">if</span> (out_distances)</div><div class="line"><a name="l00836"></a><span class="lineno">  836</span>&#160;          out_distances[0] = std::sqrt(distances(index));</div><div class="line"><a name="l00837"></a><span class="lineno">  837</span>&#160;</div><div class="line"><a name="l00838"></a><span class="lineno">  838</span>&#160;        <span class="keywordflow">return</span>;</div><div class="line"><a name="l00839"></a><span class="lineno">  839</span>&#160;      }</div><div class="line"><a name="l00840"></a><span class="lineno">  840</span>&#160;</div><div class="line"><a name="l00841"></a><span class="lineno">  841</span>&#160;      Eigen::VectorXi idx(n_samples);</div><div class="line"><a name="l00842"></a><span class="lineno">  842</span>&#160;      std::iota(idx.data(), idx.data() + n_samples, 0);</div><div class="line"><a name="l00843"></a><span class="lineno">  843</span>&#160;      std::partial_sort(idx.data(), idx.data() + k, idx.data() + n_samples,</div><div class="line"><a name="l00844"></a><span class="lineno">  844</span>&#160;                       [&amp;distances](<span class="keywordtype">int</span> i1, <span class="keywordtype">int</span> i2) { <span class="keywordflow">return</span> distances(i1) &lt; distances(i2); });</div><div class="line"><a name="l00845"></a><span class="lineno">  845</span>&#160;</div><div class="line"><a name="l00846"></a><span class="lineno">  846</span>&#160;      <span class="keywordflow">for</span> (<span class="keywordtype">int</span> i = 0; i &lt; k; ++i)</div><div class="line"><a name="l00847"></a><span class="lineno">  847</span>&#160;        out[i] = idx(i);</div><div class="line"><a name="l00848"></a><span class="lineno">  848</span>&#160;</div><div class="line"><a name="l00849"></a><span class="lineno">  849</span>&#160;      <span class="keywordflow">if</span> (out_distances) {</div><div class="line"><a name="l00850"></a><span class="lineno">  850</span>&#160;        <span class="keywordflow">for</span> (<span class="keywordtype">int</span> i = 0; i &lt; k; ++i)</div><div class="line"><a name="l00851"></a><span class="lineno">  851</span>&#160;          out_distances[i] = std::sqrt(distances(idx(i)));</div><div class="line"><a name="l00852"></a><span class="lineno">  852</span>&#160;      }</div><div class="line"><a name="l00853"></a><span class="lineno">  853</span>&#160;    }</div><div class="line"><a name="l00854"></a><span class="lineno">  854</span>&#160;</div><div class="line"><a name="l00862"></a><span class="lineno"><a class="line" href="class_mrpt.html#ab06ab10bca1f953c6581f7d412b4661c">  862</a></span>&#160;    <span class="keyword">static</span> <span class="keywordtype">void</span> <a class="code" href="class_mrpt.html#ab06ab10bca1f953c6581f7d412b4661c">exact_knn</a>(<span class="keyword">const</span> Eigen::Ref&lt;const Eigen::VectorXf&gt; &amp;q,</div><div class="line"><a name="l00863"></a><span class="lineno">  863</span>&#160;                          <span class="keyword">const</span> Eigen::Ref&lt;const Eigen::MatrixXf&gt; &amp;X,</div><div class="line"><a name="l00864"></a><span class="lineno">  864</span>&#160;                          <span class="keywordtype">int</span> k, <span class="keywordtype">int</span> *out, <span class="keywordtype">float</span> *out_distances = <span class="keyword">nullptr</span>) {</div><div class="line"><a name="l00865"></a><span class="lineno">  865</span>&#160;      <a class="code" href="class_mrpt.html#a19867c3d3f83e50c88e1edde63a81985">Mrpt::exact_knn</a>(q.data(), X.data(), X.rows(), X.cols(), k, out, out_distances);</div><div class="line"><a name="l00866"></a><span class="lineno">  866</span>&#160;    }</div><div class="line"><a name="l00867"></a><span class="lineno">  867</span>&#160;</div><div class="line"><a name="l00874"></a><span class="lineno"><a class="line" href="class_mrpt.html#a708704bc812a926c1dc950c267ae513b">  874</a></span>&#160;    <span class="keywordtype">void</span> <a class="code" href="class_mrpt.html#a708704bc812a926c1dc950c267ae513b">exact_knn</a>(<span class="keyword">const</span> <span class="keywordtype">float</span> *q, <span class="keywordtype">int</span> k, <span class="keywordtype">int</span> *out, <span class="keywordtype">float</span> *out_distances = <span class="keyword">nullptr</span>)<span class="keyword"> const </span>{</div><div class="line"><a name="l00875"></a><span class="lineno">  875</span>&#160;      <a class="code" href="class_mrpt.html#a19867c3d3f83e50c88e1edde63a81985">Mrpt::exact_knn</a>(q, X.data(), dim, n_samples, k, out, out_distances);</div><div class="line"><a name="l00876"></a><span class="lineno">  876</span>&#160;    }</div><div class="line"><a name="l00877"></a><span class="lineno">  877</span>&#160;</div><div class="line"><a name="l00884"></a><span class="lineno"><a class="line" href="class_mrpt.html#a24f2dfef8d422cb875abd18d411d47a1">  884</a></span>&#160;    <span class="keywordtype">void</span> <a class="code" href="class_mrpt.html#a24f2dfef8d422cb875abd18d411d47a1">exact_knn</a>(<span class="keyword">const</span> Eigen::Ref&lt;const Eigen::VectorXf&gt; &amp;q, <span class="keywordtype">int</span> k, <span class="keywordtype">int</span> *out,</div><div class="line"><a name="l00885"></a><span class="lineno">  885</span>&#160;        <span class="keywordtype">float</span> *out_distances = <span class="keyword">nullptr</span>)<span class="keyword"> const </span>{</div><div class="line"><a name="l00886"></a><span class="lineno">  886</span>&#160;      <a class="code" href="class_mrpt.html#a19867c3d3f83e50c88e1edde63a81985">Mrpt::exact_knn</a>(q.data(), X.data(), dim, n_samples, k, out, out_distances);</div><div class="line"><a name="l00887"></a><span class="lineno">  887</span>&#160;    }</div><div class="line"><a name="l00888"></a><span class="lineno">  888</span>&#160;</div><div class="line"><a name="l00905"></a><span class="lineno"><a class="line" href="class_mrpt.html#af936713e7629e08e88d11346f2639ba0">  905</a></span>&#160;    <span class="keywordtype">bool</span> <a class="code" href="class_mrpt.html#af936713e7629e08e88d11346f2639ba0">save</a>(<span class="keyword">const</span> <span class="keywordtype">char</span> *path)<span class="keyword"> const </span>{</div><div class="line"><a name="l00906"></a><span class="lineno">  906</span>&#160;      FILE *fd;</div><div class="line"><a name="l00907"></a><span class="lineno">  907</span>&#160;      <span class="keywordflow">if</span> ((fd = fopen(path, <span class="stringliteral">&quot;wb&quot;</span>)) == NULL)</div><div class="line"><a name="l00908"></a><span class="lineno">  908</span>&#160;        <span class="keywordflow">return</span> <span class="keyword">false</span>;</div><div class="line"><a name="l00909"></a><span class="lineno">  909</span>&#160;</div><div class="line"><a name="l00910"></a><span class="lineno">  910</span>&#160;      <span class="keywordtype">int</span> i = index_type;</div><div class="line"><a name="l00911"></a><span class="lineno">  911</span>&#160;      fwrite(&amp;i, <span class="keyword">sizeof</span>(<span class="keywordtype">int</span>), 1, fd);</div><div class="line"><a name="l00912"></a><span class="lineno">  912</span>&#160;</div><div class="line"><a name="l00913"></a><span class="lineno">  913</span>&#160;      <span class="keywordflow">if</span> (index_type == 2) {</div><div class="line"><a name="l00914"></a><span class="lineno">  914</span>&#160;        write_parameter_list(opt_pars, fd);</div><div class="line"><a name="l00915"></a><span class="lineno">  915</span>&#160;      }</div><div class="line"><a name="l00916"></a><span class="lineno">  916</span>&#160;</div><div class="line"><a name="l00917"></a><span class="lineno">  917</span>&#160;      write_parameters(&amp;par, fd);</div><div class="line"><a name="l00918"></a><span class="lineno">  918</span>&#160;      fwrite(&amp;n_trees, <span class="keyword">sizeof</span>(<span class="keywordtype">int</span>), 1, fd);</div><div class="line"><a name="l00919"></a><span class="lineno">  919</span>&#160;      fwrite(&amp;depth, <span class="keyword">sizeof</span>(<span class="keywordtype">int</span>), 1, fd);</div><div class="line"><a name="l00920"></a><span class="lineno">  920</span>&#160;      fwrite(&amp;density, <span class="keyword">sizeof</span>(<span class="keywordtype">float</span>), 1, fd);</div><div class="line"><a name="l00921"></a><span class="lineno">  921</span>&#160;</div><div class="line"><a name="l00922"></a><span class="lineno">  922</span>&#160;      fwrite(split_points.data(), <span class="keyword">sizeof</span>(float), n_array * n_trees, fd);</div><div class="line"><a name="l00923"></a><span class="lineno">  923</span>&#160;</div><div class="line"><a name="l00924"></a><span class="lineno">  924</span>&#160;      <span class="comment">// save tree leaves</span></div><div class="line"><a name="l00925"></a><span class="lineno">  925</span>&#160;      <span class="keywordflow">for</span> (<span class="keywordtype">int</span> i = 0; i &lt; n_trees; ++i) {</div><div class="line"><a name="l00926"></a><span class="lineno">  926</span>&#160;        <span class="keywordtype">int</span> sz = tree_leaves[i].size();</div><div class="line"><a name="l00927"></a><span class="lineno">  927</span>&#160;        fwrite(&amp;sz, <span class="keyword">sizeof</span>(<span class="keywordtype">int</span>), 1, fd);</div><div class="line"><a name="l00928"></a><span class="lineno">  928</span>&#160;        fwrite(&amp;tree_leaves[i][0], <span class="keyword">sizeof</span>(<span class="keywordtype">int</span>), sz, fd);</div><div class="line"><a name="l00929"></a><span class="lineno">  929</span>&#160;      }</div><div class="line"><a name="l00930"></a><span class="lineno">  930</span>&#160;</div><div class="line"><a name="l00931"></a><span class="lineno">  931</span>&#160;      <span class="comment">// save random matrix</span></div><div class="line"><a name="l00932"></a><span class="lineno">  932</span>&#160;      <span class="keywordflow">if</span> (density &lt; 1) {</div><div class="line"><a name="l00933"></a><span class="lineno">  933</span>&#160;        <span class="keywordtype">int</span> non_zeros = sparse_random_matrix.nonZeros();</div><div class="line"><a name="l00934"></a><span class="lineno">  934</span>&#160;        fwrite(&amp;non_zeros, <span class="keyword">sizeof</span>(<span class="keywordtype">int</span>), 1, fd);</div><div class="line"><a name="l00935"></a><span class="lineno">  935</span>&#160;        <span class="keywordflow">for</span> (<span class="keywordtype">int</span> k = 0; k &lt; sparse_random_matrix.outerSize(); ++k) {</div><div class="line"><a name="l00936"></a><span class="lineno">  936</span>&#160;          <span class="keywordflow">for</span> (Eigen::SparseMatrix&lt;float, Eigen::RowMajor&gt;::InnerIterator it(sparse_random_matrix, k); it; ++it) {</div><div class="line"><a name="l00937"></a><span class="lineno">  937</span>&#160;            <span class="keywordtype">float</span> val = it.value();</div><div class="line"><a name="l00938"></a><span class="lineno">  938</span>&#160;            <span class="keywordtype">int</span> row = it.row(), col = it.col();</div><div class="line"><a name="l00939"></a><span class="lineno">  939</span>&#160;            fwrite(&amp;row, <span class="keyword">sizeof</span>(<span class="keywordtype">int</span>), 1, fd);</div><div class="line"><a name="l00940"></a><span class="lineno">  940</span>&#160;            fwrite(&amp;col, <span class="keyword">sizeof</span>(<span class="keywordtype">int</span>), 1, fd);</div><div class="line"><a name="l00941"></a><span class="lineno">  941</span>&#160;            fwrite(&amp;val, <span class="keyword">sizeof</span>(<span class="keywordtype">float</span>), 1, fd);</div><div class="line"><a name="l00942"></a><span class="lineno">  942</span>&#160;          }</div><div class="line"><a name="l00943"></a><span class="lineno">  943</span>&#160;        }</div><div class="line"><a name="l00944"></a><span class="lineno">  944</span>&#160;      } <span class="keywordflow">else</span> {</div><div class="line"><a name="l00945"></a><span class="lineno">  945</span>&#160;        fwrite(dense_random_matrix.data(), <span class="keyword">sizeof</span>(float), n_pool * dim, fd);</div><div class="line"><a name="l00946"></a><span class="lineno">  946</span>&#160;      }</div><div class="line"><a name="l00947"></a><span class="lineno">  947</span>&#160;</div><div class="line"><a name="l00948"></a><span class="lineno">  948</span>&#160;      fclose(fd);</div><div class="line"><a name="l00949"></a><span class="lineno">  949</span>&#160;      <span class="keywordflow">return</span> <span class="keyword">true</span>;</div><div class="line"><a name="l00950"></a><span class="lineno">  950</span>&#160;    }</div><div class="line"><a name="l00951"></a><span class="lineno">  951</span>&#160;</div><div class="line"><a name="l00958"></a><span class="lineno"><a class="line" href="class_mrpt.html#af31f777278ec7030e7351b6ed57ca766">  958</a></span>&#160;    <span class="keywordtype">bool</span> <a class="code" href="class_mrpt.html#af31f777278ec7030e7351b6ed57ca766">load</a>(<span class="keyword">const</span> <span class="keywordtype">char</span> *path) {</div><div class="line"><a name="l00959"></a><span class="lineno">  959</span>&#160;      FILE *fd;</div><div class="line"><a name="l00960"></a><span class="lineno">  960</span>&#160;      <span class="keywordflow">if</span> ((fd = fopen(path, <span class="stringliteral">&quot;rb&quot;</span>)) == NULL)</div><div class="line"><a name="l00961"></a><span class="lineno">  961</span>&#160;        <span class="keywordflow">return</span> <span class="keyword">false</span>;</div><div class="line"><a name="l00962"></a><span class="lineno">  962</span>&#160;</div><div class="line"><a name="l00963"></a><span class="lineno">  963</span>&#160;      <span class="keywordtype">int</span> i;</div><div class="line"><a name="l00964"></a><span class="lineno">  964</span>&#160;      fread(&amp;i, <span class="keyword">sizeof</span>(<span class="keywordtype">int</span>), 1, fd);</div><div class="line"><a name="l00965"></a><span class="lineno">  965</span>&#160;      index_type = <span class="keyword">static_cast&lt;</span>itype<span class="keyword">&gt;</span>(i);</div><div class="line"><a name="l00966"></a><span class="lineno">  966</span>&#160;      <span class="keywordflow">if</span> (index_type == autotuned_unpruned) {</div><div class="line"><a name="l00967"></a><span class="lineno">  967</span>&#160;        read_parameter_list(fd);</div><div class="line"><a name="l00968"></a><span class="lineno">  968</span>&#160;      }</div><div class="line"><a name="l00969"></a><span class="lineno">  969</span>&#160;</div><div class="line"><a name="l00970"></a><span class="lineno">  970</span>&#160;      read_parameters(&amp;par, fd);</div><div class="line"><a name="l00971"></a><span class="lineno">  971</span>&#160;      fread(&amp;n_trees, <span class="keyword">sizeof</span>(<span class="keywordtype">int</span>), 1, fd);</div><div class="line"><a name="l00972"></a><span class="lineno">  972</span>&#160;      fread(&amp;depth, <span class="keyword">sizeof</span>(<span class="keywordtype">int</span>), 1, fd);</div><div class="line"><a name="l00973"></a><span class="lineno">  973</span>&#160;      fread(&amp;density, <span class="keyword">sizeof</span>(<span class="keywordtype">float</span>), 1, fd);</div><div class="line"><a name="l00974"></a><span class="lineno">  974</span>&#160;</div><div class="line"><a name="l00975"></a><span class="lineno">  975</span>&#160;      n_pool = n_trees * depth;</div><div class="line"><a name="l00976"></a><span class="lineno">  976</span>&#160;      n_array = 1 &lt;&lt; (depth + 1);</div><div class="line"><a name="l00977"></a><span class="lineno">  977</span>&#160;</div><div class="line"><a name="l00978"></a><span class="lineno">  978</span>&#160;      count_first_leaf_indices_all(leaf_first_indices_all, n_samples, depth);</div><div class="line"><a name="l00979"></a><span class="lineno">  979</span>&#160;      leaf_first_indices = leaf_first_indices_all[depth];</div><div class="line"><a name="l00980"></a><span class="lineno">  980</span>&#160;</div><div class="line"><a name="l00981"></a><span class="lineno">  981</span>&#160;      split_points = Eigen::MatrixXf(n_array, n_trees);</div><div class="line"><a name="l00982"></a><span class="lineno">  982</span>&#160;      fread(split_points.data(), <span class="keyword">sizeof</span>(float), n_array * n_trees, fd);</div><div class="line"><a name="l00983"></a><span class="lineno">  983</span>&#160;</div><div class="line"><a name="l00984"></a><span class="lineno">  984</span>&#160;      <span class="comment">// load tree leaves</span></div><div class="line"><a name="l00985"></a><span class="lineno">  985</span>&#160;      tree_leaves = std::vector&lt;std::vector&lt;int&gt;&gt;(n_trees);</div><div class="line"><a name="l00986"></a><span class="lineno">  986</span>&#160;      <span class="keywordflow">for</span> (<span class="keywordtype">int</span> i = 0; i &lt; n_trees; ++i) {</div><div class="line"><a name="l00987"></a><span class="lineno">  987</span>&#160;        <span class="keywordtype">int</span> sz;</div><div class="line"><a name="l00988"></a><span class="lineno">  988</span>&#160;        fread(&amp;sz, <span class="keyword">sizeof</span>(<span class="keywordtype">int</span>), 1, fd);</div><div class="line"><a name="l00989"></a><span class="lineno">  989</span>&#160;        std::vector&lt;int&gt; leaves(sz);</div><div class="line"><a name="l00990"></a><span class="lineno">  990</span>&#160;        fread(&amp;leaves[0], <span class="keyword">sizeof</span>(<span class="keywordtype">int</span>), sz, fd);</div><div class="line"><a name="l00991"></a><span class="lineno">  991</span>&#160;        tree_leaves[i] = leaves;</div><div class="line"><a name="l00992"></a><span class="lineno">  992</span>&#160;      }</div><div class="line"><a name="l00993"></a><span class="lineno">  993</span>&#160;</div><div class="line"><a name="l00994"></a><span class="lineno">  994</span>&#160;      <span class="comment">// load random matrix</span></div><div class="line"><a name="l00995"></a><span class="lineno">  995</span>&#160;      <span class="keywordflow">if</span> (density &lt; 1) {</div><div class="line"><a name="l00996"></a><span class="lineno">  996</span>&#160;        <span class="keywordtype">int</span> non_zeros;</div><div class="line"><a name="l00997"></a><span class="lineno">  997</span>&#160;        fread(&amp;non_zeros, <span class="keyword">sizeof</span>(<span class="keywordtype">int</span>), 1, fd);</div><div class="line"><a name="l00998"></a><span class="lineno">  998</span>&#160;</div><div class="line"><a name="l00999"></a><span class="lineno">  999</span>&#160;        sparse_random_matrix = Eigen::SparseMatrix&lt;float&gt;(n_pool, dim);</div><div class="line"><a name="l01000"></a><span class="lineno"> 1000</span>&#160;        std::vector&lt;Eigen::Triplet&lt;float&gt;&gt; triplets;</div><div class="line"><a name="l01001"></a><span class="lineno"> 1001</span>&#160;        <span class="keywordflow">for</span> (<span class="keywordtype">int</span> k = 0; k &lt; non_zeros; ++k) {</div><div class="line"><a name="l01002"></a><span class="lineno"> 1002</span>&#160;          <span class="keywordtype">int</span> row, col;</div><div class="line"><a name="l01003"></a><span class="lineno"> 1003</span>&#160;          <span class="keywordtype">float</span> val;</div><div class="line"><a name="l01004"></a><span class="lineno"> 1004</span>&#160;          fread(&amp;row, <span class="keyword">sizeof</span>(<span class="keywordtype">int</span>), 1, fd);</div><div class="line"><a name="l01005"></a><span class="lineno"> 1005</span>&#160;          fread(&amp;col, <span class="keyword">sizeof</span>(<span class="keywordtype">int</span>), 1, fd);</div><div class="line"><a name="l01006"></a><span class="lineno"> 1006</span>&#160;          fread(&amp;val, <span class="keyword">sizeof</span>(<span class="keywordtype">float</span>), 1, fd);</div><div class="line"><a name="l01007"></a><span class="lineno"> 1007</span>&#160;          triplets.push_back(Eigen::Triplet&lt;float&gt;(row, col, val));</div><div class="line"><a name="l01008"></a><span class="lineno"> 1008</span>&#160;        }</div><div class="line"><a name="l01009"></a><span class="lineno"> 1009</span>&#160;</div><div class="line"><a name="l01010"></a><span class="lineno"> 1010</span>&#160;        sparse_random_matrix.setFromTriplets(triplets.begin(), triplets.end());</div><div class="line"><a name="l01011"></a><span class="lineno"> 1011</span>&#160;        sparse_random_matrix.makeCompressed();</div><div class="line"><a name="l01012"></a><span class="lineno"> 1012</span>&#160;      } <span class="keywordflow">else</span> {</div><div class="line"><a name="l01013"></a><span class="lineno"> 1013</span>&#160;        dense_random_matrix = Eigen::Matrix&lt;float, Eigen::Dynamic, Eigen::Dynamic, Eigen::RowMajor&gt;(n_pool, dim);</div><div class="line"><a name="l01014"></a><span class="lineno"> 1014</span>&#160;        fread(dense_random_matrix.data(), <span class="keyword">sizeof</span>(float), n_pool * dim, fd);</div><div class="line"><a name="l01015"></a><span class="lineno"> 1015</span>&#160;      }</div><div class="line"><a name="l01016"></a><span class="lineno"> 1016</span>&#160;</div><div class="line"><a name="l01017"></a><span class="lineno"> 1017</span>&#160;      fclose(fd);</div><div class="line"><a name="l01018"></a><span class="lineno"> 1018</span>&#160;</div><div class="line"><a name="l01019"></a><span class="lineno"> 1019</span>&#160;      k = par.<a class="code" href="struct_mrpt___parameters.html#ad8b374d6671223a498e67cb98d8ea191">k</a>;</div><div class="line"><a name="l01020"></a><span class="lineno"> 1020</span>&#160;      votes = par.<a class="code" href="struct_mrpt___parameters.html#ab34a16df707e0abd7b8b0bf4959f36b6">votes</a>;</div><div class="line"><a name="l01021"></a><span class="lineno"> 1021</span>&#160;      <span class="keywordflow">return</span> <span class="keyword">true</span>;</div><div class="line"><a name="l01022"></a><span class="lineno"> 1022</span>&#160;    }</div><div class="line"><a name="l01023"></a><span class="lineno"> 1023</span>&#160;</div><div class="line"><a name="l01029"></a><span class="lineno"><a class="line" href="class_mrpt.html#a960f1af81f2b56db7eb8cd0d5a352c14"> 1029</a></span>&#160;    <span class="keywordtype">bool</span> <a class="code" href="class_mrpt.html#a960f1af81f2b56db7eb8cd0d5a352c14">empty</a>()<span class="keyword"> const </span>{</div><div class="line"><a name="l01030"></a><span class="lineno"> 1030</span>&#160;      <span class="keywordflow">return</span> n_trees == 0;</div><div class="line"><a name="l01031"></a><span class="lineno"> 1031</span>&#160;    }</div><div class="line"><a name="l01032"></a><span class="lineno"> 1032</span>&#160;</div><div class="line"><a name="l01039"></a><span class="lineno"> 1039</span>&#160;    <span class="keyword">friend</span> <span class="keyword">class </span>MrptTest;</div><div class="line"><a name="l01040"></a><span class="lineno"> 1040</span>&#160;    <span class="keyword">friend</span> <span class="keyword">class </span>UtilityTest;</div><div class="line"><a name="l01041"></a><span class="lineno"> 1041</span>&#160;</div><div class="line"><a name="l01045"></a><span class="lineno"> 1045</span>&#160; <span class="keyword">private</span>:</div><div class="line"><a name="l01046"></a><span class="lineno"> 1046</span>&#160;</div><div class="line"><a name="l01051"></a><span class="lineno"> 1051</span>&#160;    <span class="keywordtype">void</span> grow_subtree(std::vector&lt;int&gt;::iterator begin, std::vector&lt;int&gt;::iterator end,</div><div class="line"><a name="l01052"></a><span class="lineno"> 1052</span>&#160;          <span class="keywordtype">int</span> tree_level, <span class="keywordtype">int</span> i, <span class="keywordtype">int</span> n_tree, <span class="keyword">const</span> Eigen::MatrixXf &amp;tree_projections) {</div><div class="line"><a name="l01053"></a><span class="lineno"> 1053</span>&#160;      <span class="keywordtype">int</span> n = end - begin;</div><div class="line"><a name="l01054"></a><span class="lineno"> 1054</span>&#160;      <span class="keywordtype">int</span> idx_left = 2 * i + 1;</div><div class="line"><a name="l01055"></a><span class="lineno"> 1055</span>&#160;      <span class="keywordtype">int</span> idx_right = idx_left + 1;</div><div class="line"><a name="l01056"></a><span class="lineno"> 1056</span>&#160;</div><div class="line"><a name="l01057"></a><span class="lineno"> 1057</span>&#160;      <span class="keywordflow">if</span> (tree_level == depth) <span class="keywordflow">return</span>;</div><div class="line"><a name="l01058"></a><span class="lineno"> 1058</span>&#160;</div><div class="line"><a name="l01059"></a><span class="lineno"> 1059</span>&#160;      std::nth_element(begin, begin + n / 2, end,</div><div class="line"><a name="l01060"></a><span class="lineno"> 1060</span>&#160;          [&amp;tree_projections, tree_level] (<span class="keywordtype">int</span> i1, <span class="keywordtype">int</span> i2) {</div><div class="line"><a name="l01061"></a><span class="lineno"> 1061</span>&#160;            <span class="keywordflow">return</span> tree_projections(tree_level, i1) &lt; tree_projections(tree_level, i2);</div><div class="line"><a name="l01062"></a><span class="lineno"> 1062</span>&#160;          });</div><div class="line"><a name="l01063"></a><span class="lineno"> 1063</span>&#160;      <span class="keyword">auto</span> mid = end - n / 2;</div><div class="line"><a name="l01064"></a><span class="lineno"> 1064</span>&#160;</div><div class="line"><a name="l01065"></a><span class="lineno"> 1065</span>&#160;      <span class="keywordflow">if</span> (n % 2) {</div><div class="line"><a name="l01066"></a><span class="lineno"> 1066</span>&#160;        split_points(i, n_tree) = tree_projections(tree_level, *(mid - 1));</div><div class="line"><a name="l01067"></a><span class="lineno"> 1067</span>&#160;      } <span class="keywordflow">else</span> {</div><div class="line"><a name="l01068"></a><span class="lineno"> 1068</span>&#160;        <span class="keyword">auto</span> left_it = std::max_element(begin, mid,</div><div class="line"><a name="l01069"></a><span class="lineno"> 1069</span>&#160;            [&amp;tree_projections, tree_level] (<span class="keywordtype">int</span> i1, <span class="keywordtype">int</span> i2) {</div><div class="line"><a name="l01070"></a><span class="lineno"> 1070</span>&#160;              <span class="keywordflow">return</span> tree_projections(tree_level, i1) &lt; tree_projections(tree_level, i2);</div><div class="line"><a name="l01071"></a><span class="lineno"> 1071</span>&#160;            });</div><div class="line"><a name="l01072"></a><span class="lineno"> 1072</span>&#160;        split_points(i, n_tree) = (tree_projections(tree_level, *mid) +</div><div class="line"><a name="l01073"></a><span class="lineno"> 1073</span>&#160;          tree_projections(tree_level, *left_it)) / 2.0;</div><div class="line"><a name="l01074"></a><span class="lineno"> 1074</span>&#160;      }</div><div class="line"><a name="l01075"></a><span class="lineno"> 1075</span>&#160;</div><div class="line"><a name="l01076"></a><span class="lineno"> 1076</span>&#160;      grow_subtree(begin, mid, tree_level + 1, idx_left, n_tree, tree_projections);</div><div class="line"><a name="l01077"></a><span class="lineno"> 1077</span>&#160;      grow_subtree(mid, end, tree_level + 1, idx_right, n_tree, tree_projections);</div><div class="line"><a name="l01078"></a><span class="lineno"> 1078</span>&#160;    }</div><div class="line"><a name="l01079"></a><span class="lineno"> 1079</span>&#160;</div><div class="line"><a name="l01083"></a><span class="lineno"> 1083</span>&#160;    <span class="keywordtype">void</span> <a class="code" href="class_mrpt.html#a19867c3d3f83e50c88e1edde63a81985">exact_knn</a>(<span class="keyword">const</span> Eigen::Map&lt;const Eigen::VectorXf&gt; &amp;q, <span class="keywordtype">int</span> k, <span class="keyword">const</span> Eigen::VectorXi &amp;indices,</div><div class="line"><a name="l01084"></a><span class="lineno"> 1084</span>&#160;                   <span class="keywordtype">int</span> n_elected, <span class="keywordtype">int</span> *out, <span class="keywordtype">float</span> *out_distances = <span class="keyword">nullptr</span>)<span class="keyword"> const </span>{</div><div class="line"><a name="l01085"></a><span class="lineno"> 1085</span>&#160;</div><div class="line"><a name="l01086"></a><span class="lineno"> 1086</span>&#160;      <span class="keywordflow">if</span> (!n_elected) {</div><div class="line"><a name="l01087"></a><span class="lineno"> 1087</span>&#160;        <span class="keywordflow">for</span> (<span class="keywordtype">int</span> i = 0; i &lt; k; ++i)</div><div class="line"><a name="l01088"></a><span class="lineno"> 1088</span>&#160;          out[i] = -1;</div><div class="line"><a name="l01089"></a><span class="lineno"> 1089</span>&#160;</div><div class="line"><a name="l01090"></a><span class="lineno"> 1090</span>&#160;        <span class="keywordflow">if</span> (out_distances) {</div><div class="line"><a name="l01091"></a><span class="lineno"> 1091</span>&#160;          <span class="keywordflow">for</span> (<span class="keywordtype">int</span> i = 0; i &lt; k; ++i)</div><div class="line"><a name="l01092"></a><span class="lineno"> 1092</span>&#160;            out_distances[i] = -1;</div><div class="line"><a name="l01093"></a><span class="lineno"> 1093</span>&#160;        }</div><div class="line"><a name="l01094"></a><span class="lineno"> 1094</span>&#160;</div><div class="line"><a name="l01095"></a><span class="lineno"> 1095</span>&#160;        <span class="keywordflow">return</span>;</div><div class="line"><a name="l01096"></a><span class="lineno"> 1096</span>&#160;      }</div><div class="line"><a name="l01097"></a><span class="lineno"> 1097</span>&#160;</div><div class="line"><a name="l01098"></a><span class="lineno"> 1098</span>&#160;      Eigen::VectorXf distances(n_elected);</div><div class="line"><a name="l01099"></a><span class="lineno"> 1099</span>&#160;</div><div class="line"><a name="l01100"></a><span class="lineno"> 1100</span>&#160;<span class="preprocessor">      #pragma omp parallel for</span></div><div class="line"><a name="l01101"></a><span class="lineno"> 1101</span>&#160;      <span class="keywordflow">for</span> (<span class="keywordtype">int</span> i = 0; i &lt; n_elected; ++i)</div><div class="line"><a name="l01102"></a><span class="lineno"> 1102</span>&#160;        distances(i) = (X.col(indices(i)) - q).squaredNorm();</div><div class="line"><a name="l01103"></a><span class="lineno"> 1103</span>&#160;</div><div class="line"><a name="l01104"></a><span class="lineno"> 1104</span>&#160;      <span class="keywordflow">if</span> (k == 1) {</div><div class="line"><a name="l01105"></a><span class="lineno"> 1105</span>&#160;        Eigen::MatrixXf::Index index;</div><div class="line"><a name="l01106"></a><span class="lineno"> 1106</span>&#160;        distances.minCoeff(&amp;index);</div><div class="line"><a name="l01107"></a><span class="lineno"> 1107</span>&#160;        out[0] = n_elected ? indices(index) : -1;</div><div class="line"><a name="l01108"></a><span class="lineno"> 1108</span>&#160;</div><div class="line"><a name="l01109"></a><span class="lineno"> 1109</span>&#160;        <span class="keywordflow">if</span> (out_distances)</div><div class="line"><a name="l01110"></a><span class="lineno"> 1110</span>&#160;          out_distances[0] = n_elected ? std::sqrt(distances(index)) : -1;</div><div class="line"><a name="l01111"></a><span class="lineno"> 1111</span>&#160;</div><div class="line"><a name="l01112"></a><span class="lineno"> 1112</span>&#160;        <span class="keywordflow">return</span>;</div><div class="line"><a name="l01113"></a><span class="lineno"> 1113</span>&#160;      }</div><div class="line"><a name="l01114"></a><span class="lineno"> 1114</span>&#160;</div><div class="line"><a name="l01115"></a><span class="lineno"> 1115</span>&#160;      <span class="keywordtype">int</span> n_to_sort = n_elected &gt; k ? k : n_elected;</div><div class="line"><a name="l01116"></a><span class="lineno"> 1116</span>&#160;      Eigen::VectorXi idx(n_elected);</div><div class="line"><a name="l01117"></a><span class="lineno"> 1117</span>&#160;      std::iota(idx.data(), idx.data() + n_elected, 0);</div><div class="line"><a name="l01118"></a><span class="lineno"> 1118</span>&#160;      std::partial_sort(idx.data(), idx.data() + n_to_sort, idx.data() + n_elected,</div><div class="line"><a name="l01119"></a><span class="lineno"> 1119</span>&#160;                       [&amp;distances](<span class="keywordtype">int</span> i1, <span class="keywordtype">int</span> i2) { <span class="keywordflow">return</span> distances(i1) &lt; distances(i2); });</div><div class="line"><a name="l01120"></a><span class="lineno"> 1120</span>&#160;</div><div class="line"><a name="l01121"></a><span class="lineno"> 1121</span>&#160;      <span class="keywordflow">for</span> (<span class="keywordtype">int</span> i = 0; i &lt; k; ++i)</div><div class="line"><a name="l01122"></a><span class="lineno"> 1122</span>&#160;        out[i] = i &lt; n_elected ? indices(idx(i)) : -1;</div><div class="line"><a name="l01123"></a><span class="lineno"> 1123</span>&#160;</div><div class="line"><a name="l01124"></a><span class="lineno"> 1124</span>&#160;      <span class="keywordflow">if</span> (out_distances) {</div><div class="line"><a name="l01125"></a><span class="lineno"> 1125</span>&#160;        <span class="keywordflow">for</span> (<span class="keywordtype">int</span> i = 0; i &lt; k; ++i)</div><div class="line"><a name="l01126"></a><span class="lineno"> 1126</span>&#160;          out_distances[i] = i &lt; n_elected ? std::sqrt(distances(idx(i))) : -1;</div><div class="line"><a name="l01127"></a><span class="lineno"> 1127</span>&#160;      }</div><div class="line"><a name="l01128"></a><span class="lineno"> 1128</span>&#160;    }</div><div class="line"><a name="l01129"></a><span class="lineno"> 1129</span>&#160;</div><div class="line"><a name="l01130"></a><span class="lineno"> 1130</span>&#160;    <span class="keywordtype">void</span> prune(<span class="keywordtype">double</span> target_recall) {</div><div class="line"><a name="l01131"></a><span class="lineno"> 1131</span>&#160;      <span class="keywordflow">if</span> (target_recall &lt; 0.0 - epsilon || target_recall &gt; 1.0 + epsilon) {</div><div class="line"><a name="l01132"></a><span class="lineno"> 1132</span>&#160;        <span class="keywordflow">throw</span> std::out_of_range(<span class="stringliteral">&quot;Target recall must be on the interval [0,1].&quot;</span>);</div><div class="line"><a name="l01133"></a><span class="lineno"> 1133</span>&#160;      }</div><div class="line"><a name="l01134"></a><span class="lineno"> 1134</span>&#160;</div><div class="line"><a name="l01135"></a><span class="lineno"> 1135</span>&#160;      par = <a class="code" href="class_mrpt.html#a82a6cd3ce04ff8307bfd00d99fc5400a">parameters</a>(target_recall);</div><div class="line"><a name="l01136"></a><span class="lineno"> 1136</span>&#160;      <span class="keywordflow">if</span> (!par.<a class="code" href="struct_mrpt___parameters.html#a4a0ab518bfa3786b59e58481bb7de814">n_trees</a>) {</div><div class="line"><a name="l01137"></a><span class="lineno"> 1137</span>&#160;        <span class="keywordflow">return</span>;</div><div class="line"><a name="l01138"></a><span class="lineno"> 1138</span>&#160;      }</div><div class="line"><a name="l01139"></a><span class="lineno"> 1139</span>&#160;</div><div class="line"><a name="l01140"></a><span class="lineno"> 1140</span>&#160;      <span class="keywordtype">int</span> depth_max = depth;</div><div class="line"><a name="l01141"></a><span class="lineno"> 1141</span>&#160;</div><div class="line"><a name="l01142"></a><span class="lineno"> 1142</span>&#160;      n_trees = par.<a class="code" href="struct_mrpt___parameters.html#a4a0ab518bfa3786b59e58481bb7de814">n_trees</a>;</div><div class="line"><a name="l01143"></a><span class="lineno"> 1143</span>&#160;      depth = par.<a class="code" href="struct_mrpt___parameters.html#a74ecd1a5f8320ec0cb30d2bfb7979fd0">depth</a>;</div><div class="line"><a name="l01144"></a><span class="lineno"> 1144</span>&#160;      votes = par.<a class="code" href="struct_mrpt___parameters.html#ab34a16df707e0abd7b8b0bf4959f36b6">votes</a>;</div><div class="line"><a name="l01145"></a><span class="lineno"> 1145</span>&#160;      n_pool = depth * n_trees;</div><div class="line"><a name="l01146"></a><span class="lineno"> 1146</span>&#160;      n_array = 1 &lt;&lt; (depth + 1);</div><div class="line"><a name="l01147"></a><span class="lineno"> 1147</span>&#160;</div><div class="line"><a name="l01148"></a><span class="lineno"> 1148</span>&#160;      tree_leaves.resize(n_trees);</div><div class="line"><a name="l01149"></a><span class="lineno"> 1149</span>&#160;      tree_leaves.shrink_to_fit();</div><div class="line"><a name="l01150"></a><span class="lineno"> 1150</span>&#160;      split_points.conservativeResize(n_array, n_trees);</div><div class="line"><a name="l01151"></a><span class="lineno"> 1151</span>&#160;      leaf_first_indices = leaf_first_indices_all[depth];</div><div class="line"><a name="l01152"></a><span class="lineno"> 1152</span>&#160;</div><div class="line"><a name="l01153"></a><span class="lineno"> 1153</span>&#160;      <span class="keywordflow">if</span> (density &lt; 1) {</div><div class="line"><a name="l01154"></a><span class="lineno"> 1154</span>&#160;        Eigen::SparseMatrix&lt;float, Eigen::RowMajor&gt; srm_new(n_pool, dim);</div><div class="line"><a name="l01155"></a><span class="lineno"> 1155</span>&#160;        <span class="keywordflow">for</span> (<span class="keywordtype">int</span> n_tree = 0; n_tree &lt; n_trees; ++n_tree)</div><div class="line"><a name="l01156"></a><span class="lineno"> 1156</span>&#160;          srm_new.middleRows(n_tree * depth, depth) = sparse_random_matrix.middleRows(n_tree * depth_max, depth);</div><div class="line"><a name="l01157"></a><span class="lineno"> 1157</span>&#160;        sparse_random_matrix = srm_new;</div><div class="line"><a name="l01158"></a><span class="lineno"> 1158</span>&#160;      } <span class="keywordflow">else</span> {</div><div class="line"><a name="l01159"></a><span class="lineno"> 1159</span>&#160;        Eigen::Matrix&lt;float, Eigen::Dynamic, Eigen::Dynamic, Eigen::RowMajor&gt; drm_new(n_pool, dim);</div><div class="line"><a name="l01160"></a><span class="lineno"> 1160</span>&#160;        <span class="keywordflow">for</span> (<span class="keywordtype">int</span> n_tree = 0; n_tree &lt; n_trees; ++n_tree)</div><div class="line"><a name="l01161"></a><span class="lineno"> 1161</span>&#160;          drm_new.middleRows(n_tree * depth, depth) = dense_random_matrix.middleRows(n_tree * depth_max, depth);</div><div class="line"><a name="l01162"></a><span class="lineno"> 1162</span>&#160;        dense_random_matrix = drm_new;</div><div class="line"><a name="l01163"></a><span class="lineno"> 1163</span>&#160;      }</div><div class="line"><a name="l01164"></a><span class="lineno"> 1164</span>&#160;</div><div class="line"><a name="l01165"></a><span class="lineno"> 1165</span>&#160;      index_type = autotuned;</div><div class="line"><a name="l01166"></a><span class="lineno"> 1166</span>&#160;    }</div><div class="line"><a name="l01167"></a><span class="lineno"> 1167</span>&#160;</div><div class="line"><a name="l01168"></a><span class="lineno"> 1168</span>&#160;    <span class="keywordtype">void</span> count_elected(<span class="keyword">const</span> Eigen::VectorXf &amp;q, <span class="keyword">const</span> Eigen::Map&lt;Eigen::VectorXi&gt; &amp;exact, <span class="keywordtype">int</span> votes_max,</div><div class="line"><a name="l01169"></a><span class="lineno"> 1169</span>&#160;                       std::vector&lt;Eigen::MatrixXd&gt; &amp;recalls, std::vector&lt;Eigen::MatrixXd&gt; &amp;cs_sizes)<span class="keyword"> const </span>{</div><div class="line"><a name="l01170"></a><span class="lineno"> 1170</span>&#160;      Eigen::VectorXf projected_query(n_pool);</div><div class="line"><a name="l01171"></a><span class="lineno"> 1171</span>&#160;      <span class="keywordflow">if</span> (density &lt; 1)</div><div class="line"><a name="l01172"></a><span class="lineno"> 1172</span>&#160;        projected_query.noalias() = sparse_random_matrix * q;</div><div class="line"><a name="l01173"></a><span class="lineno"> 1173</span>&#160;      <span class="keywordflow">else</span></div><div class="line"><a name="l01174"></a><span class="lineno"> 1174</span>&#160;        projected_query.noalias() = dense_random_matrix * q;</div><div class="line"><a name="l01175"></a><span class="lineno"> 1175</span>&#160;</div><div class="line"><a name="l01176"></a><span class="lineno"> 1176</span>&#160;      <span class="keywordtype">int</span> depth_min = depth - recalls.size() + 1;</div><div class="line"><a name="l01177"></a><span class="lineno"> 1177</span>&#160;      std::vector&lt;std::vector&lt;int&gt;&gt; start_indices(n_trees);</div><div class="line"><a name="l01178"></a><span class="lineno"> 1178</span>&#160;</div><div class="line"><a name="l01179"></a><span class="lineno"> 1179</span>&#160;<span class="preprocessor">      #pragma omp parallel for</span></div><div class="line"><a name="l01180"></a><span class="lineno"> 1180</span>&#160;      <span class="keywordflow">for</span> (<span class="keywordtype">int</span> n_tree = 0; n_tree &lt; n_trees; ++n_tree) {</div><div class="line"><a name="l01181"></a><span class="lineno"> 1181</span>&#160;        start_indices[n_tree] = std::vector&lt;int&gt;(depth - depth_min + 1);</div><div class="line"><a name="l01182"></a><span class="lineno"> 1182</span>&#160;        <span class="keywordtype">int</span> idx_tree = 0;</div><div class="line"><a name="l01183"></a><span class="lineno"> 1183</span>&#160;        <span class="keywordflow">for</span> (<span class="keywordtype">int</span> d = 0; d &lt; depth; ++d) {</div><div class="line"><a name="l01184"></a><span class="lineno"> 1184</span>&#160;          <span class="keyword">const</span> <span class="keywordtype">int</span> j = n_tree * depth + d;</div><div class="line"><a name="l01185"></a><span class="lineno"> 1185</span>&#160;          <span class="keyword">const</span> <span class="keywordtype">int</span> idx_left = 2 * idx_tree + 1;</div><div class="line"><a name="l01186"></a><span class="lineno"> 1186</span>&#160;          <span class="keyword">const</span> <span class="keywordtype">int</span> idx_right = idx_left + 1;</div><div class="line"><a name="l01187"></a><span class="lineno"> 1187</span>&#160;          <span class="keyword">const</span> <span class="keywordtype">float</span> split_point = split_points(idx_tree, n_tree);</div><div class="line"><a name="l01188"></a><span class="lineno"> 1188</span>&#160;          <span class="keywordflow">if</span> (projected_query(j) &lt;= split_point) {</div><div class="line"><a name="l01189"></a><span class="lineno"> 1189</span>&#160;            idx_tree = idx_left;</div><div class="line"><a name="l01190"></a><span class="lineno"> 1190</span>&#160;          } <span class="keywordflow">else</span> {</div><div class="line"><a name="l01191"></a><span class="lineno"> 1191</span>&#160;            idx_tree = idx_right;</div><div class="line"><a name="l01192"></a><span class="lineno"> 1192</span>&#160;          }</div><div class="line"><a name="l01193"></a><span class="lineno"> 1193</span>&#160;          <span class="keywordflow">if</span> (d &gt;= depth_min - 1)</div><div class="line"><a name="l01194"></a><span class="lineno"> 1194</span>&#160;            start_indices[n_tree][d - depth_min + 1] = idx_tree - (1 &lt;&lt; (d + 1)) + 1;</div><div class="line"><a name="l01195"></a><span class="lineno"> 1195</span>&#160;        }</div><div class="line"><a name="l01196"></a><span class="lineno"> 1196</span>&#160;      }</div><div class="line"><a name="l01197"></a><span class="lineno"> 1197</span>&#160;</div><div class="line"><a name="l01198"></a><span class="lineno"> 1198</span>&#160;      <span class="keyword">const</span> <span class="keywordtype">int</span> *exact_begin = exact.data();</div><div class="line"><a name="l01199"></a><span class="lineno"> 1199</span>&#160;      <span class="keyword">const</span> <span class="keywordtype">int</span> *exact_end = exact.data() + exact.size();</div><div class="line"><a name="l01200"></a><span class="lineno"> 1200</span>&#160;</div><div class="line"><a name="l01201"></a><span class="lineno"> 1201</span>&#160;      <span class="keywordflow">for</span> (<span class="keywordtype">int</span> depth_crnt = depth_min; depth_crnt &lt;= depth; ++depth_crnt) {</div><div class="line"><a name="l01202"></a><span class="lineno"> 1202</span>&#160;        Eigen::VectorXi votes = Eigen::VectorXi::Zero(n_samples);</div><div class="line"><a name="l01203"></a><span class="lineno"> 1203</span>&#160;        <span class="keyword">const</span> std::vector&lt;int&gt; &amp;leaf_first_indices = leaf_first_indices_all[depth_crnt];</div><div class="line"><a name="l01204"></a><span class="lineno"> 1204</span>&#160;</div><div class="line"><a name="l01205"></a><span class="lineno"> 1205</span>&#160;        Eigen::MatrixXd recall(votes_max, n_trees);</div><div class="line"><a name="l01206"></a><span class="lineno"> 1206</span>&#160;        Eigen::MatrixXd candidate_set_size(votes_max, n_trees);</div><div class="line"><a name="l01207"></a><span class="lineno"> 1207</span>&#160;        recall.col(0) = Eigen::VectorXd::Zero(votes_max);</div><div class="line"><a name="l01208"></a><span class="lineno"> 1208</span>&#160;        candidate_set_size.col(0) = Eigen::VectorXd::Zero(votes_max);</div><div class="line"><a name="l01209"></a><span class="lineno"> 1209</span>&#160;</div><div class="line"><a name="l01210"></a><span class="lineno"> 1210</span>&#160;        <span class="comment">// count votes</span></div><div class="line"><a name="l01211"></a><span class="lineno"> 1211</span>&#160;        <span class="keywordflow">for</span> (<span class="keywordtype">int</span> n_tree = 0; n_tree &lt; n_trees; ++n_tree) {</div><div class="line"><a name="l01212"></a><span class="lineno"> 1212</span>&#160;          std::vector&lt;int&gt; &amp;found_leaves = start_indices[n_tree];</div><div class="line"><a name="l01213"></a><span class="lineno"> 1213</span>&#160;</div><div class="line"><a name="l01214"></a><span class="lineno"> 1214</span>&#160;          <span class="keywordflow">if</span> (n_tree) {</div><div class="line"><a name="l01215"></a><span class="lineno"> 1215</span>&#160;            recall.col(n_tree) = recall.col(n_tree - 1);</div><div class="line"><a name="l01216"></a><span class="lineno"> 1216</span>&#160;            candidate_set_size.col(n_tree) = candidate_set_size.col(n_tree - 1);</div><div class="line"><a name="l01217"></a><span class="lineno"> 1217</span>&#160;          }</div><div class="line"><a name="l01218"></a><span class="lineno"> 1218</span>&#160;</div><div class="line"><a name="l01219"></a><span class="lineno"> 1219</span>&#160;          <span class="keywordtype">int</span> leaf_begin = leaf_first_indices[found_leaves[depth_crnt - depth_min]];</div><div class="line"><a name="l01220"></a><span class="lineno"> 1220</span>&#160;          <span class="keywordtype">int</span> leaf_end = leaf_first_indices[found_leaves[depth_crnt - depth_min] + 1];</div><div class="line"><a name="l01221"></a><span class="lineno"> 1221</span>&#160;</div><div class="line"><a name="l01222"></a><span class="lineno"> 1222</span>&#160;          <span class="keyword">const</span> std::vector&lt;int&gt; &amp;indices = tree_leaves[n_tree];</div><div class="line"><a name="l01223"></a><span class="lineno"> 1223</span>&#160;          <span class="keywordflow">for</span> (<span class="keywordtype">int</span> i = leaf_begin; i &lt; leaf_end; ++i) {</div><div class="line"><a name="l01224"></a><span class="lineno"> 1224</span>&#160;            <span class="keywordtype">int</span> idx = indices[i];</div><div class="line"><a name="l01225"></a><span class="lineno"> 1225</span>&#160;            <span class="keywordtype">int</span> v = ++votes(idx);</div><div class="line"><a name="l01226"></a><span class="lineno"> 1226</span>&#160;            <span class="keywordflow">if</span> (v &lt;= votes_max) {</div><div class="line"><a name="l01227"></a><span class="lineno"> 1227</span>&#160;              candidate_set_size(v - 1, n_tree)++;</div><div class="line"><a name="l01228"></a><span class="lineno"> 1228</span>&#160;              <span class="keywordflow">if</span> (std::find(exact_begin, exact_end, idx) != exact_end)</div><div class="line"><a name="l01229"></a><span class="lineno"> 1229</span>&#160;                recall(v - 1, n_tree)++;</div><div class="line"><a name="l01230"></a><span class="lineno"> 1230</span>&#160;            }</div><div class="line"><a name="l01231"></a><span class="lineno"> 1231</span>&#160;          }</div><div class="line"><a name="l01232"></a><span class="lineno"> 1232</span>&#160;        }</div><div class="line"><a name="l01233"></a><span class="lineno"> 1233</span>&#160;</div><div class="line"><a name="l01234"></a><span class="lineno"> 1234</span>&#160;        recalls[depth_crnt - depth_min] = recall;</div><div class="line"><a name="l01235"></a><span class="lineno"> 1235</span>&#160;        cs_sizes[depth_crnt - depth_min] = candidate_set_size;</div><div class="line"><a name="l01236"></a><span class="lineno"> 1236</span>&#160;      }</div><div class="line"><a name="l01237"></a><span class="lineno"> 1237</span>&#160;    }</div><div class="line"><a name="l01238"></a><span class="lineno"> 1238</span>&#160;</div><div class="line"><a name="l01248"></a><span class="lineno"> 1248</span>&#160;    <span class="keyword">static</span> <span class="keywordtype">void</span> build_sparse_random_matrix(Eigen::SparseMatrix&lt;float, Eigen::RowMajor&gt; &amp;sparse_random_matrix,</div><div class="line"><a name="l01249"></a><span class="lineno"> 1249</span>&#160;                                           <span class="keywordtype">int</span> n_row, <span class="keywordtype">int</span> n_col, <span class="keywordtype">float</span> density, <span class="keywordtype">int</span> seed = 0) {</div><div class="line"><a name="l01250"></a><span class="lineno"> 1250</span>&#160;      sparse_random_matrix = Eigen::SparseMatrix&lt;float, Eigen::RowMajor&gt;(n_row, n_col);</div><div class="line"><a name="l01251"></a><span class="lineno"> 1251</span>&#160;</div><div class="line"><a name="l01252"></a><span class="lineno"> 1252</span>&#160;      std::random_device rd;</div><div class="line"><a name="l01253"></a><span class="lineno"> 1253</span>&#160;      <span class="keywordtype">int</span> s = seed ? seed : rd();</div><div class="line"><a name="l01254"></a><span class="lineno"> 1254</span>&#160;      std::mt19937 gen(s);</div><div class="line"><a name="l01255"></a><span class="lineno"> 1255</span>&#160;      std::uniform_real_distribution&lt;float&gt; uni_dist(0, 1);</div><div class="line"><a name="l01256"></a><span class="lineno"> 1256</span>&#160;      std::normal_distribution&lt;float&gt; norm_dist(0, 1);</div><div class="line"><a name="l01257"></a><span class="lineno"> 1257</span>&#160;</div><div class="line"><a name="l01258"></a><span class="lineno"> 1258</span>&#160;      std::vector&lt;Eigen::Triplet&lt;float&gt;&gt; triplets;</div><div class="line"><a name="l01259"></a><span class="lineno"> 1259</span>&#160;      <span class="keywordflow">for</span> (<span class="keywordtype">int</span> j = 0; j &lt; n_row; ++j) {</div><div class="line"><a name="l01260"></a><span class="lineno"> 1260</span>&#160;        <span class="keywordflow">for</span> (<span class="keywordtype">int</span> i = 0; i &lt; n_col; ++i) {</div><div class="line"><a name="l01261"></a><span class="lineno"> 1261</span>&#160;          <span class="keywordflow">if</span> (uni_dist(gen) &gt; density) <span class="keywordflow">continue</span>;</div><div class="line"><a name="l01262"></a><span class="lineno"> 1262</span>&#160;          triplets.push_back(Eigen::Triplet&lt;float&gt;(j, i, norm_dist(gen)));</div><div class="line"><a name="l01263"></a><span class="lineno"> 1263</span>&#160;        }</div><div class="line"><a name="l01264"></a><span class="lineno"> 1264</span>&#160;      }</div><div class="line"><a name="l01265"></a><span class="lineno"> 1265</span>&#160;</div><div class="line"><a name="l01266"></a><span class="lineno"> 1266</span>&#160;      sparse_random_matrix.setFromTriplets(triplets.begin(), triplets.end());</div><div class="line"><a name="l01267"></a><span class="lineno"> 1267</span>&#160;      sparse_random_matrix.makeCompressed();</div><div class="line"><a name="l01268"></a><span class="lineno"> 1268</span>&#160;    }</div><div class="line"><a name="l01269"></a><span class="lineno"> 1269</span>&#160;</div><div class="line"><a name="l01270"></a><span class="lineno"> 1270</span>&#160;    <span class="comment">/*</span></div><div class="line"><a name="l01271"></a><span class="lineno"> 1271</span>&#160;<span class="comment">    * Builds a random dense matrix for use in random projection. The components of</span></div><div class="line"><a name="l01272"></a><span class="lineno"> 1272</span>&#160;<span class="comment">    * the matrix are drawn from the standard normal distribution.</span></div><div class="line"><a name="l01273"></a><span class="lineno"> 1273</span>&#160;<span class="comment">    */</span></div><div class="line"><a name="l01274"></a><span class="lineno"> 1274</span>&#160;    <span class="keyword">static</span> <span class="keywordtype">void</span> build_dense_random_matrix(Eigen::Matrix&lt;float, Eigen::Dynamic, Eigen::Dynamic, Eigen::RowMajor&gt; &amp;dense_random_matrix,</div><div class="line"><a name="l01275"></a><span class="lineno"> 1275</span>&#160;                                          <span class="keywordtype">int</span> n_row, <span class="keywordtype">int</span> n_col, <span class="keywordtype">int</span> seed = 0) {</div><div class="line"><a name="l01276"></a><span class="lineno"> 1276</span>&#160;      dense_random_matrix = Eigen::Matrix&lt;float, Eigen::Dynamic, Eigen::Dynamic, Eigen::RowMajor&gt;(n_row, n_col);</div><div class="line"><a name="l01277"></a><span class="lineno"> 1277</span>&#160;</div><div class="line"><a name="l01278"></a><span class="lineno"> 1278</span>&#160;      std::random_device rd;</div><div class="line"><a name="l01279"></a><span class="lineno"> 1279</span>&#160;      <span class="keywordtype">int</span> s = seed ? seed : rd();</div><div class="line"><a name="l01280"></a><span class="lineno"> 1280</span>&#160;      std::mt19937 gen(s);</div><div class="line"><a name="l01281"></a><span class="lineno"> 1281</span>&#160;      std::normal_distribution&lt;float&gt; normal_dist(0, 1);</div><div class="line"><a name="l01282"></a><span class="lineno"> 1282</span>&#160;</div><div class="line"><a name="l01283"></a><span class="lineno"> 1283</span>&#160;      std::generate(dense_random_matrix.data(), dense_random_matrix.data() + n_row * n_col,</div><div class="line"><a name="l01284"></a><span class="lineno"> 1284</span>&#160;                    [&amp;normal_dist, &amp;gen] { <span class="keywordflow">return</span> normal_dist(gen); });</div><div class="line"><a name="l01285"></a><span class="lineno"> 1285</span>&#160;    }</div><div class="line"><a name="l01286"></a><span class="lineno"> 1286</span>&#160;</div><div class="line"><a name="l01287"></a><span class="lineno"> 1287</span>&#160;    <span class="keywordtype">void</span> compute_exact(<span class="keyword">const</span> Eigen::Map&lt;const Eigen::MatrixXf&gt; &amp;Q, Eigen::MatrixXi &amp;out_exact,</div><div class="line"><a name="l01288"></a><span class="lineno"> 1288</span>&#160;                       <span class="keyword">const</span> std::vector&lt;int&gt; &amp;indices_test = {}) <span class="keyword">const</span> {</div><div class="line"><a name="l01289"></a><span class="lineno"> 1289</span>&#160;      <span class="keywordtype">int</span> n_test = Q.cols();</div><div class="line"><a name="l01290"></a><span class="lineno"> 1290</span>&#160;</div><div class="line"><a name="l01291"></a><span class="lineno"> 1291</span>&#160;      Eigen::VectorXi idx(n_samples);</div><div class="line"><a name="l01292"></a><span class="lineno"> 1292</span>&#160;      std::iota(idx.data(), idx.data() + n_samples, 0);</div><div class="line"><a name="l01293"></a><span class="lineno"> 1293</span>&#160;</div><div class="line"><a name="l01294"></a><span class="lineno"> 1294</span>&#160;      <span class="keywordflow">for</span> (<span class="keywordtype">int</span> i = 0; i &lt; n_test; ++i) {</div><div class="line"><a name="l01295"></a><span class="lineno"> 1295</span>&#160;        <span class="keywordflow">if</span>(!indices_test.empty()) {</div><div class="line"><a name="l01296"></a><span class="lineno"> 1296</span>&#160;          std::remove(idx.data(), idx.data() + n_samples, indices_test[i]);</div><div class="line"><a name="l01297"></a><span class="lineno"> 1297</span>&#160;        }</div><div class="line"><a name="l01298"></a><span class="lineno"> 1298</span>&#160;        <a class="code" href="class_mrpt.html#a19867c3d3f83e50c88e1edde63a81985">exact_knn</a>(Eigen::Map&lt;const Eigen::VectorXf&gt;(Q.data() + i * dim, dim), k, idx,</div><div class="line"><a name="l01299"></a><span class="lineno"> 1299</span>&#160;                  (indices_test.empty() ? n_samples : n_samples - 1), out_exact.data() + i * k);</div><div class="line"><a name="l01300"></a><span class="lineno"> 1300</span>&#160;        std::sort(out_exact.data() + i * k, out_exact.data() + i * k + k);</div><div class="line"><a name="l01301"></a><span class="lineno"> 1301</span>&#160;        <span class="keywordflow">if</span>(!indices_test.empty()) {</div><div class="line"><a name="l01302"></a><span class="lineno"> 1302</span>&#160;          idx[n_samples - 1] = indices_test[i];</div><div class="line"><a name="l01303"></a><span class="lineno"> 1303</span>&#160;        }</div><div class="line"><a name="l01304"></a><span class="lineno"> 1304</span>&#160;      }</div><div class="line"><a name="l01305"></a><span class="lineno"> 1305</span>&#160;    }</div><div class="line"><a name="l01306"></a><span class="lineno"> 1306</span>&#160;</div><div class="line"><a name="l01307"></a><span class="lineno"> 1307</span>&#160;    <span class="keyword">static</span> <span class="keywordtype">bool</span> is_faster(<span class="keyword">const</span> <a class="code" href="struct_mrpt___parameters.html">Mrpt_Parameters</a> &amp;par1, <span class="keyword">const</span> <a class="code" href="struct_mrpt___parameters.html">Mrpt_Parameters</a> &amp;par2) {</div><div class="line"><a name="l01308"></a><span class="lineno"> 1308</span>&#160;      <span class="keywordflow">return</span> par1.<a class="code" href="struct_mrpt___parameters.html#a35ec37c03c3ff33e3dc6a9944d581327">estimated_qtime</a> &lt; par2.<a class="code" href="struct_mrpt___parameters.html#a35ec37c03c3ff33e3dc6a9944d581327">estimated_qtime</a>;</div><div class="line"><a name="l01309"></a><span class="lineno"> 1309</span>&#160;    }</div><div class="line"><a name="l01310"></a><span class="lineno"> 1310</span>&#160;</div><div class="line"><a name="l01311"></a><span class="lineno"> 1311</span>&#160;    <span class="keywordtype">void</span> vote(<span class="keyword">const</span> Eigen::VectorXf &amp;projected_query, <span class="keywordtype">int</span> vote_threshold, Eigen::VectorXi &amp;elected,</div><div class="line"><a name="l01312"></a><span class="lineno"> 1312</span>&#160;      <span class="keywordtype">int</span> &amp;n_elected, <span class="keywordtype">int</span> n_trees, <span class="keywordtype">int</span> depth_crnt) {</div><div class="line"><a name="l01313"></a><span class="lineno"> 1313</span>&#160;      std::vector&lt;int&gt; found_leaves(n_trees);</div><div class="line"><a name="l01314"></a><span class="lineno"> 1314</span>&#160;      <span class="keyword">const</span> std::vector&lt;int&gt; &amp;leaf_first_indices = leaf_first_indices_all[depth_crnt];</div><div class="line"><a name="l01315"></a><span class="lineno"> 1315</span>&#160;</div><div class="line"><a name="l01316"></a><span class="lineno"> 1316</span>&#160;<span class="preprocessor">      #pragma omp parallel for</span></div><div class="line"><a name="l01317"></a><span class="lineno"> 1317</span>&#160;      <span class="keywordflow">for</span> (<span class="keywordtype">int</span> n_tree = 0; n_tree &lt; n_trees; ++n_tree) {</div><div class="line"><a name="l01318"></a><span class="lineno"> 1318</span>&#160;        <span class="keywordtype">int</span> idx_tree = 0;</div><div class="line"><a name="l01319"></a><span class="lineno"> 1319</span>&#160;        <span class="keywordflow">for</span> (<span class="keywordtype">int</span> d = 0; d &lt; depth_crnt; ++d) {</div><div class="line"><a name="l01320"></a><span class="lineno"> 1320</span>&#160;          <span class="keyword">const</span> <span class="keywordtype">int</span> j = n_tree * depth + d;</div><div class="line"><a name="l01321"></a><span class="lineno"> 1321</span>&#160;          <span class="keyword">const</span> <span class="keywordtype">int</span> idx_left = 2 * idx_tree + 1;</div><div class="line"><a name="l01322"></a><span class="lineno"> 1322</span>&#160;          <span class="keyword">const</span> <span class="keywordtype">int</span> idx_right = idx_left + 1;</div><div class="line"><a name="l01323"></a><span class="lineno"> 1323</span>&#160;          <span class="keyword">const</span> <span class="keywordtype">float</span> split_point = split_points(idx_tree, n_tree);</div><div class="line"><a name="l01324"></a><span class="lineno"> 1324</span>&#160;          <span class="keywordflow">if</span> (projected_query(j) &lt;= split_point) {</div><div class="line"><a name="l01325"></a><span class="lineno"> 1325</span>&#160;            idx_tree = idx_left;</div><div class="line"><a name="l01326"></a><span class="lineno"> 1326</span>&#160;          } <span class="keywordflow">else</span> {</div><div class="line"><a name="l01327"></a><span class="lineno"> 1327</span>&#160;            idx_tree = idx_right;</div><div class="line"><a name="l01328"></a><span class="lineno"> 1328</span>&#160;          }</div><div class="line"><a name="l01329"></a><span class="lineno"> 1329</span>&#160;        }</div><div class="line"><a name="l01330"></a><span class="lineno"> 1330</span>&#160;        found_leaves[n_tree] = idx_tree - (1 &lt;&lt; depth_crnt) + 1;</div><div class="line"><a name="l01331"></a><span class="lineno"> 1331</span>&#160;      }</div><div class="line"><a name="l01332"></a><span class="lineno"> 1332</span>&#160;</div><div class="line"><a name="l01333"></a><span class="lineno"> 1333</span>&#160;      <span class="keywordtype">int</span> max_leaf_size = n_samples / (1 &lt;&lt; depth_crnt) + 1;</div><div class="line"><a name="l01334"></a><span class="lineno"> 1334</span>&#160;      elected = Eigen::VectorXi(n_trees * max_leaf_size);</div><div class="line"><a name="l01335"></a><span class="lineno"> 1335</span>&#160;      Eigen::VectorXi votes = Eigen::VectorXi::Zero(n_samples);</div><div class="line"><a name="l01336"></a><span class="lineno"> 1336</span>&#160;</div><div class="line"><a name="l01337"></a><span class="lineno"> 1337</span>&#160;      <span class="comment">// count votes</span></div><div class="line"><a name="l01338"></a><span class="lineno"> 1338</span>&#160;      <span class="keywordflow">for</span> (<span class="keywordtype">int</span> n_tree = 0; n_tree &lt; n_trees; ++n_tree) {</div><div class="line"><a name="l01339"></a><span class="lineno"> 1339</span>&#160;        <span class="keywordtype">int</span> leaf_begin = leaf_first_indices[found_leaves[n_tree]];</div><div class="line"><a name="l01340"></a><span class="lineno"> 1340</span>&#160;        <span class="keywordtype">int</span> leaf_end = leaf_first_indices[found_leaves[n_tree] + 1];</div><div class="line"><a name="l01341"></a><span class="lineno"> 1341</span>&#160;        <span class="keyword">const</span> std::vector&lt;int&gt; &amp;indices = tree_leaves[n_tree];</div><div class="line"><a name="l01342"></a><span class="lineno"> 1342</span>&#160;        <span class="keywordflow">for</span> (<span class="keywordtype">int</span> i = leaf_begin; i &lt; leaf_end; ++i) {</div><div class="line"><a name="l01343"></a><span class="lineno"> 1343</span>&#160;          <span class="keywordtype">int</span> idx = indices[i];</div><div class="line"><a name="l01344"></a><span class="lineno"> 1344</span>&#160;          <span class="keywordflow">if</span> (++votes(idx) == vote_threshold)</div><div class="line"><a name="l01345"></a><span class="lineno"> 1345</span>&#160;            elected(n_elected++) = idx;</div><div class="line"><a name="l01346"></a><span class="lineno"> 1346</span>&#160;        }</div><div class="line"><a name="l01347"></a><span class="lineno"> 1347</span>&#160;      }</div><div class="line"><a name="l01348"></a><span class="lineno"> 1348</span>&#160;    }</div><div class="line"><a name="l01349"></a><span class="lineno"> 1349</span>&#160;</div><div class="line"><a name="l01350"></a><span class="lineno"> 1350</span>&#160;    std::pair&lt;double,double&gt; fit_projection_times(<span class="keyword">const</span> Eigen::Map&lt;const Eigen::MatrixXf&gt; &amp;Q,</div><div class="line"><a name="l01351"></a><span class="lineno"> 1351</span>&#160;          std::vector&lt;int&gt; &amp;exact_x) {</div><div class="line"><a name="l01352"></a><span class="lineno"> 1352</span>&#160;      std::vector&lt;double&gt; projection_times, projection_x;</div><div class="line"><a name="l01353"></a><span class="lineno"> 1353</span>&#160;      <span class="keywordtype">long</span> <span class="keywordtype">double</span> idx_sum = 0;</div><div class="line"><a name="l01354"></a><span class="lineno"> 1354</span>&#160;</div><div class="line"><a name="l01355"></a><span class="lineno"> 1355</span>&#160;      std::vector&lt;int&gt; tested_trees {1,2,3,4,5,7,10,15,20,25,30,40,50};</div><div class="line"><a name="l01356"></a><span class="lineno"> 1356</span>&#160;      generate_x(tested_trees, n_trees, 10, n_trees);</div><div class="line"><a name="l01357"></a><span class="lineno"> 1357</span>&#160;</div><div class="line"><a name="l01358"></a><span class="lineno"> 1358</span>&#160;      <span class="keywordflow">for</span> (<span class="keywordtype">int</span> d = depth_min; d &lt;= depth; ++d) {</div><div class="line"><a name="l01359"></a><span class="lineno"> 1359</span>&#160;        <span class="keywordflow">for</span> (<span class="keywordtype">int</span> i = 0; i &lt; (int) tested_trees.size(); ++i) {</div><div class="line"><a name="l01360"></a><span class="lineno"> 1360</span>&#160;          <span class="keywordtype">int</span> t = tested_trees[i];</div><div class="line"><a name="l01361"></a><span class="lineno"> 1361</span>&#160;          <span class="keywordtype">int</span> n_random_vectors = t * d;</div><div class="line"><a name="l01362"></a><span class="lineno"> 1362</span>&#160;          projection_x.push_back(n_random_vectors);</div><div class="line"><a name="l01363"></a><span class="lineno"> 1363</span>&#160;          Eigen::SparseMatrix&lt;float, Eigen::RowMajor&gt; sparse_mat;</div><div class="line"><a name="l01364"></a><span class="lineno"> 1364</span>&#160;          Eigen::Matrix&lt;float, Eigen::Dynamic, Eigen::Dynamic, Eigen::RowMajor&gt; dense_mat;</div><div class="line"><a name="l01365"></a><span class="lineno"> 1365</span>&#160;</div><div class="line"><a name="l01366"></a><span class="lineno"> 1366</span>&#160;          <span class="keywordflow">if</span> (density &lt; 1) {</div><div class="line"><a name="l01367"></a><span class="lineno"> 1367</span>&#160;            build_sparse_random_matrix(sparse_mat, n_random_vectors, dim, density);</div><div class="line"><a name="l01368"></a><span class="lineno"> 1368</span>&#160;          } <span class="keywordflow">else</span> {</div><div class="line"><a name="l01369"></a><span class="lineno"> 1369</span>&#160;            build_dense_random_matrix(dense_mat, n_random_vectors, dim);</div><div class="line"><a name="l01370"></a><span class="lineno"> 1370</span>&#160;          }</div><div class="line"><a name="l01371"></a><span class="lineno"> 1371</span>&#160;</div><div class="line"><a name="l01372"></a><span class="lineno"> 1372</span>&#160;          <span class="keywordtype">double</span> start_proj = omp_get_wtime();</div><div class="line"><a name="l01373"></a><span class="lineno"> 1373</span>&#160;          Eigen::VectorXf projected_query(n_random_vectors);</div><div class="line"><a name="l01374"></a><span class="lineno"> 1374</span>&#160;</div><div class="line"><a name="l01375"></a><span class="lineno"> 1375</span>&#160;          <span class="keywordflow">if</span> (density &lt; 1) {</div><div class="line"><a name="l01376"></a><span class="lineno"> 1376</span>&#160;            projected_query.noalias() = sparse_mat * Q.col(0);</div><div class="line"><a name="l01377"></a><span class="lineno"> 1377</span>&#160;          } <span class="keywordflow">else</span> {</div><div class="line"><a name="l01378"></a><span class="lineno"> 1378</span>&#160;            projected_query.noalias() = dense_mat * Q.col(0);</div><div class="line"><a name="l01379"></a><span class="lineno"> 1379</span>&#160;          }</div><div class="line"><a name="l01380"></a><span class="lineno"> 1380</span>&#160;</div><div class="line"><a name="l01381"></a><span class="lineno"> 1381</span>&#160;          <span class="keywordtype">double</span> end_proj = omp_get_wtime();</div><div class="line"><a name="l01382"></a><span class="lineno"> 1382</span>&#160;          projection_times.push_back(end_proj - start_proj);</div><div class="line"><a name="l01383"></a><span class="lineno"> 1383</span>&#160;          idx_sum += projected_query.norm();</div><div class="line"><a name="l01384"></a><span class="lineno"> 1384</span>&#160;</div><div class="line"><a name="l01385"></a><span class="lineno"> 1385</span>&#160;          <span class="keywordtype">int</span> votes_index = votes_max &lt; t ? votes_max : t;</div><div class="line"><a name="l01386"></a><span class="lineno"> 1386</span>&#160;          <span class="keywordflow">for</span> (<span class="keywordtype">int</span> v = 1; v &lt;= votes_index; ++v) {</div><div class="line"><a name="l01387"></a><span class="lineno"> 1387</span>&#160;            <span class="keywordtype">int</span> cs_size = get_candidate_set_size(t, d, v);</div><div class="line"><a name="l01388"></a><span class="lineno"> 1388</span>&#160;            <span class="keywordflow">if</span> (cs_size &gt; 0) exact_x.push_back(cs_size);</div><div class="line"><a name="l01389"></a><span class="lineno"> 1389</span>&#160;          }</div><div class="line"><a name="l01390"></a><span class="lineno"> 1390</span>&#160;        }</div><div class="line"><a name="l01391"></a><span class="lineno"> 1391</span>&#160;      }</div><div class="line"><a name="l01392"></a><span class="lineno"> 1392</span>&#160;</div><div class="line"><a name="l01393"></a><span class="lineno"> 1393</span>&#160;      <span class="comment">// use results to ensure that the compiler does not optimize away the timed code.</span></div><div class="line"><a name="l01394"></a><span class="lineno"> 1394</span>&#160;      projection_x[0] += idx_sum &gt; 1.0 ? 0.0000 : 0.0001;</div><div class="line"><a name="l01395"></a><span class="lineno"> 1395</span>&#160;      <span class="keywordflow">return</span> fit_theil_sen(projection_x, projection_times);</div><div class="line"><a name="l01396"></a><span class="lineno"> 1396</span>&#160;    }</div><div class="line"><a name="l01397"></a><span class="lineno"> 1397</span>&#160;</div><div class="line"><a name="l01398"></a><span class="lineno"> 1398</span>&#160;    std::vector&lt;std::map&lt;int,std::pair&lt;double,double&gt;&gt;&gt; fit_voting_times(<span class="keyword">const</span> Eigen::Map&lt;const Eigen::MatrixXf&gt; &amp;Q) {</div><div class="line"><a name="l01399"></a><span class="lineno"> 1399</span>&#160;      <span class="keywordtype">int</span> n_test = Q.cols();</div><div class="line"><a name="l01400"></a><span class="lineno"> 1400</span>&#160;</div><div class="line"><a name="l01401"></a><span class="lineno"> 1401</span>&#160;      std::random_device rd;</div><div class="line"><a name="l01402"></a><span class="lineno"> 1402</span>&#160;      std::mt19937 rng(rd());</div><div class="line"><a name="l01403"></a><span class="lineno"> 1403</span>&#160;      std::uniform_int_distribution&lt;int&gt; uni(0, n_test - 1);</div><div class="line"><a name="l01404"></a><span class="lineno"> 1404</span>&#160;</div><div class="line"><a name="l01405"></a><span class="lineno"> 1405</span>&#160;      std::vector&lt;int&gt; tested_trees {1,2,3,4,5,7,10,15,20,25,30,40,50};</div><div class="line"><a name="l01406"></a><span class="lineno"> 1406</span>&#160;      generate_x(tested_trees, n_trees, 10, n_trees);</div><div class="line"><a name="l01407"></a><span class="lineno"> 1407</span>&#160;      std::vector&lt;int&gt; vote_thresholds_x {1,2,3,4,5,6,7,8,9,10,11,12,13,14,15};</div><div class="line"><a name="l01408"></a><span class="lineno"> 1408</span>&#160;      generate_x(vote_thresholds_x, votes_max, 10, votes_max);</div><div class="line"><a name="l01409"></a><span class="lineno"> 1409</span>&#160;</div><div class="line"><a name="l01410"></a><span class="lineno"> 1410</span>&#160;      beta_voting = std::vector&lt;std::map&lt;int,std::pair&lt;double,double&gt;&gt;&gt;();</div><div class="line"><a name="l01411"></a><span class="lineno"> 1411</span>&#160;</div><div class="line"><a name="l01412"></a><span class="lineno"> 1412</span>&#160;      <span class="keywordflow">for</span> (<span class="keywordtype">int</span> d = depth_min; d &lt;= depth; ++d) {</div><div class="line"><a name="l01413"></a><span class="lineno"> 1413</span>&#160;        std::map&lt;int,std::pair&lt;double,double&gt;&gt; beta;</div><div class="line"><a name="l01414"></a><span class="lineno"> 1414</span>&#160;        <span class="keywordflow">for</span> (<span class="keyword">const</span> <span class="keyword">auto</span> &amp;v : vote_thresholds_x) {</div><div class="line"><a name="l01415"></a><span class="lineno"> 1415</span>&#160;          <span class="keywordtype">long</span> <span class="keywordtype">double</span> idx_sum = 0;</div><div class="line"><a name="l01416"></a><span class="lineno"> 1416</span>&#160;          std::vector&lt;double&gt; voting_times, voting_x;</div><div class="line"><a name="l01417"></a><span class="lineno"> 1417</span>&#160;</div><div class="line"><a name="l01418"></a><span class="lineno"> 1418</span>&#160;          <span class="keywordflow">for</span> (<span class="keywordtype">int</span> i = 0; i &lt; (int) tested_trees.size(); ++i) {</div><div class="line"><a name="l01419"></a><span class="lineno"> 1419</span>&#160;            <span class="keywordtype">int</span> t = tested_trees[i];</div><div class="line"><a name="l01420"></a><span class="lineno"> 1420</span>&#160;            <span class="keywordtype">int</span> n_el = 0;</div><div class="line"><a name="l01421"></a><span class="lineno"> 1421</span>&#160;            Eigen::VectorXi elected;</div><div class="line"><a name="l01422"></a><span class="lineno"> 1422</span>&#160;            <span class="keyword">auto</span> ri = uni(rng);</div><div class="line"><a name="l01423"></a><span class="lineno"> 1423</span>&#160;</div><div class="line"><a name="l01424"></a><span class="lineno"> 1424</span>&#160;            Eigen::VectorXf projected_query(n_trees * depth);</div><div class="line"><a name="l01425"></a><span class="lineno"> 1425</span>&#160;            <span class="keywordflow">if</span> (density &lt; 1) {</div><div class="line"><a name="l01426"></a><span class="lineno"> 1426</span>&#160;              projected_query.noalias() = sparse_random_matrix * Q.col(ri);</div><div class="line"><a name="l01427"></a><span class="lineno"> 1427</span>&#160;            } <span class="keywordflow">else</span> {</div><div class="line"><a name="l01428"></a><span class="lineno"> 1428</span>&#160;              projected_query.noalias() = dense_random_matrix * Q.col(ri);</div><div class="line"><a name="l01429"></a><span class="lineno"> 1429</span>&#160;            }</div><div class="line"><a name="l01430"></a><span class="lineno"> 1430</span>&#160;</div><div class="line"><a name="l01431"></a><span class="lineno"> 1431</span>&#160;            <span class="keywordtype">double</span> start_voting = omp_get_wtime();</div><div class="line"><a name="l01432"></a><span class="lineno"> 1432</span>&#160;            vote(projected_query, v, elected, n_el, t, d);</div><div class="line"><a name="l01433"></a><span class="lineno"> 1433</span>&#160;            <span class="keywordtype">double</span> end_voting = omp_get_wtime();</div><div class="line"><a name="l01434"></a><span class="lineno"> 1434</span>&#160;</div><div class="line"><a name="l01435"></a><span class="lineno"> 1435</span>&#160;            voting_times.push_back(end_voting - start_voting);</div><div class="line"><a name="l01436"></a><span class="lineno"> 1436</span>&#160;            voting_x.push_back(t);</div><div class="line"><a name="l01437"></a><span class="lineno"> 1437</span>&#160;            <span class="keywordflow">for</span> (<span class="keywordtype">int</span> i = 0; i &lt; n_el; ++i)</div><div class="line"><a name="l01438"></a><span class="lineno"> 1438</span>&#160;              idx_sum += elected(i);</div><div class="line"><a name="l01439"></a><span class="lineno"> 1439</span>&#160;          }</div><div class="line"><a name="l01440"></a><span class="lineno"> 1440</span>&#160;          voting_x[0] += idx_sum &gt; 1.0 ? 0.0 : 0.00001;</div><div class="line"><a name="l01441"></a><span class="lineno"> 1441</span>&#160;          beta[v] = fit_theil_sen(voting_x, voting_times);</div><div class="line"><a name="l01442"></a><span class="lineno"> 1442</span>&#160;        }</div><div class="line"><a name="l01443"></a><span class="lineno"> 1443</span>&#160;        beta_voting.push_back(beta);</div><div class="line"><a name="l01444"></a><span class="lineno"> 1444</span>&#160;      }</div><div class="line"><a name="l01445"></a><span class="lineno"> 1445</span>&#160;</div><div class="line"><a name="l01446"></a><span class="lineno"> 1446</span>&#160;      <span class="keywordflow">return</span> beta_voting;</div><div class="line"><a name="l01447"></a><span class="lineno"> 1447</span>&#160;    }</div><div class="line"><a name="l01448"></a><span class="lineno"> 1448</span>&#160;</div><div class="line"><a name="l01449"></a><span class="lineno"> 1449</span>&#160;    <span class="keyword">static</span> <span class="keywordtype">void</span> generate_x(std::vector&lt;int&gt; &amp;x, <span class="keywordtype">int</span> max_generated, <span class="keywordtype">int</span> n_tested, <span class="keywordtype">int</span> max_val) {</div><div class="line"><a name="l01450"></a><span class="lineno"> 1450</span>&#160;      n_tested = max_generated &gt; n_tested ? n_tested : max_val;</div><div class="line"><a name="l01451"></a><span class="lineno"> 1451</span>&#160;      <span class="keywordtype">int</span> increment = max_generated / n_tested;</div><div class="line"><a name="l01452"></a><span class="lineno"> 1452</span>&#160;      <span class="keywordflow">for</span> (<span class="keywordtype">int</span> i = 1; i &lt;= n_tested; ++i) {</div><div class="line"><a name="l01453"></a><span class="lineno"> 1453</span>&#160;        <span class="keywordflow">if</span> (std::find(x.begin(), x.end(), i * increment) == x.end() &amp;&amp; i * increment &lt;= max_generated) {</div><div class="line"><a name="l01454"></a><span class="lineno"> 1454</span>&#160;          x.push_back(i * increment);</div><div class="line"><a name="l01455"></a><span class="lineno"> 1455</span>&#160;        }</div><div class="line"><a name="l01456"></a><span class="lineno"> 1456</span>&#160;      }</div><div class="line"><a name="l01457"></a><span class="lineno"> 1457</span>&#160;</div><div class="line"><a name="l01458"></a><span class="lineno"> 1458</span>&#160;      <span class="keyword">auto</span> end = std::remove_if(x.begin(), x.end(), [max_val](<span class="keywordtype">int</span> t) { <span class="keywordflow">return</span> t &gt; max_val; });</div><div class="line"><a name="l01459"></a><span class="lineno"> 1459</span>&#160;      x.erase(end, x.end());</div><div class="line"><a name="l01460"></a><span class="lineno"> 1460</span>&#160;    }</div><div class="line"><a name="l01461"></a><span class="lineno"> 1461</span>&#160;</div><div class="line"><a name="l01462"></a><span class="lineno"> 1462</span>&#160;    std::pair&lt;double,double&gt; fit_exact_times(<span class="keyword">const</span> Eigen::Map&lt;const Eigen::MatrixXf&gt; &amp;Q) {</div><div class="line"><a name="l01463"></a><span class="lineno"> 1463</span>&#160;      std::vector&lt;int&gt; s_tested {1,2,5,10,20,35,50,75,100,150,200,300,400,500};</div><div class="line"><a name="l01464"></a><span class="lineno"> 1464</span>&#160;      generate_x(s_tested, n_samples / 20, 20, n_samples);</div><div class="line"><a name="l01465"></a><span class="lineno"> 1465</span>&#160;</div><div class="line"><a name="l01466"></a><span class="lineno"> 1466</span>&#160;      <span class="keywordtype">int</span> n_test = Q.cols();</div><div class="line"><a name="l01467"></a><span class="lineno"> 1467</span>&#160;      std::vector&lt;double&gt; exact_times;</div><div class="line"><a name="l01468"></a><span class="lineno"> 1468</span>&#160;      <span class="keywordtype">long</span> <span class="keywordtype">double</span> idx_sum = 0;</div><div class="line"><a name="l01469"></a><span class="lineno"> 1469</span>&#160;</div><div class="line"><a name="l01470"></a><span class="lineno"> 1470</span>&#160;      std::random_device rd;</div><div class="line"><a name="l01471"></a><span class="lineno"> 1471</span>&#160;      std::mt19937 rng(rd());</div><div class="line"><a name="l01472"></a><span class="lineno"> 1472</span>&#160;      std::uniform_int_distribution&lt;int&gt; uni(0, n_test - 1);</div><div class="line"><a name="l01473"></a><span class="lineno"> 1473</span>&#160;      std::uniform_int_distribution&lt;int&gt; uni2(0, n_samples - 1);</div><div class="line"><a name="l01474"></a><span class="lineno"> 1474</span>&#160;</div><div class="line"><a name="l01475"></a><span class="lineno"> 1475</span>&#160;      std::vector&lt;double&gt; ex;</div><div class="line"><a name="l01476"></a><span class="lineno"> 1476</span>&#160;      <span class="keywordtype">int</span> n_sim = 20;</div><div class="line"><a name="l01477"></a><span class="lineno"> 1477</span>&#160;      <span class="keywordflow">for</span> (<span class="keywordtype">int</span> i = 0; i &lt; (int) s_tested.size(); ++i) {</div><div class="line"><a name="l01478"></a><span class="lineno"> 1478</span>&#160;        <span class="keywordtype">double</span> mean_exact_time = 0;</div><div class="line"><a name="l01479"></a><span class="lineno"> 1479</span>&#160;        <span class="keywordtype">int</span> s_size = s_tested[i];</div><div class="line"><a name="l01480"></a><span class="lineno"> 1480</span>&#160;        ex.push_back(s_size);</div><div class="line"><a name="l01481"></a><span class="lineno"> 1481</span>&#160;</div><div class="line"><a name="l01482"></a><span class="lineno"> 1482</span>&#160;        <span class="keywordflow">for</span> (<span class="keywordtype">int</span> m = 0; m &lt; n_sim; ++m) {</div><div class="line"><a name="l01483"></a><span class="lineno"> 1483</span>&#160;          <span class="keyword">auto</span> ri = uni(rng);</div><div class="line"><a name="l01484"></a><span class="lineno"> 1484</span>&#160;          Eigen::VectorXi elected(s_size);</div><div class="line"><a name="l01485"></a><span class="lineno"> 1485</span>&#160;          <span class="keywordflow">for</span> (<span class="keywordtype">int</span> j = 0; j &lt; elected.size(); ++j)</div><div class="line"><a name="l01486"></a><span class="lineno"> 1486</span>&#160;            elected(j) = uni2(rng);</div><div class="line"><a name="l01487"></a><span class="lineno"> 1487</span>&#160;</div><div class="line"><a name="l01488"></a><span class="lineno"> 1488</span>&#160;          <span class="keywordtype">double</span> start_exact = omp_get_wtime();</div><div class="line"><a name="l01489"></a><span class="lineno"> 1489</span>&#160;          std::vector&lt;int&gt; res(k);</div><div class="line"><a name="l01490"></a><span class="lineno"> 1490</span>&#160;          <a class="code" href="class_mrpt.html#a19867c3d3f83e50c88e1edde63a81985">exact_knn</a>(Eigen::Map&lt;const Eigen::VectorXf&gt;(Q.data() + ri * dim, dim), k, elected, s_size, &amp;res[0]);</div><div class="line"><a name="l01491"></a><span class="lineno"> 1491</span>&#160;          <span class="keywordtype">double</span> end_exact = omp_get_wtime();</div><div class="line"><a name="l01492"></a><span class="lineno"> 1492</span>&#160;          mean_exact_time += (end_exact - start_exact);</div><div class="line"><a name="l01493"></a><span class="lineno"> 1493</span>&#160;</div><div class="line"><a name="l01494"></a><span class="lineno"> 1494</span>&#160;          <span class="keywordflow">for</span> (<span class="keywordtype">int</span> l = 0; l &lt; k; ++l)</div><div class="line"><a name="l01495"></a><span class="lineno"> 1495</span>&#160;            idx_sum += res[l];</div><div class="line"><a name="l01496"></a><span class="lineno"> 1496</span>&#160;        }</div><div class="line"><a name="l01497"></a><span class="lineno"> 1497</span>&#160;        mean_exact_time /= n_sim;</div><div class="line"><a name="l01498"></a><span class="lineno"> 1498</span>&#160;        exact_times.push_back(mean_exact_time);</div><div class="line"><a name="l01499"></a><span class="lineno"> 1499</span>&#160;      }</div><div class="line"><a name="l01500"></a><span class="lineno"> 1500</span>&#160;</div><div class="line"><a name="l01501"></a><span class="lineno"> 1501</span>&#160;      ex[0] += idx_sum &gt; 1.0 ? 0.0 : 0.00001;</div><div class="line"><a name="l01502"></a><span class="lineno"> 1502</span>&#160;      <span class="keywordflow">return</span> fit_theil_sen(ex, exact_times);</div><div class="line"><a name="l01503"></a><span class="lineno"> 1503</span>&#160;    }</div><div class="line"><a name="l01504"></a><span class="lineno"> 1504</span>&#160;</div><div class="line"><a name="l01505"></a><span class="lineno"> 1505</span>&#160;    std::set&lt;Mrpt_Parameters,decltype(is_faster)*&gt; list_parameters(<span class="keyword">const</span> std::vector&lt;Eigen::MatrixXd&gt; &amp;recalls) {</div><div class="line"><a name="l01506"></a><span class="lineno"> 1506</span>&#160;      std::set&lt;Mrpt_Parameters,decltype(is_faster)*&gt; pars(is_faster);</div><div class="line"><a name="l01507"></a><span class="lineno"> 1507</span>&#160;      std::vector&lt;Eigen::MatrixXd&gt; query_times(depth - depth_min + 1);</div><div class="line"><a name="l01508"></a><span class="lineno"> 1508</span>&#160;      <span class="keywordflow">for</span> (<span class="keywordtype">int</span> d = depth_min; d &lt;= depth; ++d) {</div><div class="line"><a name="l01509"></a><span class="lineno"> 1509</span>&#160;        Eigen::MatrixXd query_time = Eigen::MatrixXd::Zero(votes_max, n_trees);</div><div class="line"><a name="l01510"></a><span class="lineno"> 1510</span>&#160;</div><div class="line"><a name="l01511"></a><span class="lineno"> 1511</span>&#160;        <span class="keywordflow">for</span> (<span class="keywordtype">int</span> t = 1; t &lt;= n_trees; ++t) {</div><div class="line"><a name="l01512"></a><span class="lineno"> 1512</span>&#160;          <span class="keywordtype">int</span> votes_index = votes_max &lt; t ? votes_max : t;</div><div class="line"><a name="l01513"></a><span class="lineno"> 1513</span>&#160;          <span class="keywordflow">for</span> (<span class="keywordtype">int</span> v = 1; v &lt;= votes_index; ++v) {</div><div class="line"><a name="l01514"></a><span class="lineno"> 1514</span>&#160;            <span class="keywordtype">double</span> qt = get_query_time(t, d, v);</div><div class="line"><a name="l01515"></a><span class="lineno"> 1515</span>&#160;            query_time(v - 1, t - 1) = qt;</div><div class="line"><a name="l01516"></a><span class="lineno"> 1516</span>&#160;            <a class="code" href="struct_mrpt___parameters.html">Mrpt_Parameters</a> p;</div><div class="line"><a name="l01517"></a><span class="lineno"> 1517</span>&#160;            p.<a class="code" href="struct_mrpt___parameters.html#a4a0ab518bfa3786b59e58481bb7de814">n_trees</a> = t;</div><div class="line"><a name="l01518"></a><span class="lineno"> 1518</span>&#160;            p.<a class="code" href="struct_mrpt___parameters.html#a74ecd1a5f8320ec0cb30d2bfb7979fd0">depth</a> = d;</div><div class="line"><a name="l01519"></a><span class="lineno"> 1519</span>&#160;            p.<a class="code" href="struct_mrpt___parameters.html#ab34a16df707e0abd7b8b0bf4959f36b6">votes</a> = v;</div><div class="line"><a name="l01520"></a><span class="lineno"> 1520</span>&#160;            p.<a class="code" href="struct_mrpt___parameters.html#ad8b374d6671223a498e67cb98d8ea191">k</a> = k;</div><div class="line"><a name="l01521"></a><span class="lineno"> 1521</span>&#160;            p.<a class="code" href="struct_mrpt___parameters.html#a35ec37c03c3ff33e3dc6a9944d581327">estimated_qtime</a> = qt;</div><div class="line"><a name="l01522"></a><span class="lineno"> 1522</span>&#160;            p.<a class="code" href="struct_mrpt___parameters.html#ab2fd9db6c27b3024a4e2990c2b4b52ca">estimated_recall</a> = recalls[d - depth_min](v - 1, t - 1);</div><div class="line"><a name="l01523"></a><span class="lineno"> 1523</span>&#160;            pars.insert(p);</div><div class="line"><a name="l01524"></a><span class="lineno"> 1524</span>&#160;          }</div><div class="line"><a name="l01525"></a><span class="lineno"> 1525</span>&#160;        }</div><div class="line"><a name="l01526"></a><span class="lineno"> 1526</span>&#160;</div><div class="line"><a name="l01527"></a><span class="lineno"> 1527</span>&#160;        query_times[d - depth_min] = query_time;</div><div class="line"><a name="l01528"></a><span class="lineno"> 1528</span>&#160;      }</div><div class="line"><a name="l01529"></a><span class="lineno"> 1529</span>&#160;</div><div class="line"><a name="l01530"></a><span class="lineno"> 1530</span>&#160;      <span class="keywordflow">return</span> pars;</div><div class="line"><a name="l01531"></a><span class="lineno"> 1531</span>&#160;    }</div><div class="line"><a name="l01532"></a><span class="lineno"> 1532</span>&#160;</div><div class="line"><a name="l01533"></a><span class="lineno"> 1533</span>&#160;    std::set&lt;Mrpt_Parameters,decltype(is_faster)*&gt; pareto_frontier(<span class="keyword">const</span> std::set&lt;<a class="code" href="struct_mrpt___parameters.html">Mrpt_Parameters</a>,decltype(is_faster)*&gt; &amp;pars) {</div><div class="line"><a name="l01534"></a><span class="lineno"> 1534</span>&#160;      opt_pars = std::set&lt;Mrpt_Parameters,decltype(is_faster)*&gt;(is_faster);</div><div class="line"><a name="l01535"></a><span class="lineno"> 1535</span>&#160;      <span class="keywordtype">double</span> best_recall = -1.0;</div><div class="line"><a name="l01536"></a><span class="lineno"> 1536</span>&#160;      <span class="keywordflow">for</span> (<span class="keyword">const</span> <span class="keyword">auto</span> &amp;p : pars) { <span class="comment">// compute pareto frontier for query times and recalls</span></div><div class="line"><a name="l01537"></a><span class="lineno"> 1537</span>&#160;        <span class="keywordflow">if</span> (p.estimated_recall &gt; best_recall) {</div><div class="line"><a name="l01538"></a><span class="lineno"> 1538</span>&#160;          opt_pars.insert(p);</div><div class="line"><a name="l01539"></a><span class="lineno"> 1539</span>&#160;          best_recall = p.estimated_recall;</div><div class="line"><a name="l01540"></a><span class="lineno"> 1540</span>&#160;        }</div><div class="line"><a name="l01541"></a><span class="lineno"> 1541</span>&#160;      }</div><div class="line"><a name="l01542"></a><span class="lineno"> 1542</span>&#160;</div><div class="line"><a name="l01543"></a><span class="lineno"> 1543</span>&#160;      <span class="keywordflow">return</span> opt_pars;</div><div class="line"><a name="l01544"></a><span class="lineno"> 1544</span>&#160;    }</div><div class="line"><a name="l01545"></a><span class="lineno"> 1545</span>&#160;</div><div class="line"><a name="l01546"></a><span class="lineno"> 1546</span>&#160;    <span class="keywordtype">void</span> fit_times(<span class="keyword">const</span> Eigen::Map&lt;const Eigen::MatrixXf&gt; &amp;Q) {</div><div class="line"><a name="l01547"></a><span class="lineno"> 1547</span>&#160;      std::vector&lt;int&gt; exact_x;</div><div class="line"><a name="l01548"></a><span class="lineno"> 1548</span>&#160;      beta_projection = fit_projection_times(Q, exact_x);</div><div class="line"><a name="l01549"></a><span class="lineno"> 1549</span>&#160;      beta_voting = fit_voting_times(Q);</div><div class="line"><a name="l01550"></a><span class="lineno"> 1550</span>&#160;      beta_exact = fit_exact_times(Q);</div><div class="line"><a name="l01551"></a><span class="lineno"> 1551</span>&#160;    }</div><div class="line"><a name="l01552"></a><span class="lineno"> 1552</span>&#160;</div><div class="line"><a name="l01553"></a><span class="lineno"> 1553</span>&#160;    <span class="keyword">static</span> std::pair&lt;double,double&gt; fit_theil_sen(<span class="keyword">const</span> std::vector&lt;double&gt; &amp;x,</div><div class="line"><a name="l01554"></a><span class="lineno"> 1554</span>&#160;                                                  <span class="keyword">const</span> std::vector&lt;double&gt; &amp;y) {</div><div class="line"><a name="l01555"></a><span class="lineno"> 1555</span>&#160;      <span class="keywordtype">int</span> n = x.size();</div><div class="line"><a name="l01556"></a><span class="lineno"> 1556</span>&#160;      std::vector&lt;double&gt; slopes;</div><div class="line"><a name="l01557"></a><span class="lineno"> 1557</span>&#160;      <span class="keywordflow">for</span> (<span class="keywordtype">int</span> i = 0; i &lt; n; ++i) {</div><div class="line"><a name="l01558"></a><span class="lineno"> 1558</span>&#160;        <span class="keywordflow">for</span> (<span class="keywordtype">int</span> j = 0; j &lt; n; ++j) {</div><div class="line"><a name="l01559"></a><span class="lineno"> 1559</span>&#160;          <span class="keywordflow">if</span> (i != j)</div><div class="line"><a name="l01560"></a><span class="lineno"> 1560</span>&#160;            slopes.push_back((y[j] - y[i]) / (x[j] - x[i]));</div><div class="line"><a name="l01561"></a><span class="lineno"> 1561</span>&#160;        }</div><div class="line"><a name="l01562"></a><span class="lineno"> 1562</span>&#160;      }</div><div class="line"><a name="l01563"></a><span class="lineno"> 1563</span>&#160;</div><div class="line"><a name="l01564"></a><span class="lineno"> 1564</span>&#160;      <span class="keywordtype">int</span> n_slopes = slopes.size();</div><div class="line"><a name="l01565"></a><span class="lineno"> 1565</span>&#160;      std::nth_element(slopes.begin(), slopes.begin() + n_slopes / 2, slopes.end());</div><div class="line"><a name="l01566"></a><span class="lineno"> 1566</span>&#160;      <span class="keywordtype">double</span> slope = *(slopes.begin() + n_slopes / 2);</div><div class="line"><a name="l01567"></a><span class="lineno"> 1567</span>&#160;</div><div class="line"><a name="l01568"></a><span class="lineno"> 1568</span>&#160;      std::vector&lt;double&gt; residuals(n);</div><div class="line"><a name="l01569"></a><span class="lineno"> 1569</span>&#160;      <span class="keywordflow">for</span> (<span class="keywordtype">int</span> i = 0; i &lt; n; ++i)</div><div class="line"><a name="l01570"></a><span class="lineno"> 1570</span>&#160;        residuals[i] = y[i] - slope * x[i];</div><div class="line"><a name="l01571"></a><span class="lineno"> 1571</span>&#160;</div><div class="line"><a name="l01572"></a><span class="lineno"> 1572</span>&#160;      std::nth_element(residuals.begin(), residuals.begin() + n / 2, residuals.end());</div><div class="line"><a name="l01573"></a><span class="lineno"> 1573</span>&#160;      <span class="keywordtype">double</span> intercept = *(residuals.begin() + n / 2);</div><div class="line"><a name="l01574"></a><span class="lineno"> 1574</span>&#160;</div><div class="line"><a name="l01575"></a><span class="lineno"> 1575</span>&#160;      <span class="keywordflow">return</span> std::make_pair(intercept, slope);</div><div class="line"><a name="l01576"></a><span class="lineno"> 1576</span>&#160;    }</div><div class="line"><a name="l01577"></a><span class="lineno"> 1577</span>&#160;</div><div class="line"><a name="l01578"></a><span class="lineno"> 1578</span>&#160;    <span class="keywordtype">void</span> write_parameters(<span class="keyword">const</span> <a class="code" href="struct_mrpt___parameters.html">Mrpt_Parameters</a> *p, FILE *fd)<span class="keyword"> const </span>{</div><div class="line"><a name="l01579"></a><span class="lineno"> 1579</span>&#160;      <span class="keywordflow">if</span> (!fd) {</div><div class="line"><a name="l01580"></a><span class="lineno"> 1580</span>&#160;        <span class="keywordflow">return</span>;</div><div class="line"><a name="l01581"></a><span class="lineno"> 1581</span>&#160;      }</div><div class="line"><a name="l01582"></a><span class="lineno"> 1582</span>&#160;</div><div class="line"><a name="l01583"></a><span class="lineno"> 1583</span>&#160;      fwrite(&amp;p-&gt;<a class="code" href="struct_mrpt___parameters.html#a4a0ab518bfa3786b59e58481bb7de814">n_trees</a>, <span class="keyword">sizeof</span>(<span class="keywordtype">int</span>), 1, fd);</div><div class="line"><a name="l01584"></a><span class="lineno"> 1584</span>&#160;      fwrite(&amp;p-&gt;<a class="code" href="struct_mrpt___parameters.html#a74ecd1a5f8320ec0cb30d2bfb7979fd0">depth</a>, <span class="keyword">sizeof</span>(<span class="keywordtype">int</span>), 1, fd);</div><div class="line"><a name="l01585"></a><span class="lineno"> 1585</span>&#160;      fwrite(&amp;p-&gt;<a class="code" href="struct_mrpt___parameters.html#ab34a16df707e0abd7b8b0bf4959f36b6">votes</a>, <span class="keyword">sizeof</span>(<span class="keywordtype">int</span>), 1, fd);</div><div class="line"><a name="l01586"></a><span class="lineno"> 1586</span>&#160;      fwrite(&amp;p-&gt;<a class="code" href="struct_mrpt___parameters.html#ad8b374d6671223a498e67cb98d8ea191">k</a>, <span class="keyword">sizeof</span>(<span class="keywordtype">int</span>), 1, fd);</div><div class="line"><a name="l01587"></a><span class="lineno"> 1587</span>&#160;      fwrite(&amp;p-&gt;<a class="code" href="struct_mrpt___parameters.html#a35ec37c03c3ff33e3dc6a9944d581327">estimated_qtime</a>, <span class="keyword">sizeof</span>(<span class="keywordtype">double</span>), 1, fd);</div><div class="line"><a name="l01588"></a><span class="lineno"> 1588</span>&#160;      fwrite(&amp;p-&gt;<a class="code" href="struct_mrpt___parameters.html#ab2fd9db6c27b3024a4e2990c2b4b52ca">estimated_recall</a>, <span class="keyword">sizeof</span>(<span class="keywordtype">double</span>), 1, fd);</div><div class="line"><a name="l01589"></a><span class="lineno"> 1589</span>&#160;    }</div><div class="line"><a name="l01590"></a><span class="lineno"> 1590</span>&#160;</div><div class="line"><a name="l01591"></a><span class="lineno"> 1591</span>&#160;    <span class="keywordtype">void</span> read_parameters(<a class="code" href="struct_mrpt___parameters.html">Mrpt_Parameters</a> *p, FILE *fd) {</div><div class="line"><a name="l01592"></a><span class="lineno"> 1592</span>&#160;      fread(&amp;p-&gt;<a class="code" href="struct_mrpt___parameters.html#a4a0ab518bfa3786b59e58481bb7de814">n_trees</a>, <span class="keyword">sizeof</span>(<span class="keywordtype">int</span>), 1, fd);</div><div class="line"><a name="l01593"></a><span class="lineno"> 1593</span>&#160;      fread(&amp;p-&gt;<a class="code" href="struct_mrpt___parameters.html#a74ecd1a5f8320ec0cb30d2bfb7979fd0">depth</a>, <span class="keyword">sizeof</span>(<span class="keywordtype">int</span>), 1, fd);</div><div class="line"><a name="l01594"></a><span class="lineno"> 1594</span>&#160;      fread(&amp;p-&gt;<a class="code" href="struct_mrpt___parameters.html#ab34a16df707e0abd7b8b0bf4959f36b6">votes</a>, <span class="keyword">sizeof</span>(<span class="keywordtype">int</span>), 1, fd);</div><div class="line"><a name="l01595"></a><span class="lineno"> 1595</span>&#160;      fread(&amp;p-&gt;<a class="code" href="struct_mrpt___parameters.html#ad8b374d6671223a498e67cb98d8ea191">k</a>, <span class="keyword">sizeof</span>(<span class="keywordtype">int</span>), 1, fd);</div><div class="line"><a name="l01596"></a><span class="lineno"> 1596</span>&#160;      fread(&amp;p-&gt;<a class="code" href="struct_mrpt___parameters.html#a35ec37c03c3ff33e3dc6a9944d581327">estimated_qtime</a>, <span class="keyword">sizeof</span>(<span class="keywordtype">double</span>), 1, fd);</div><div class="line"><a name="l01597"></a><span class="lineno"> 1597</span>&#160;      fread(&amp;p-&gt;<a class="code" href="struct_mrpt___parameters.html#ab2fd9db6c27b3024a4e2990c2b4b52ca">estimated_recall</a>, <span class="keyword">sizeof</span>(<span class="keywordtype">double</span>), 1, fd);</div><div class="line"><a name="l01598"></a><span class="lineno"> 1598</span>&#160;    }</div><div class="line"><a name="l01599"></a><span class="lineno"> 1599</span>&#160;</div><div class="line"><a name="l01600"></a><span class="lineno"> 1600</span>&#160;    <span class="keywordtype">void</span> write_parameter_list(<span class="keyword">const</span> std::set&lt;<a class="code" href="struct_mrpt___parameters.html">Mrpt_Parameters</a>,decltype(is_faster)*&gt; &amp;pars, FILE *fd)<span class="keyword"> const </span>{</div><div class="line"><a name="l01601"></a><span class="lineno"> 1601</span>&#160;      <span class="keywordflow">if</span> (!fd) {</div><div class="line"><a name="l01602"></a><span class="lineno"> 1602</span>&#160;        <span class="keywordflow">return</span>;</div><div class="line"><a name="l01603"></a><span class="lineno"> 1603</span>&#160;      }</div><div class="line"><a name="l01604"></a><span class="lineno"> 1604</span>&#160;</div><div class="line"><a name="l01605"></a><span class="lineno"> 1605</span>&#160;      <span class="keywordtype">int</span> par_sz = pars.size();</div><div class="line"><a name="l01606"></a><span class="lineno"> 1606</span>&#160;      fwrite(&amp;par_sz, <span class="keyword">sizeof</span>(<span class="keywordtype">int</span>), 1, fd);</div><div class="line"><a name="l01607"></a><span class="lineno"> 1607</span>&#160;</div><div class="line"><a name="l01608"></a><span class="lineno"> 1608</span>&#160;      <span class="keywordflow">for</span> (<span class="keyword">const</span> <span class="keyword">auto</span> p : pars)</div><div class="line"><a name="l01609"></a><span class="lineno"> 1609</span>&#160;        write_parameters(&amp;p, fd);</div><div class="line"><a name="l01610"></a><span class="lineno"> 1610</span>&#160;    }</div><div class="line"><a name="l01611"></a><span class="lineno"> 1611</span>&#160;</div><div class="line"><a name="l01612"></a><span class="lineno"> 1612</span>&#160;    <span class="keywordtype">void</span> read_parameter_list(FILE *fd) {</div><div class="line"><a name="l01613"></a><span class="lineno"> 1613</span>&#160;      <span class="keywordflow">if</span> (!fd) {</div><div class="line"><a name="l01614"></a><span class="lineno"> 1614</span>&#160;        <span class="keywordflow">return</span>;</div><div class="line"><a name="l01615"></a><span class="lineno"> 1615</span>&#160;      }</div><div class="line"><a name="l01616"></a><span class="lineno"> 1616</span>&#160;</div><div class="line"><a name="l01617"></a><span class="lineno"> 1617</span>&#160;      opt_pars = std::set&lt;Mrpt_Parameters,decltype(is_faster)*&gt;(is_faster);</div><div class="line"><a name="l01618"></a><span class="lineno"> 1618</span>&#160;      <span class="keywordtype">int</span> par_sz = 0;</div><div class="line"><a name="l01619"></a><span class="lineno"> 1619</span>&#160;      fread(&amp;par_sz, <span class="keyword">sizeof</span>(<span class="keywordtype">int</span>), 1, fd);</div><div class="line"><a name="l01620"></a><span class="lineno"> 1620</span>&#160;</div><div class="line"><a name="l01621"></a><span class="lineno"> 1621</span>&#160;      <span class="keywordflow">for</span> (<span class="keywordtype">int</span> i = 0; i &lt; par_sz; ++i) {</div><div class="line"><a name="l01622"></a><span class="lineno"> 1622</span>&#160;        <a class="code" href="struct_mrpt___parameters.html">Mrpt_Parameters</a> p;</div><div class="line"><a name="l01623"></a><span class="lineno"> 1623</span>&#160;        read_parameters(&amp;p, fd);</div><div class="line"><a name="l01624"></a><span class="lineno"> 1624</span>&#160;        opt_pars.insert(p);</div><div class="line"><a name="l01625"></a><span class="lineno"> 1625</span>&#160;      }</div><div class="line"><a name="l01626"></a><span class="lineno"> 1626</span>&#160;    }</div><div class="line"><a name="l01627"></a><span class="lineno"> 1627</span>&#160;</div><div class="line"><a name="l01628"></a><span class="lineno"> 1628</span>&#160;    <a class="code" href="struct_mrpt___parameters.html">Mrpt_Parameters</a> <a class="code" href="class_mrpt.html#a82a6cd3ce04ff8307bfd00d99fc5400a">parameters</a>(<span class="keywordtype">double</span> target_recall)<span class="keyword"> const </span>{</div><div class="line"><a name="l01629"></a><span class="lineno"> 1629</span>&#160;      <span class="keywordtype">double</span> tr = target_recall - epsilon;</div><div class="line"><a name="l01630"></a><span class="lineno"> 1630</span>&#160;      <span class="keywordflow">for</span> (<span class="keyword">const</span> <span class="keyword">auto</span> &amp;p : opt_pars) {</div><div class="line"><a name="l01631"></a><span class="lineno"> 1631</span>&#160;        <span class="keywordflow">if</span> (p.<a class="code" href="struct_mrpt___parameters.html#ab2fd9db6c27b3024a4e2990c2b4b52ca">estimated_recall</a> &gt; tr) {</div><div class="line"><a name="l01632"></a><span class="lineno"> 1632</span>&#160;          <span class="keywordflow">return</span> p;</div><div class="line"><a name="l01633"></a><span class="lineno"> 1633</span>&#160;        }</div><div class="line"><a name="l01634"></a><span class="lineno"> 1634</span>&#160;      }</div><div class="line"><a name="l01635"></a><span class="lineno"> 1635</span>&#160;</div><div class="line"><a name="l01636"></a><span class="lineno"> 1636</span>&#160;      <span class="keywordflow">if</span> (!opt_pars.empty()) {</div><div class="line"><a name="l01637"></a><span class="lineno"> 1637</span>&#160;        <span class="keywordflow">return</span> *(opt_pars.rbegin());</div><div class="line"><a name="l01638"></a><span class="lineno"> 1638</span>&#160;      }</div><div class="line"><a name="l01639"></a><span class="lineno"> 1639</span>&#160;</div><div class="line"><a name="l01640"></a><span class="lineno"> 1640</span>&#160;      <span class="keywordflow">return</span> <a class="code" href="struct_mrpt___parameters.html">Mrpt_Parameters</a>();</div><div class="line"><a name="l01641"></a><span class="lineno"> 1641</span>&#160;    }</div><div class="line"><a name="l01642"></a><span class="lineno"> 1642</span>&#160;</div><div class="line"><a name="l01648"></a><span class="lineno"> 1648</span>&#160;    <span class="keyword">static</span> <span class="keywordtype">void</span> count_leaf_sizes(<span class="keywordtype">int</span> n, <span class="keywordtype">int</span> level, <span class="keywordtype">int</span> tree_depth, std::vector&lt;int&gt; &amp;out_leaf_sizes) {</div><div class="line"><a name="l01649"></a><span class="lineno"> 1649</span>&#160;      <span class="keywordflow">if</span> (level == tree_depth) {</div><div class="line"><a name="l01650"></a><span class="lineno"> 1650</span>&#160;        out_leaf_sizes.push_back(n);</div><div class="line"><a name="l01651"></a><span class="lineno"> 1651</span>&#160;        <span class="keywordflow">return</span>;</div><div class="line"><a name="l01652"></a><span class="lineno"> 1652</span>&#160;      }</div><div class="line"><a name="l01653"></a><span class="lineno"> 1653</span>&#160;</div><div class="line"><a name="l01654"></a><span class="lineno"> 1654</span>&#160;      count_leaf_sizes(n - n / 2, level + 1, tree_depth, out_leaf_sizes);</div><div class="line"><a name="l01655"></a><span class="lineno"> 1655</span>&#160;      count_leaf_sizes(n / 2, level + 1, tree_depth, out_leaf_sizes);</div><div class="line"><a name="l01656"></a><span class="lineno"> 1656</span>&#160;    }</div><div class="line"><a name="l01657"></a><span class="lineno"> 1657</span>&#160;</div><div class="line"><a name="l01664"></a><span class="lineno"> 1664</span>&#160;    <span class="keyword">static</span> <span class="keywordtype">void</span> count_first_leaf_indices(std::vector&lt;int&gt; &amp;indices, <span class="keywordtype">int</span> n, <span class="keywordtype">int</span> depth) {</div><div class="line"><a name="l01665"></a><span class="lineno"> 1665</span>&#160;      std::vector&lt;int&gt; leaf_sizes;</div><div class="line"><a name="l01666"></a><span class="lineno"> 1666</span>&#160;      count_leaf_sizes(n, 0, depth, leaf_sizes);</div><div class="line"><a name="l01667"></a><span class="lineno"> 1667</span>&#160;</div><div class="line"><a name="l01668"></a><span class="lineno"> 1668</span>&#160;      indices = std::vector&lt;int&gt;(leaf_sizes.size() + 1);</div><div class="line"><a name="l01669"></a><span class="lineno"> 1669</span>&#160;      indices[0] = 0;</div><div class="line"><a name="l01670"></a><span class="lineno"> 1670</span>&#160;      <span class="keywordflow">for</span> (<span class="keywordtype">int</span> i = 0; i &lt; (int) leaf_sizes.size(); ++i)</div><div class="line"><a name="l01671"></a><span class="lineno"> 1671</span>&#160;        indices[i + 1] = indices[i] + leaf_sizes[i];</div><div class="line"><a name="l01672"></a><span class="lineno"> 1672</span>&#160;    }</div><div class="line"><a name="l01673"></a><span class="lineno"> 1673</span>&#160;</div><div class="line"><a name="l01674"></a><span class="lineno"> 1674</span>&#160;    <span class="keyword">static</span> <span class="keywordtype">void</span> count_first_leaf_indices_all(std::vector&lt;std::vector&lt;int&gt;&gt; &amp;indices, <span class="keywordtype">int</span> n, <span class="keywordtype">int</span> depth_max) {</div><div class="line"><a name="l01675"></a><span class="lineno"> 1675</span>&#160;      <span class="keywordflow">for</span> (<span class="keywordtype">int</span> d = 0; d &lt;= depth_max; ++d) {</div><div class="line"><a name="l01676"></a><span class="lineno"> 1676</span>&#160;        std::vector&lt;int&gt; idx;</div><div class="line"><a name="l01677"></a><span class="lineno"> 1677</span>&#160;        count_first_leaf_indices(idx, n, d);</div><div class="line"><a name="l01678"></a><span class="lineno"> 1678</span>&#160;        indices.push_back(idx);</div><div class="line"><a name="l01679"></a><span class="lineno"> 1679</span>&#160;      }</div><div class="line"><a name="l01680"></a><span class="lineno"> 1680</span>&#160;    }</div><div class="line"><a name="l01681"></a><span class="lineno"> 1681</span>&#160;</div><div class="line"><a name="l01682"></a><span class="lineno"> 1682</span>&#160;    <span class="keyword">static</span> <span class="keywordtype">double</span> predict_theil_sen(<span class="keywordtype">double</span> x, std::pair&lt;double,double&gt; beta) {</div><div class="line"><a name="l01683"></a><span class="lineno"> 1683</span>&#160;      <span class="keywordflow">return</span> beta.first + beta.second * x;</div><div class="line"><a name="l01684"></a><span class="lineno"> 1684</span>&#160;    }</div><div class="line"><a name="l01685"></a><span class="lineno"> 1685</span>&#160;</div><div class="line"><a name="l01686"></a><span class="lineno"> 1686</span>&#160;    <span class="keywordtype">double</span> get_candidate_set_size(<span class="keywordtype">int</span> tree, <span class="keywordtype">int</span> depth, <span class="keywordtype">int</span> v)<span class="keyword"> const </span>{</div><div class="line"><a name="l01687"></a><span class="lineno"> 1687</span>&#160;      <span class="keywordflow">return</span> cs_sizes[depth - depth_min](v - 1, tree - 1);</div><div class="line"><a name="l01688"></a><span class="lineno"> 1688</span>&#160;    }</div><div class="line"><a name="l01689"></a><span class="lineno"> 1689</span>&#160;</div><div class="line"><a name="l01690"></a><span class="lineno"> 1690</span>&#160;    <span class="keywordtype">double</span> get_projection_time(<span class="keywordtype">int</span> n_trees, <span class="keywordtype">int</span> depth, <span class="keywordtype">int</span> v)<span class="keyword"> const </span>{</div><div class="line"><a name="l01691"></a><span class="lineno"> 1691</span>&#160;      <span class="keywordflow">return</span> predict_theil_sen(n_trees * depth, beta_projection);</div><div class="line"><a name="l01692"></a><span class="lineno"> 1692</span>&#160;    }</div><div class="line"><a name="l01693"></a><span class="lineno"> 1693</span>&#160;</div><div class="line"><a name="l01694"></a><span class="lineno"> 1694</span>&#160;    <span class="keywordtype">double</span> get_voting_time(<span class="keywordtype">int</span> n_trees, <span class="keywordtype">int</span> depth, <span class="keywordtype">int</span> v)<span class="keyword"> const </span>{</div><div class="line"><a name="l01695"></a><span class="lineno"> 1695</span>&#160;      <span class="keyword">const</span> std::map&lt;int,std::pair&lt;double,double&gt;&gt; &amp;beta = beta_voting[depth - depth_min];</div><div class="line"><a name="l01696"></a><span class="lineno"> 1696</span>&#160;</div><div class="line"><a name="l01697"></a><span class="lineno"> 1697</span>&#160;      <span class="keywordflow">if</span> (v &lt;= 0 || beta.empty()) {</div><div class="line"><a name="l01698"></a><span class="lineno"> 1698</span>&#160;        <span class="keywordflow">return</span> 0.0;</div><div class="line"><a name="l01699"></a><span class="lineno"> 1699</span>&#160;      }</div><div class="line"><a name="l01700"></a><span class="lineno"> 1700</span>&#160;</div><div class="line"><a name="l01701"></a><span class="lineno"> 1701</span>&#160;      <span class="keywordflow">for</span> (<span class="keyword">const</span> <span class="keyword">auto</span> &amp;b : beta) {</div><div class="line"><a name="l01702"></a><span class="lineno"> 1702</span>&#160;        <span class="keywordflow">if</span> (v &lt;= b.first) {</div><div class="line"><a name="l01703"></a><span class="lineno"> 1703</span>&#160;          <span class="keywordflow">return</span> predict_theil_sen(n_trees, b.second);</div><div class="line"><a name="l01704"></a><span class="lineno"> 1704</span>&#160;        }</div><div class="line"><a name="l01705"></a><span class="lineno"> 1705</span>&#160;      }</div><div class="line"><a name="l01706"></a><span class="lineno"> 1706</span>&#160;</div><div class="line"><a name="l01707"></a><span class="lineno"> 1707</span>&#160;      <span class="keywordflow">return</span> predict_theil_sen(n_trees, beta.rbegin()-&gt;second);</div><div class="line"><a name="l01708"></a><span class="lineno"> 1708</span>&#160;    }</div><div class="line"><a name="l01709"></a><span class="lineno"> 1709</span>&#160;</div><div class="line"><a name="l01710"></a><span class="lineno"> 1710</span>&#160;    <span class="keywordtype">double</span> get_exact_time(<span class="keywordtype">int</span> n_trees, <span class="keywordtype">int</span> depth, <span class="keywordtype">int</span> v)<span class="keyword"> const </span>{</div><div class="line"><a name="l01711"></a><span class="lineno"> 1711</span>&#160;      <span class="keywordflow">return</span> predict_theil_sen(get_candidate_set_size(n_trees, depth, v), beta_exact);</div><div class="line"><a name="l01712"></a><span class="lineno"> 1712</span>&#160;    }</div><div class="line"><a name="l01713"></a><span class="lineno"> 1713</span>&#160;</div><div class="line"><a name="l01714"></a><span class="lineno"> 1714</span>&#160;    <span class="keywordtype">double</span> get_query_time(<span class="keywordtype">int</span> tree, <span class="keywordtype">int</span> depth, <span class="keywordtype">int</span> v)<span class="keyword"> const </span>{</div><div class="line"><a name="l01715"></a><span class="lineno"> 1715</span>&#160;      <span class="keywordflow">return</span> get_projection_time(tree, depth, v)</div><div class="line"><a name="l01716"></a><span class="lineno"> 1716</span>&#160;           + get_voting_time(tree, depth, v)</div><div class="line"><a name="l01717"></a><span class="lineno"> 1717</span>&#160;           + get_exact_time(tree, depth, v);</div><div class="line"><a name="l01718"></a><span class="lineno"> 1718</span>&#160;    }</div><div class="line"><a name="l01719"></a><span class="lineno"> 1719</span>&#160;</div><div class="line"><a name="l01720"></a><span class="lineno"> 1720</span>&#160;    std::vector&lt;int&gt; sample_indices(<span class="keywordtype">int</span> n_test, <span class="keywordtype">int</span> seed = 0)<span class="keyword"> const </span>{</div><div class="line"><a name="l01721"></a><span class="lineno"> 1721</span>&#160;      std::random_device rd;</div><div class="line"><a name="l01722"></a><span class="lineno"> 1722</span>&#160;      <span class="keywordtype">int</span> s = seed ? seed : rd();</div><div class="line"><a name="l01723"></a><span class="lineno"> 1723</span>&#160;      std::mt19937 gen(s);</div><div class="line"><a name="l01724"></a><span class="lineno"> 1724</span>&#160;</div><div class="line"><a name="l01725"></a><span class="lineno"> 1725</span>&#160;      std::vector&lt;int&gt; indices_data(n_samples);</div><div class="line"><a name="l01726"></a><span class="lineno"> 1726</span>&#160;      std::iota(indices_data.begin(), indices_data.end(), 0);</div><div class="line"><a name="l01727"></a><span class="lineno"> 1727</span>&#160;      std::shuffle(indices_data.begin(), indices_data.end(), gen);</div><div class="line"><a name="l01728"></a><span class="lineno"> 1728</span>&#160;      <span class="keywordflow">return</span> std::vector&lt;int&gt;(indices_data.begin(), indices_data.begin() + n_test);</div><div class="line"><a name="l01729"></a><span class="lineno"> 1729</span>&#160;    }</div><div class="line"><a name="l01730"></a><span class="lineno"> 1730</span>&#160;</div><div class="line"><a name="l01731"></a><span class="lineno"> 1731</span>&#160;    Eigen::MatrixXf <a class="code" href="class_mrpt.html#add8686a73cfae8d8391e85314acbe81a">subset</a>(<span class="keyword">const</span> std::vector&lt;int&gt; &amp;indices)<span class="keyword"> const </span>{</div><div class="line"><a name="l01732"></a><span class="lineno"> 1732</span>&#160;      <span class="keywordtype">int</span> n_test = indices.size();</div><div class="line"><a name="l01733"></a><span class="lineno"> 1733</span>&#160;      Eigen::MatrixXf Q = Eigen::MatrixXf(dim, n_test);</div><div class="line"><a name="l01734"></a><span class="lineno"> 1734</span>&#160;      <span class="keywordflow">for</span>(<span class="keywordtype">int</span> i = 0; i &lt; n_test; ++i)</div><div class="line"><a name="l01735"></a><span class="lineno"> 1735</span>&#160;        Q.col(i) = X.col(indices[i]);</div><div class="line"><a name="l01736"></a><span class="lineno"> 1736</span>&#160;</div><div class="line"><a name="l01737"></a><span class="lineno"> 1737</span>&#160;      <span class="keywordflow">return</span> Q;</div><div class="line"><a name="l01738"></a><span class="lineno"> 1738</span>&#160;    }</div><div class="line"><a name="l01739"></a><span class="lineno"> 1739</span>&#160;</div><div class="line"><a name="l01740"></a><span class="lineno"> 1740</span>&#160;</div><div class="line"><a name="l01741"></a><span class="lineno"> 1741</span>&#160;    <span class="keyword">const</span> Eigen::Map&lt;const Eigen::MatrixXf&gt; X; <span class="comment">// the data matrix</span></div><div class="line"><a name="l01742"></a><span class="lineno"> 1742</span>&#160;    Eigen::MatrixXf split_points; <span class="comment">// all split points in all trees</span></div><div class="line"><a name="l01743"></a><span class="lineno"> 1743</span>&#160;    std::vector&lt;std::vector&lt;int&gt;&gt; tree_leaves; <span class="comment">// contains all leaves of all trees</span></div><div class="line"><a name="l01744"></a><span class="lineno"> 1744</span>&#160;    Eigen::Matrix&lt;float, Eigen::Dynamic, Eigen::Dynamic, Eigen::RowMajor&gt; dense_random_matrix; <span class="comment">// random vectors needed for all the RP-trees</span></div><div class="line"><a name="l01745"></a><span class="lineno"> 1745</span>&#160;    Eigen::SparseMatrix&lt;float, Eigen::RowMajor&gt; sparse_random_matrix; <span class="comment">// random vectors needed for all the RP-trees</span></div><div class="line"><a name="l01746"></a><span class="lineno"> 1746</span>&#160;    std::vector&lt;std::vector&lt;int&gt;&gt; leaf_first_indices_all; <span class="comment">// first indices for each level</span></div><div class="line"><a name="l01747"></a><span class="lineno"> 1747</span>&#160;    std::vector&lt;int&gt; leaf_first_indices; <span class="comment">// first indices of each leaf of tree in tree_leaves</span></div><div class="line"><a name="l01748"></a><span class="lineno"> 1748</span>&#160;</div><div class="line"><a name="l01749"></a><span class="lineno"> 1749</span>&#160;    <span class="keyword">const</span> <span class="keywordtype">int</span> n_samples; <span class="comment">// sample size of data</span></div><div class="line"><a name="l01750"></a><span class="lineno"> 1750</span>&#160;    <span class="keyword">const</span> <span class="keywordtype">int</span> dim; <span class="comment">// dimension of data</span></div><div class="line"><a name="l01751"></a><span class="lineno"> 1751</span>&#160;    <a class="code" href="struct_mrpt___parameters.html">Mrpt_Parameters</a> par;</div><div class="line"><a name="l01752"></a><span class="lineno"> 1752</span>&#160;    <span class="keywordtype">int</span> n_trees = 0; <span class="comment">// number of RP-trees</span></div><div class="line"><a name="l01753"></a><span class="lineno"> 1753</span>&#160;    <span class="keywordtype">int</span> depth = 0; <span class="comment">// depth of an RP-tree with median split</span></div><div class="line"><a name="l01754"></a><span class="lineno"> 1754</span>&#160;    <span class="keywordtype">float</span> density = -1.0; <span class="comment">// expected ratio of non-zero components in a projection matrix</span></div><div class="line"><a name="l01755"></a><span class="lineno"> 1755</span>&#160;    <span class="keywordtype">int</span> n_pool = 0; <span class="comment">// amount of random vectors needed for all the RP-trees</span></div><div class="line"><a name="l01756"></a><span class="lineno"> 1756</span>&#160;    <span class="keywordtype">int</span> n_array = 0; <span class="comment">// length of the one RP-tree as array</span></div><div class="line"><a name="l01757"></a><span class="lineno"> 1757</span>&#160;    <span class="keywordtype">int</span> votes = 0; <span class="comment">// optimal number of votes to use</span></div><div class="line"><a name="l01758"></a><span class="lineno"> 1758</span>&#160;    <span class="keywordtype">int</span> k = 0;</div><div class="line"><a name="l01759"></a><span class="lineno"> 1759</span>&#160;    <span class="keyword">enum</span> itype {normal, autotuned, autotuned_unpruned};</div><div class="line"><a name="l01760"></a><span class="lineno"> 1760</span>&#160;    itype index_type = normal;</div><div class="line"><a name="l01761"></a><span class="lineno"> 1761</span>&#160;</div><div class="line"><a name="l01762"></a><span class="lineno"> 1762</span>&#160;    <span class="comment">// Member variables used in autotuning:</span></div><div class="line"><a name="l01763"></a><span class="lineno"> 1763</span>&#160;    <span class="keywordtype">int</span> depth_min = 0;</div><div class="line"><a name="l01764"></a><span class="lineno"> 1764</span>&#160;    <span class="keywordtype">int</span> votes_max = 0;</div><div class="line"><a name="l01765"></a><span class="lineno"> 1765</span>&#160;    <span class="keyword">const</span> <span class="keywordtype">double</span> epsilon = 0.0001; <span class="comment">// error bound for comparisons of recall levels</span></div><div class="line"><a name="l01766"></a><span class="lineno"> 1766</span>&#160;    std::vector&lt;Eigen::MatrixXd&gt; cs_sizes;</div><div class="line"><a name="l01767"></a><span class="lineno"> 1767</span>&#160;    std::pair&lt;double,double&gt; beta_projection, beta_exact;</div><div class="line"><a name="l01768"></a><span class="lineno"> 1768</span>&#160;    std::vector&lt;std::map&lt;int,std::pair&lt;double,double&gt;&gt;&gt; beta_voting;</div><div class="line"><a name="l01769"></a><span class="lineno"> 1769</span>&#160;    std::set&lt;Mrpt_Parameters,decltype(is_faster)*&gt; opt_pars;</div><div class="line"><a name="l01770"></a><span class="lineno"> 1770</span>&#160;};</div><div class="line"><a name="l01771"></a><span class="lineno"> 1771</span>&#160;</div><div class="line"><a name="l01772"></a><span class="lineno"> 1772</span>&#160;<span class="preprocessor">#endif // CPP_MRPT_H_</span></div><div class="ttc" id="class_mrpt_html_a35235eb7acf4bff5fbfb12c9cd82f0b8"><div class="ttname"><a href="class_mrpt.html#a35235eb7acf4bff5fbfb12c9cd82f0b8">Mrpt::query</a></div><div class="ttdeci">void query(const Eigen::Ref&lt; const Eigen::VectorXf &gt; &amp;q, int k, int vote_threshold, int *out, float *out_distances=nullptr, int *out_n_elected=nullptr) const</div><div class="ttdef"><b>Definition:</b> Mrpt.h:740</div></div>
<div class="ttc" id="class_mrpt_html_a16fac96c266abcc778694e0b0cdc5f3c"><div class="ttname"><a href="class_mrpt.html#a16fac96c266abcc778694e0b0cdc5f3c">Mrpt::query</a></div><div class="ttdeci">void query(const float *q, int *out, float *out_distances=nullptr, int *out_n_elected=nullptr) const</div><div class="ttdef"><b>Definition:</b> Mrpt.h:767</div></div>
<div class="ttc" id="class_mrpt_html_a64adc0d6952d12e4599459f8654e860e"><div class="ttname"><a href="class_mrpt.html#a64adc0d6952d12e4599459f8654e860e">Mrpt::is_autotuned</a></div><div class="ttdeci">bool is_autotuned() const</div><div class="ttdef"><b>Definition:</b> Mrpt.h:298</div></div>
<div class="ttc" id="class_mrpt_html_a637e67fe6eba9a6711f5eee65ad4a0da"><div class="ttname"><a href="class_mrpt.html#a637e67fe6eba9a6711f5eee65ad4a0da">Mrpt::query</a></div><div class="ttdeci">void query(const float *data, int k, int vote_threshold, int *out, float *out_distances=nullptr, int *out_n_elected=nullptr) const</div><div class="ttdef"><b>Definition:</b> Mrpt.h:661</div></div>
<div class="ttc" id="struct_mrpt___parameters_html_ab34a16df707e0abd7b8b0bf4959f36b6"><div class="ttname"><a href="struct_mrpt___parameters.html#ab34a16df707e0abd7b8b0bf4959f36b6">Mrpt_Parameters::votes</a></div><div class="ttdeci">int votes</div><div class="ttdef"><b>Definition:</b> Mrpt.h:23</div></div>
<div class="ttc" id="class_mrpt_html_a40eae6f75c79b4ff0f5400b14f8d81c2"><div class="ttname"><a href="class_mrpt.html#a40eae6f75c79b4ff0f5400b14f8d81c2">Mrpt::grow</a></div><div class="ttdeci">void grow(const float *data, int n_test, int k_, int trees_max=-1, int depth_max=-1, int depth_min_=-1, int votes_max_=-1, float density_=-1.0, int seed=0, const std::vector&lt; int &gt; &amp;indices_test={})</div><div class="ttdef"><b>Definition:</b> Mrpt.h:344</div></div>
<div class="ttc" id="class_mrpt_html_a82a6cd3ce04ff8307bfd00d99fc5400a"><div class="ttname"><a href="class_mrpt.html#a82a6cd3ce04ff8307bfd00d99fc5400a">Mrpt::parameters</a></div><div class="ttdeci">Mrpt_Parameters parameters() const</div><div class="ttdef"><b>Definition:</b> Mrpt.h:281</div></div>
<div class="ttc" id="class_mrpt_html_a0907893c90a6255d3cf8d11e940c6ce3"><div class="ttname"><a href="class_mrpt.html#a0907893c90a6255d3cf8d11e940c6ce3">Mrpt::optimal_parameters</a></div><div class="ttdeci">std::vector&lt; Mrpt_Parameters &gt; optimal_parameters() const</div><div class="ttdef"><b>Definition:</b> Mrpt.h:625</div></div>
<div class="ttc" id="class_mrpt_html_af31f777278ec7030e7351b6ed57ca766"><div class="ttname"><a href="class_mrpt.html#af31f777278ec7030e7351b6ed57ca766">Mrpt::load</a></div><div class="ttdeci">bool load(const char *path)</div><div class="ttdef"><b>Definition:</b> Mrpt.h:958</div></div>
<div class="ttc" id="struct_mrpt___parameters_html"><div class="ttname"><a href="struct_mrpt___parameters.html">Mrpt_Parameters</a></div><div class="ttdef"><b>Definition:</b> Mrpt.h:19</div></div>
<div class="ttc" id="struct_mrpt___parameters_html_a74ecd1a5f8320ec0cb30d2bfb7979fd0"><div class="ttname"><a href="struct_mrpt___parameters.html#a74ecd1a5f8320ec0cb30d2bfb7979fd0">Mrpt_Parameters::depth</a></div><div class="ttdeci">int depth</div><div class="ttdef"><b>Definition:</b> Mrpt.h:21</div></div>
<div class="ttc" id="struct_mrpt___parameters_html_ab2fd9db6c27b3024a4e2990c2b4b52ca"><div class="ttname"><a href="struct_mrpt___parameters.html#ab2fd9db6c27b3024a4e2990c2b4b52ca">Mrpt_Parameters::estimated_recall</a></div><div class="ttdeci">double estimated_recall</div><div class="ttdef"><b>Definition:</b> Mrpt.h:25</div></div>
<div class="ttc" id="class_mrpt_html_a19867c3d3f83e50c88e1edde63a81985"><div class="ttname"><a href="class_mrpt.html#a19867c3d3f83e50c88e1edde63a81985">Mrpt::exact_knn</a></div><div class="ttdeci">static void exact_knn(const float *q_data, const float *X_data, int dim, int n_samples, int k, int *out, float *out_distances=nullptr)</div><div class="ttdef"><b>Definition:</b> Mrpt.h:814</div></div>
<div class="ttc" id="class_mrpt_html_a708704bc812a926c1dc950c267ae513b"><div class="ttname"><a href="class_mrpt.html#a708704bc812a926c1dc950c267ae513b">Mrpt::exact_knn</a></div><div class="ttdeci">void exact_knn(const float *q, int k, int *out, float *out_distances=nullptr) const</div><div class="ttdef"><b>Definition:</b> Mrpt.h:874</div></div>
<div class="ttc" id="class_mrpt_html_a6911b63fb6fb87129f893831391f42dd"><div class="ttname"><a href="class_mrpt.html#a6911b63fb6fb87129f893831391f42dd">Mrpt::grow_autotune</a></div><div class="ttdeci">void grow_autotune(double target_recall, int k_, int trees_max=-1, int depth_max=-1, int depth_min_=-1, int votes_max_=-1, float density_=-1.0, int seed=0, int n_test=100)</div><div class="ttdef"><b>Definition:</b> Mrpt.h:256</div></div>
<div class="ttc" id="class_mrpt_html_a4e4195e845c91014df816f4e27a38b48"><div class="ttname"><a href="class_mrpt.html#a4e4195e845c91014df816f4e27a38b48">Mrpt::Mrpt</a></div><div class="ttdeci">Mrpt(const float *X_, int dim_, int n_samples_)</div><div class="ttdef"><b>Definition:</b> Mrpt.h:60</div></div>
<div class="ttc" id="class_mrpt_html"><div class="ttname"><a href="class_mrpt.html">Mrpt</a></div><div class="ttdef"><b>Definition:</b> Mrpt.h:28</div></div>
<div class="ttc" id="struct_mrpt___parameters_html_a4a0ab518bfa3786b59e58481bb7de814"><div class="ttname"><a href="struct_mrpt___parameters.html#a4a0ab518bfa3786b59e58481bb7de814">Mrpt_Parameters::n_trees</a></div><div class="ttdeci">int n_trees</div><div class="ttdef"><b>Definition:</b> Mrpt.h:20</div></div>
<div class="ttc" id="class_mrpt_html_ab06ab10bca1f953c6581f7d412b4661c"><div class="ttname"><a href="class_mrpt.html#ab06ab10bca1f953c6581f7d412b4661c">Mrpt::exact_knn</a></div><div class="ttdeci">static void exact_knn(const Eigen::Ref&lt; const Eigen::VectorXf &gt; &amp;q, const Eigen::Ref&lt; const Eigen::MatrixXf &gt; &amp;X, int k, int *out, float *out_distances=nullptr)</div><div class="ttdef"><b>Definition:</b> Mrpt.h:862</div></div>
<div class="ttc" id="class_mrpt_html_a2bf7a8162fc7f6bd56d10097aeedffc8"><div class="ttname"><a href="class_mrpt.html#a2bf7a8162fc7f6bd56d10097aeedffc8">Mrpt::grow</a></div><div class="ttdeci">void grow(double target_recall, const float *Q, int n_test, int k_, int trees_max=-1, int depth_max=-1, int depth_min_=-1, int votes_max_=-1, float density=-1.0, int seed=0, const std::vector&lt; int &gt; &amp;indices_test={})</div><div class="ttdef"><b>Definition:</b> Mrpt.h:218</div></div>
<div class="ttc" id="class_mrpt_html_a9579b70d0553126363f6112a10ffb0eb"><div class="ttname"><a href="class_mrpt.html#a9579b70d0553126363f6112a10ffb0eb">Mrpt::query</a></div><div class="ttdeci">void query(const Eigen::Ref&lt; const Eigen::VectorXf &gt; &amp;q, int *out, float *out_distances=nullptr, int *out_n_elected=nullptr) const</div><div class="ttdef"><b>Definition:</b> Mrpt.h:788</div></div>
<div class="ttc" id="class_mrpt_html_a4dc319eab5e2bc9a5a6d7a6007cef34d"><div class="ttname"><a href="class_mrpt.html#a4dc319eab5e2bc9a5a6d7a6007cef34d">Mrpt::subset_pointer</a></div><div class="ttdeci">Mrpt * subset_pointer(double target_recall) const</div><div class="ttdef"><b>Definition:</b> Mrpt.h:577</div></div>
<div class="ttc" id="class_mrpt_html_a7d789c72d12dd668fc5ffb157a9a52f4"><div class="ttname"><a href="class_mrpt.html#a7d789c72d12dd668fc5ffb157a9a52f4">Mrpt::grow</a></div><div class="ttdeci">void grow(int n_trees_, int depth_, float density_=-1.0, int seed=0)</div><div class="ttdef"><b>Definition:</b> Mrpt.h:84</div></div>
<div class="ttc" id="class_mrpt_html_af936713e7629e08e88d11346f2639ba0"><div class="ttname"><a href="class_mrpt.html#af936713e7629e08e88d11346f2639ba0">Mrpt::save</a></div><div class="ttdeci">bool save(const char *path) const</div><div class="ttdef"><b>Definition:</b> Mrpt.h:905</div></div>
<div class="ttc" id="class_mrpt_html_a7a6325435fc140ac5394c2f0136ea211"><div class="ttname"><a href="class_mrpt.html#a7a6325435fc140ac5394c2f0136ea211">Mrpt::grow</a></div><div class="ttdeci">void grow(const Eigen::Ref&lt; const Eigen::MatrixXf &gt; &amp;Q, int k_, int trees_max=-1, int depth_max=-1, int depth_min_=-1, int votes_max_=-1, float density_=-1.0, int seed=0)</div><div class="ttdef"><b>Definition:</b> Mrpt.h:468</div></div>
<div class="ttc" id="class_mrpt_html_a24f2dfef8d422cb875abd18d411d47a1"><div class="ttname"><a href="class_mrpt.html#a24f2dfef8d422cb875abd18d411d47a1">Mrpt::exact_knn</a></div><div class="ttdeci">void exact_knn(const Eigen::Ref&lt; const Eigen::VectorXf &gt; &amp;q, int k, int *out, float *out_distances=nullptr) const</div><div class="ttdef"><b>Definition:</b> Mrpt.h:884</div></div>
<div class="ttc" id="class_mrpt_html_a960f1af81f2b56db7eb8cd0d5a352c14"><div class="ttname"><a href="class_mrpt.html#a960f1af81f2b56db7eb8cd0d5a352c14">Mrpt::empty</a></div><div class="ttdeci">bool empty() const</div><div class="ttdef"><b>Definition:</b> Mrpt.h:1029</div></div>
<div class="ttc" id="struct_mrpt___parameters_html_ad8b374d6671223a498e67cb98d8ea191"><div class="ttname"><a href="struct_mrpt___parameters.html#ad8b374d6671223a498e67cb98d8ea191">Mrpt_Parameters::k</a></div><div class="ttdeci">int k</div><div class="ttdef"><b>Definition:</b> Mrpt.h:22</div></div>
<div class="ttc" id="class_mrpt_html_a7360964458339aab0ddedde324a03e47"><div class="ttname"><a href="class_mrpt.html#a7360964458339aab0ddedde324a03e47">Mrpt::grow</a></div><div class="ttdeci">void grow(double target_recall, const Eigen::Ref&lt; const Eigen::MatrixXf &gt; &amp;Q, int k_, int trees_max=-1, int depth_max=-1, int depth_min_=-1, int votes_max_=-1, float density=-1.0, int seed=0)</div><div class="ttdef"><b>Definition:</b> Mrpt.h:178</div></div>
<div class="ttc" id="class_mrpt_html_a1e77a5c97c6ae94d17162f65041821bc"><div class="ttname"><a href="class_mrpt.html#a1e77a5c97c6ae94d17162f65041821bc">Mrpt::Mrpt</a></div><div class="ttdeci">Mrpt(const Eigen::Ref&lt; const Eigen::MatrixXf &gt; &amp;X_)</div><div class="ttdef"><b>Definition:</b> Mrpt.h:49</div></div>
<div class="ttc" id="class_mrpt_html_add8686a73cfae8d8391e85314acbe81a"><div class="ttname"><a href="class_mrpt.html#add8686a73cfae8d8391e85314acbe81a">Mrpt::subset</a></div><div class="ttdeci">Mrpt subset(double target_recall) const</div><div class="ttdef"><b>Definition:</b> Mrpt.h:527</div></div>
<div class="ttc" id="struct_mrpt___parameters_html_a35ec37c03c3ff33e3dc6a9944d581327"><div class="ttname"><a href="struct_mrpt___parameters.html#a35ec37c03c3ff33e3dc6a9944d581327">Mrpt_Parameters::estimated_qtime</a></div><div class="ttdeci">double estimated_qtime</div><div class="ttdef"><b>Definition:</b> Mrpt.h:24</div></div>
<div class="ttc" id="class_mrpt_html_a9ee717a90c0548f31577b033540dda44"><div class="ttname"><a href="class_mrpt.html#a9ee717a90c0548f31577b033540dda44">Mrpt::grow_autotune</a></div><div class="ttdeci">void grow_autotune(int k_, int trees_max=-1, int depth_max=-1, int depth_min_=-1, int votes_max_=-1, float density_=-1.0, int seed=0, int n_test=100)</div><div class="ttdef"><b>Definition:</b> Mrpt.h:503</div></div>
</div><!-- fragment --></div><!-- contents -->
<!-- start footer part -->
<hr class="footer"/><address class="footer"><small>
Generated by &#160;<a href="http://www.doxygen.org/index.html">
<img class="footer" src="doxygen.png" alt="doxygen"/>
</a> 1.8.14
</small></address>
</body>
</html>
