<div dir="ltr"><br><div class="gmail_extra"><br><div class="gmail_quote">On Wed, Jul 6, 2016 at 3:29 AM,  <span dir="ltr"><<a href="mailto:josef.pktd@gmail.com" target="_blank">josef.pktd@gmail.com</a>></span> wrote:<br><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex"><div dir="ltr"><br><div class="gmail_extra"><br><div class="gmail_quote"><span class="">On Wed, Jul 6, 2016 at 2:21 AM, Ralf Gommers <span dir="ltr"><<a href="mailto:ralf.gommers@gmail.com" target="_blank">ralf.gommers@gmail.com</a>></span> wrote:<br><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex"><div dir="ltr"><br><div class="gmail_extra"><span><br><div class="gmail_quote">On Wed, Jul 6, 2016 at 7:06 AM, Nathaniel Smith <span dir="ltr"><<a href="mailto:njs@pobox.com" target="_blank">njs@pobox.com</a>></span> wrote:<br><br><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex"><span><p dir="ltr">On Jul 5, 2016 9:09 PM, "Joseph Fox-Rabinovitz" <<a href="mailto:jfoxrabinovitz@gmail.com" target="_blank">jfoxrabinovitz@gmail.com</a>> wrote:<br>
><br>
> Hi,<br>
><br>
> I have generalized np.atleast_1d, np.atleast_2d, np.atleast_3d with a<br>
> function np.atleast_nd in PR#7804<br>
> (<a href="https://github.com/numpy/numpy/pull/7804" target="_blank">https://github.com/numpy/numpy/pull/7804</a>).<br>
><br>
> As a result of this PR, I have a couple of questions about<br>
> `np.atleast_3d`. `np.atleast_3d` appears to do something weird with<br>
> the dimensions: If the input is 1D, it prepends and appends a size-1<br>
> dimension. If the input is 2D, it appends a size-1 dimension. This is<br>
> inconsistent with `np.atleast_2d`, which always prepends (as does<br>
> `np.atleast_nd`).<br>
><br>
>   - Is there any reason for this behavior?<br>
>   - Can it be cleaned up (e.g., by reimplementing `np.atleast_3d` in<br>
> terms of `np.atleast_nd`, which is actually much simpler)? This would<br>
> be a slight API change since the output would not be exactly the same.</p>
</span><p dir="ltr">Changing atleast_3d seems likely to break a bunch of stuff...</p>
<p dir="ltr">Beyond that, I find it hard to have an opinion about the best design for these functions, because I don't think I've ever encountered a situation where they were actually what I wanted. I'm not a big fan of coercing dimensions in the first place, for the usual "refuse to guess" reasons. And then generally if I do want to coerce an array to another dimension, then I have some opinion about where the new dimensions should go, and/or I have some opinion about the minimum acceptable starting dimension, and/or I have a maximum dimension in mind. (E.g. "coerce 1d inputs into a column matrix; 0d or 3d inputs are an error" -- atleast_2d is zero-for-three on that requirements list.)</p>
<p dir="ltr">I don't know how typical I am in this. But it does make me wonder if the atleast_* functions act as an attractive nuisance, where new users take their presence as an implicit recommendation that they are actually a useful thing to reach for, even though they... aren't that. And maybe we should be recommending folk move away from them rather than trying to extend them further?</p>
<p dir="ltr">Or maybe they're totally useful and I'm just missing it. What's your use case that motivates atleast_nd?</p><span><font color="#888888"></font></span>
</blockquote></div></span>I think you're just missing it:) atleast_1d/2d are used quite a bit in Scipy and Statsmodels (those are the only ones I checked), and in the large majority of cases it's the best thing to use there. There's a bunch of atleast_2d calls with a transpose appended because the input needs to be treated as columns instead of rows, but that's still efficient and readable enough.<br></div></div></blockquote><div><br></div><div><br></div></span><div>As Ralph pointed out its usage in statsmodels. I do find them useful as replacement for several lines of ifs and reshapes</div><div><br></div><div>We stilll need in many cases the atleast_2d_cols, that appends the newaxis if necessary.</div><div><br></div><div>roughly the equivalent of</div><div><br></div><div>if x.ndim == 1:</div><div>    x = x[:, None]</div><div>else:</div><div>    x = np.atleast_2d(x)</div><span class="HOEnZb"><font color="#888888"><div><br></div><div>Josef</div><div> </div></font></span><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex"><span class=""><div dir="ltr"><div class="gmail_extra"><br></div><div class="gmail_extra">For 3D/nD I can see that you'd need more control over where the dimensions go, but 1D/2D are fine.</div></div></span></blockquote></div></div></div></blockquote><div><br></div><div><br></div><div>statsmodels has currently very little code with ndim >2, so I have no overview of possible use cases, but it would be necessary to have full control over the added axis since axis have a strict meaning and stats still prefer Fortran order to default numpy/C ordering.</div><div><br></div><div>Josef</div><div><br></div><div> </div><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex"><div dir="ltr"><div class="gmail_extra"><div class="gmail_quote"><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex"><span class=""><div dir="ltr"><div class="gmail_extra"><span><font color="#888888"><br><br></font></span></div><span><font color="#888888"><div class="gmail_extra">Ralf<br><br></div></font></span></div>
<br></span><span class="">_______________________________________________<br>
NumPy-Discussion mailing list<br>
<a href="mailto:NumPy-Discussion@scipy.org" target="_blank">NumPy-Discussion@scipy.org</a><br>
<a href="https://mail.scipy.org/mailman/listinfo/numpy-discussion" rel="noreferrer" target="_blank">https://mail.scipy.org/mailman/listinfo/numpy-discussion</a><br>
<br></span></blockquote></div><br></div></div>
</blockquote></div><br></div></div>