digitalmars.D.learn - built-in int[] opSliceOpAssign throws?
- monarch_dodra (29/29) Sep 12 2012 --------
- Namespace (28/28) Sep 12 2012 You are right, slice isn't nothrow, this should may be fixed.
- monarch_dodra (19/47) Sep 12 2012 I think I foun out what is going on: It is a problem with
- Jonathan M Davis (11/16) Sep 12 2012 I believe that dup and idup have the same problem, and it definitely nee...
--------
nothrow void foo1(int[] a)
{
foreach(i; 0..10)
{
a[i] = 5;
a[i] += 5;
}
}
nothrow void foo2(int[] a) //10
{
a[0..10] = 5; //12
a[0..10] += 5; //13
}
--------
main.d(13): Error: _arrayExpSliceAddass_i is not nothrow
main.d(10): Error: function main.foo2 'foo2' is nothrow yet may
throw
--------
Making an out of range call in foo2 (after removing nothrow)
throws an _ERROR_ in both line 12 and 13 (if commenting 12):
core.exception.RangeError main(12): Range violation
core.exception.RangeError main(13): Range violation
Is there any situation where it actually *could* throw an
exception?
This is a bug? I did not find any entries in Bug report. Should I
report this?
I need to make the above call in a nothrow function. Is there any
way to make it work, without any run-time cost?
Sep 12 2012
You are right, slice isn't nothrow, this should may be fixed.
But if you don't assign your dynamic array first, you have a
problem: you cannot put elements in a empty dynamic array with
arr[i] = val;, you have to use arr ~= val;
This code works:
import std.stdio;
nothrow void foo1(ref int[] a)
{
foreach(i; 0..10)
{
a ~= 5;
a[i] += 5;
}
}
void foo2(ref int[] a) //10
{
a[] = 5; //12, no explicit slice, so the whole array is
assigned with 5
a[] += 7; //13
}
void main() {
int[] a;
foo1(a);
foo2(a);
writeln(a);
}
Sep 12 2012
On Wednesday, 12 September 2012 at 09:50:09 UTC, Namespace wrote:
You are right, slice isn't nothrow, this should may be fixed.
But if you don't assign your dynamic array first, you have a
problem: you cannot put elements in a empty dynamic array with
arr[i] = val;, you have to use arr ~= val;
This code works:
import std.stdio;
nothrow void foo1(ref int[] a)
{
foreach(i; 0..10)
{
a ~= 5;
a[i] += 5;
}
}
void foo2(ref int[] a) //10
{
a[] = 5; //12, no explicit slice, so the whole array is
assigned with 5
a[] += 7; //13
}
void main() {
int[] a;
foo1(a);
foo2(a);
writeln(a);
}
I think I foun out what is going on: It is a problem with
overlap. According to specs, overlapping arrays are illegal:
--------
void main()
{
int[10] b;
b[] = 5;
b[0..6] += b[4..10]; //6
b[0..6] = b[4..10]; //7
}
--------
Here, Line 7 will produce an
"object.Exception src\rt\arraycat.d(40): overlapping array copy".
However, line 6 will produce nothing. Toying with it shows that
it produces *unspecified* behavior.
So rephrasing my question in 2 questions:
1) Shouldn't "b[0..6] += b[4..10]" throw "something"?
2) Shouldn't the thrown object be an Error (and not an Exception)?
Sep 12 2012
On Wednesday, September 12, 2012 11:20:54 monarch_dodra wrote:This is a bug? I did not find any entries in Bug report. Should I report this? I need to make the above call in a nothrow function. Is there any way to make it work, without any run-time cost?I believe that dup and idup have the same problem, and it definitely needs to be fixed. Regardless, pretty much the only sane way to call a function that throws in a nothrow function is to put a try-catch around it. try func(); catch(Exception e) assert(0, "func threw. That's supposed to be impossible."); But if you do that, you'd better be sure that it really _can't_ throw (if it can, then you need to actually handle the exception rather than asserting 0). - Jonathan M Davis
Sep 12 2012









"monarch_dodra" <monarchdodra gmail.com> 