digitalmars.D - Alias template parameters and runtime functions
- Peter Alexander <peter.alexander.au gmail.com> Oct 03 2010
- bearophile <bearophileHUGS lycos.com> Oct 03 2010
- Peter Alexander <peter.alexander.au gmail.com> Oct 03 2010
This post has two questions:
1. What are the semantics of 'alias' template parameters when initialised using
runtime functions,
e.g.:
class Foo
{
int[] data;
int foo(int i) { return data[i]; }
auto allFoo()
{
auto fun = (int i) { return foo(i); };
// alternatively: auto fun = &this.foo; (same result)
return map!(fun)(iota(data.length));
}
}
This compiles, but gives you an access violation when you try to use it. Alias
is a compile time
construct (if I am not mistaken), so why does the compiler allow this?
2. How can I achieve what allFoo() above is trying to do? Obviously I could
just return data in this
trivial case, but imagine that foo(int) did something more complex, and I
didn't want to repeat the
logic of foo(int) in allFoo().
Thanks.
Oct 03 2010
Peter Alexander:class Foo { int[] data; int foo(int i) { return data[i]; } auto allFoo() { auto fun = (int i) { return foo(i); }; // alternatively: auto fun = &this.foo; (same result) return map!(fun)(iota(data.length)); } } This compiles, but gives you an access violation when you try to use it.
But that code works with dmd 2.049: import std.algorithm: map, equal; import std.range: iota; class Foo { int[] data; this(int[] a) { this.data = a.dup; } int foo(int i) { return data[i] * 2; } auto allFoo1() { auto fun = (int i) { return foo(i); }; return map!(fun)(iota(data.length)); } auto allFoo2() { return map!((int i){ return foo(i); })(iota(data.length)); } } void main() { auto f = new Foo([1, 2, 3]); assert(equal(f.allFoo1(), [2, 4, 6])); assert(equal(f.allFoo2(), [2, 4, 6])); } Bye, bearophile
Oct 03 2010
== Quote from bearophile (bearophileHUGS lycos.com)'s articleBut that code works with dmd 2.049: ... Bye, bearophile
You're right! That makes things even more curious then. I had noticed the problem in some more complex code and just assumed it would be the same in the simple code that I presented. That said, I have managed to create a slightly more complex example that *does* break. import std.algorithm: map, equal; import std.range: iota; import std.stdio; struct Bar { float[1] m; this(float a) { m = [a]; } } class Foo { Bar[1] m; this(Bar a) { m = [a]; } const foo(int i) { return m[i]; } const allFoo1() { auto fun = (int i) { return foo(i); }; return map!(fun)(iota(m.length)); } const allFoo2() { return map!((int i){ return foo(i); })(iota(m.length)); } } void main() { Foo f = new Foo(Bar(1)); assert(equal(f.allFoo1(), [Bar(1)])); assert(equal(f.allFoo2(), [Bar(1)])); }
Oct 03 2010








Peter Alexander <peter.alexander.au gmail.com>