digitalmars.D - Workaround for DIP 1005
- Dominikus Dittes Scherkl (21/21) Feb 03 2017 DIP 1005 provides new syntax to make it possible to avoid global
- rikki cattermole (12/32) Feb 03 2017 Needless syntax sugar:
- Dominikus Dittes Scherkl (7/13) Feb 03 2017 ?!?
- Daniel N (17/20) Feb 03 2017 I like it!
- Andrei Alexandrescu (6/24) Feb 03 2017 Wow. This is... brilliant. Thanks for the great idea. I ran a few tests
- Andrei Alexandrescu (3/33) Feb 03 2017 I should also extend thanks to Dominik, who provided the initial idea
- Daniel Nielsen (6/11) Feb 03 2017 Thank you very much. I was just going to mail you directly to
- Daniel Nielsen (12/23) Feb 03 2017 I just had to try one more thing. It never ceases to amaze me
- crimaniak (3/6) Feb 04 2017 ... because of string mixins.
- Stefan Koch (4/13) Feb 04 2017 please refrain from insulting string-mixins.
- crimaniak (6/9) Feb 04 2017 Most understandable don't mean "good to use". We can do many
- Walter Bright (8/36) Feb 03 2017 I agree, it's pretty dazz! We need to give this technique a memorable na...
- =?UTF-8?Q?Ali_=c3=87ehreli?= (6/9) Feb 03 2017 If it's going to stay "imp" (which I initially found confusing; How
- Nick Sabalausky (3/5) Feb 03 2017 But they have much significance in Doom :)
- Andrei Alexandrescu (2/11) Feb 04 2017 https://github.com/dlang/druntime/pull/1756 -- Andrei
- deadalnix (2/11) Feb 03 2017 That's obviously a self important lookup.
- David Gileadi (2/16) Feb 04 2017 This. So much this.
- deadalnix (2/4) Feb 07 2017 I'm afraid you are the only one who appreciate my humor :)
- =?UTF-8?Q?Ali_=c3=87ehreli?= (4/9) Feb 13 2017 "self-important" is official now:
- Andrei Alexandrescu (2/3) Feb 04 2017 That would be indeed very very useful. -- Andrei
- Daniel Nielsen (6/10) Feb 05 2017 Okay, I will. However I currently have some obligations which
- Dominikus Dittes Scherkl (9/34) Feb 06 2017 Thank you all that you like the idea.
- Mike Parker (9/10) Feb 06 2017 IMO, as something targeted at non-D users, yes. A brief overview
- Dominikus Dittes Scherkl (15/24) Feb 06 2017 Oh yeah? For bragging about how D uses modules and doesn't need
- Mike Parker (8/17) Feb 06 2017 No, that's not quite my intent. There's an interesting story here
- Walter Bright (3/4) Feb 06 2017 It's ideally suited for an article. It's easy to grasp, and enables a ve...
- TheGag96 (5/9) Feb 06 2017 I personally think the idiom is neato and think it'd be a neat
- Andrea Fontana (15/35) Feb 07 2017 I don't understand why we can't use a template like:
- Jack Stouffer (7/21) Feb 07 2017 Because Systime's are used in the signiture, a local import
- Jack Stouffer (3/4) Feb 07 2017 I think I misunderstood your comment. Please forgive the noise.
- Chris Wright (16/17) Feb 07 2017 You can. However, that makes things awkward when you want to pass that a...
- Walter Bright (3/16) Feb 07 2017 It's an interesting idea, but it will have problems with overloading, as...
- Chris Wright (4/10) Feb 03 2017 What compiler version is this? I'm getting segfaults with both 2.072.2 a...
- =?UTF-8?Q?Ali_=c3=87ehreli?= (11/12) Feb 03 2017 I haven't followed DIP 1005 closely. Has there been any discussion of
- Walter Bright (3/8) Feb 03 2017 How does the compiler know that "std" is supposed to be an import rather...
- crimaniak (6/18) Feb 04 2017 std.datetime exists.
- Jonathan M Davis via Digitalmars-d (58/59) Feb 08 2017 This is really cool, but I have a couple of concerns about this and how ...
- Daniel N (18/29) Feb 09 2017 Thank you for your insightful feedback, I'm currently pressed for
- Dominikus Dittes Scherkl (13/28) Feb 10 2017 Of course. This is why I called it a "workaround".
- Dominikus Dittes Scherkl (13/20) Feb 10 2017 On Friday, 10 February 2017 at 13:28:43 UTC, Dominikus Dittes
- Nick Treleaven (19/33) Feb 10 2017 That's a problem with UFCS in general. Workaround:
- Meta (26/93) Feb 14 2017 It's a hack on top of a hack, but you can do something like this:
- John Colvin (7/9) Feb 14 2017 There is of course the middle way: with(from!q{std.datetime})
DIP 1005 provides new syntax to make it possible to avoid global imports. Till now global imports are necessary if a function uses types declared in some imported module within it's declaration or definition (otherwise a local import will do). But this can already be worked around with some nice trick: import someModule.SomeType; SomeType fun() { ... } can be replaced by: fun.ST fun() { import someModule.SomeType; alias ST = SomeType; ... } The same strategy works with types used in parameters or constraints, of course. Any thoughts?
Feb 03 2017
On 04/02/2017 3:43 AM, Dominikus Dittes Scherkl wrote:DIP 1005 provides new syntax to make it possible to avoid global imports. Till now global imports are necessary if a function uses types declared in some imported module within it's declaration or definition (otherwise a local import will do). But this can already be worked around with some nice trick: import someModule.SomeType; SomeType fun() { ... } can be replaced by: fun.ST fun() { import someModule.SomeType; alias ST = SomeType; ... } The same strategy works with types used in parameters or constraints, of course. Any thoughts?Needless syntax sugar: struct Foo { int x; } Foo func() { return Foo(9); } typeof(func()) func2() { return func(); } void main() { assert(func2.x == 9); }
Feb 03 2017
On Friday, 3 February 2017 at 14:59:09 UTC, rikki cattermole wrote:On 04/02/2017 3:43 AM, Dominikus Dittes Scherkl wrote:?!? What is syntactic sugar? I did NOT propose any new syntax - my workaround already works with the current D language. In fact I think DIP1005 is syntactic sugar (maybe even "needless").DIP 1005 provides new syntax to make it possible to avoid global imports. But this can already be worked around with some nice trick: Any thoughts?Needless syntax sugar:
Feb 03 2017
On Friday, 3 February 2017 at 14:43:01 UTC, Dominikus Dittes Scherkl wrote:DIP 1005 provides new syntax to make it possible to avoid global imports. Any thoughts?I like it! template imp(string mod) { mixin("import imp = " ~ mod ~ ";"); } auto fun_time(imp!"std.datetime".SysTime tm) { return tm; } void main() { import std.stdio; import std.datetime; fun_time(Clock.currTime()).writeln; }
Feb 03 2017
On 2/3/17 10:41 AM, Daniel N wrote:On Friday, 3 February 2017 at 14:43:01 UTC, Dominikus Dittes Scherkl wrote:Wow. This is... brilliant. Thanks for the great idea. I ran a few tests and it seems to be doing out of the box most of what we want with DIP1005 with no language change at all. Congratulations! AndreiDIP 1005 provides new syntax to make it possible to avoid global imports. Any thoughts?I like it! template imp(string mod) { mixin("import imp = " ~ mod ~ ";"); } auto fun_time(imp!"std.datetime".SysTime tm) { return tm; } void main() { import std.stdio; import std.datetime; fun_time(Clock.currTime()).writeln; }
Feb 03 2017
On 2/3/17 2:14 PM, Andrei Alexandrescu wrote:On 2/3/17 10:41 AM, Daniel N wrote:I should also extend thanks to Dominik, who provided the initial idea and the inspiration for this one. -- AndreiOn Friday, 3 February 2017 at 14:43:01 UTC, Dominikus Dittes Scherkl wrote:Wow. This is... brilliant. Thanks for the great idea. I ran a few tests and it seems to be doing out of the box most of what we want with DIP1005 with no language change at all. Congratulations!DIP 1005 provides new syntax to make it possible to avoid global imports. Any thoughts?I like it! template imp(string mod) { mixin("import imp = " ~ mod ~ ";"); } auto fun_time(imp!"std.datetime".SysTime tm) { return tm; } void main() { import std.stdio; import std.datetime; fun_time(Clock.currTime()).writeln; }
Feb 03 2017
On Friday, 3 February 2017 at 19:14:16 UTC, Andrei Alexandrescu wrote:Wow. This is... brilliant. Thanks for the great idea. I ran a few tests and it seems to be doing out of the box most of what we want with DIP1005 with no language change at all. Congratulations! AndreiThank you very much. I was just going to mail you directly to make sure the idea wasn't lost in the forum flood, will be exciting to see some measurements using this idiom. Daniel N
Feb 03 2017
On Friday, 3 February 2017 at 20:04:22 UTC, Daniel Nielsen wrote:On Friday, 3 February 2017 at 19:14:16 UTC, Andrei Alexandrescu wrote:I just had to try one more thing. It never ceases to amaze me what is possible in D today. template imp(string mod) { mixin("import imp = " ~ mod ~ ";"); } auto fun_time(imp!"std.datetime".SysTime tm) { return tm; } void main() { with(imp!"std.datetime") static if(is(Clock)) with(imp!"std.stdio") Clock.currTime.fun_time.writeln; }Wow. This is... brilliant. Thanks for the great idea. I ran a few tests and it seems to be doing out of the box most of what we want with DIP1005 with no language change at all. Congratulations! Andrei
Feb 03 2017
On Friday, 3 February 2017 at 21:23:22 UTC, Daniel Nielsen wrote:I just had to try one more thing. It never ceases to amaze me what is possible in D today.Yes, all except of good IDE and tooling.template imp(string mod) { mixin("import imp = " ~ mod ~ ";"); }... because of string mixins.
Feb 04 2017
On Saturday, 4 February 2017 at 11:50:50 UTC, crimaniak wrote:On Friday, 3 February 2017 at 21:23:22 UTC, Daniel Nielsen wrote:please refrain from insulting string-mixins. they are by far the most understandable of meta-programming facilities.I just had to try one more thing. It never ceases to amaze me what is possible in D today.Yes, all except of good IDE and tooling.template imp(string mod) { mixin("import imp = " ~ mod ~ ";"); }... because of string mixins.
Feb 04 2017
On Saturday, 4 February 2017 at 11:56:52 UTC, Stefan Koch wrote:please refrain from insulting string-mixins. they are by far the most understandable of meta-programming facilities.Most understandable don't mean "good to use". We can do many things with it but it's really hack and easy way to obscure code and prevent any automatic analysis of it. Because when we have string mixin in code - any tool should become a compiler to work with it. So my opinion: they must be avoided as much as possible.
Feb 04 2017
On 2/3/2017 11:14 AM, Andrei Alexandrescu wrote:On 2/3/17 10:41 AM, Daniel N wrote:I agree, it's pretty dazz! We need to give this technique a memorable name (not an acronym). I thought "Voldemort Types" turned out rather well, whereas CTFE is klunky, UFCS is even worse. The absolute worst is C++ SFINAE. Any ideas? Scherkl-Nielsen Lookup? The perfect bikeshedding moment! Daniel, Dominikus: please consider writing an article about this.On Friday, 3 February 2017 at 14:43:01 UTC, Dominikus Dittes Scherkl wrote:Wow. This is... brilliant. Thanks for the great idea. I ran a few tests and it seems to be doing out of the box most of what we want with DIP1005 with no language change at all. Congratulations!DIP 1005 provides new syntax to make it possible to avoid global imports. Any thoughts?I like it! template imp(string mod) { mixin("import imp = " ~ mod ~ ";"); } auto fun_time(imp!"std.datetime".SysTime tm) { return tm; } void main() { import std.stdio; import std.datetime; fun_time(Clock.currTime()).writeln; }
Feb 03 2017
On 02/03/2017 03:33 PM, Walter Bright wrote:On 2/3/17 10:41 AM, Daniel N wrote:auto fun_time(imp!"std.datetime".SysTime tm)We need to give this technique a memorable name (not an acronym).If it's going to stay "imp" (which I initially found confusing; How about "Import"?), "the imp idiom" might do but imps have a very insignificant appearance in Harry Potter. (There are "Pepper Imps sold at Honeydukes in Hogsmeade".) Ali
Feb 03 2017
On 02/03/2017 06:50 PM, Ali Çehreli wrote:"the imp idiom" might do but imps have a very insignificant appearance in Harry Potter.But they have much significance in Doom :) ...part of which takes place on Phobos ;)
Feb 03 2017
On 2/3/17 6:50 PM, Ali Çehreli wrote:On 02/03/2017 03:33 PM, Walter Bright wrote:https://github.com/dlang/druntime/pull/1756 -- AndreiOn 2/3/17 10:41 AM, Daniel N wrote:auto fun_time(imp!"std.datetime".SysTime tm)We need to give this technique a memorable name (not an acronym).If it's going to stay "imp" (which I initially found confusing; How about "Import"?), "the imp idiom" might do but imps have a very insignificant appearance in Harry Potter. (There are "Pepper Imps sold at Honeydukes in Hogsmeade".) Ali
Feb 04 2017
On Friday, 3 February 2017 at 23:33:58 UTC, Walter Bright wrote:I agree, it's pretty dazz! We need to give this technique a memorable name (not an acronym). I thought "Voldemort Types" turned out rather well, whereas CTFE is klunky, UFCS is even worse. The absolute worst is C++ SFINAE. Any ideas? Scherkl-Nielsen Lookup? The perfect bikeshedding moment! Daniel, Dominikus: please consider writing an article about this.That's obviously a self important lookup.
Feb 03 2017
On 2/3/17 5:00 PM, deadalnix wrote:On Friday, 3 February 2017 at 23:33:58 UTC, Walter Bright wrote:This. So much this.I agree, it's pretty dazz! We need to give this technique a memorable name (not an acronym). I thought "Voldemort Types" turned out rather well, whereas CTFE is klunky, UFCS is even worse. The absolute worst is C++ SFINAE. Any ideas? Scherkl-Nielsen Lookup? The perfect bikeshedding moment! Daniel, Dominikus: please consider writing an article about this.That's obviously a self important lookup.
Feb 04 2017
On Saturday, 4 February 2017 at 23:54:12 UTC, David Gileadi wrote:I'm afraid you are the only one who appreciate my humor :)That's obviously a self important lookup.This. So much this.
Feb 07 2017
On 02/07/2017 04:28 PM, deadalnix wrote:On Saturday, 4 February 2017 at 23:54:12 UTC, David Gileadi wrote:"self-important" is official now: https://dlang.org/blog/2017/02/13/a-new-import-idiom/ AliI'm afraid you are the only one who appreciate my humor :)That's obviously a self important lookup.This. So much this.
Feb 13 2017
On 2/3/17 6:33 PM, Walter Bright wrote:Daniel, Dominikus: please consider writing an article about this.That would be indeed very very useful. -- Andrei
Feb 04 2017
On Saturday, 4 February 2017 at 14:57:59 UTC, Andrei Alexandrescu wrote:On 2/3/17 6:33 PM, Walter Bright wrote:Okay, I will. However I currently have some obligations which demand my full attention, I estimate having a first draft ready by next weekend. DanielDaniel, Dominikus: please consider writing an article about this.That would be indeed very very useful. -- Andrei
Feb 05 2017
On Friday, 3 February 2017 at 23:33:58 UTC, Walter Bright wrote:On 2/3/2017 11:14 AM, Andrei Alexandrescu wrote:Thank you all that you like the idea. I found myself using this idiom quite some time now in my libraries but only recently realized, that it is the reason why I didn't found DIP1005 too compelling - I just didn't need it because of my workaround. So I thought I should share the idea - and Daniels extension makes it much easier to use. Have to update my libs with that :-) But is this really worth an article?On 2/3/17 10:41 AM, Daniel N wrote:I agree, it's pretty dazz! We need to give this technique a memorable name (not an acronym). I thought "Voldemort Types" turned out rather well, whereas CTFE is klunky, UFCS is even worse. The absolute worst is C++ SFINAE. Any ideas? Scherkl-Nielsen Lookup? The perfect bikeshedding moment! Daniel, Dominikus: please consider writing an article about this.On Friday, 3 February 2017 at 14:43:01 UTC, Dominikus Dittes Scherkl wrote:Wow. This is... brilliant. Thanks for the great idea. I ran a few tests and it seems to be doing out of the box most of what we want with DIP1005 with no language change at all. Congratulations!DIP 1005 provides new syntax to make it possible to avoid global imports. Any thoughts?I like it!
Feb 06 2017
On Monday, 6 February 2017 at 08:45:45 UTC, Dominikus Dittes Scherkl wrote:But is this really worth an article?IMO, as something targeted at non-D users, yes. A brief overview of D's module system, a description of the problem & how the DIP was brought up to address it, how the DIP was made irrelevant by the usage of features every D programmer knows by so many failed to realize could be used in this way... lots of meat for a post there. I'd be happy to publish something like that on the D Blog.
Feb 06 2017
On Monday, 6 February 2017 at 09:00:43 UTC, Mike Parker wrote:On Monday, 6 February 2017 at 08:45:45 UTC, Dominikus Dittes Scherkl wrote:Oh yeah? For bragging about how D uses modules and doesn't need global imports at all? This is not my style and I didn't feel that not needing global imports is something to brag about - they are useful as an overview. But with this new idiom .di-files are even more useless now - or really need to declare all the local imports to replace the lost overview.But is this really worth an article?IMO, as something targeted at non-D users, yes.A brief overview of D's module system, a description of the problem & how the DIP was brought up to address it,But I'm not an expert for this. Especially I was not aware of any problem and even didn't liked the DIP.how the DIP was made irrelevant by the usage of features every D programmer knows but so many failed to realize could be used in this way...Ok, this is the most interesting part. This is what having an idea is all about. Find new ways to put the things already there together. But hard to describe. Maybe I'll give it a try.
Feb 06 2017
On Monday, 6 February 2017 at 09:39:25 UTC, Dominikus Dittes Scherkl wrote:On Monday, 6 February 2017 at 09:00:43 UTC, Mike Parker wrote:No, that's not quite my intent. There's an interesting story here beyond the feature itself that I believe makes for a good blog post. The majority of visitors to the blog are not D users and aren't going to be familiar with the details (like the different ways a module can be imported), so such an article needs to keep them in mind.On Monday, 6 February 2017 at 08:45:45 UTC, Dominikus Dittes Scherkl wrote:Oh yeah? For bragging about how D uses modules and doesn't need global imports at all?But is this really worth an article?IMO, as something targeted at non-D users, yes.
Feb 06 2017
On 2/6/2017 12:45 AM, Dominikus Dittes Scherkl wrote:But is this really worth an article?It's ideally suited for an article. It's easy to grasp, and enables a very interesting idiom.
Feb 06 2017
On Monday, 6 February 2017 at 11:03:32 UTC, Walter Bright wrote:On 2/6/2017 12:45 AM, Dominikus Dittes Scherkl wrote:I personally think the idiom is neato and think it'd be a neat article! It could potentially inspire wonder in some non-D-users about the limitless possibilities of the language... Or scare some away. :V The idiom would pretty bizarre for newcomers lol.But is this really worth an article?It's ideally suited for an article. It's easy to grasp, and enables a very interesting idiom.
Feb 06 2017
On Friday, 3 February 2017 at 15:41:56 UTC, Daniel N wrote:On Friday, 3 February 2017 at 14:43:01 UTC, Dominikus Dittes Scherkl wrote:I don't understand why we can't use a template like: auto fun_time(SysTime)(SysTime tm) { import std.datetime; static assert (is(SysTime == std.datetime.SysTime)); return tm; } void main() { import std.stdio; import std.datetime; fun_time(Clock.currTime()).writeln; } I think I missed something.DIP 1005 provides new syntax to make it possible to avoid global imports. Any thoughts?I like it! template imp(string mod) { mixin("import imp = " ~ mod ~ ";"); } auto fun_time(imp!"std.datetime".SysTime tm) { return tm; } void main() { import std.stdio; import std.datetime; fun_time(Clock.currTime()).writeln; }
Feb 07 2017
On Tuesday, 7 February 2017 at 15:00:17 UTC, Andrea Fontana wrote:I don't understand why we can't use a template like: auto fun_time(SysTime)(SysTime tm) { import std.datetime; static assert (is(SysTime == std.datetime.SysTime)); return tm; } void main() { import std.stdio; import std.datetime; fun_time(Clock.currTime()).writeln; } I think I missed something.Because Systime's are used in the signiture, a local import cannot be used. Local imports are only evaluated if the function is used in the program and are local to that scope. Therefore, when the signitures are being evaluated, the compiler has no information about std.datetime, and therefore fails with a symbol not found error.
Feb 07 2017
On Tuesday, 7 February 2017 at 15:00:17 UTC, Andrea Fontana wrote:...I think I misunderstood your comment. Please forgive the noise. I need more coffee
Feb 07 2017
On Tue, 07 Feb 2017 15:00:17 +0000, Andrea Fontana wrote:I don't understand why we can't use a template like:You can. However, that makes things awkward when you want to pass that as a delegate. It makes it awkward to read. You don't get to specify the return type unless it happens to be the same as a parameter type. A near equivalent is: template extractDate() { import std.datetime; Date extractDate(SysTime time) { return time.date; } } Better type checking, but you need to refer to it as `extractDate!()` sometimes. That's still better than having to refer to it as `extractDate! (SysTime)`.
Feb 07 2017
On 2/7/2017 7:00 AM, Andrea Fontana wrote:I don't understand why we can't use a template like: auto fun_time(SysTime)(SysTime tm) { import std.datetime; static assert (is(SysTime == std.datetime.SysTime)); return tm; } void main() { import std.stdio; import std.datetime; fun_time(Clock.currTime()).writeln; }It's an interesting idea, but it will have problems with overloading, as the template selection will match on everything.
Feb 07 2017
On Fri, 03 Feb 2017 14:43:01 +0000, Dominikus Dittes Scherkl wrote:fun.ST fun() { import someModule.SomeType; alias ST = SomeType; ... }What compiler version is this? I'm getting segfaults with both 2.072.2 and 2.073.0. Reported as https://issues.dlang.org/show_bug.cgi?id=17140
Feb 03 2017
On 02/03/2017 06:43 AM, Dominikus Dittes Scherkl wrote:DIP 1005 provides new syntax to make it possible to avoid global imports.I haven't followed DIP 1005 closely. Has there been any discussion of automatic imports that seems to have potential to make both the user's and implementor's lives easier: void foo(std.datetime.Duration dur) { // ... } Can the compiler automatically import std.datetime for both the implementation above and the user? Or, too much beer at work on this Friday? :o) Ali
Feb 03 2017
On 2/3/2017 4:01 PM, Ali Çehreli wrote:void foo(std.datetime.Duration dur) { // ... } Can the compiler automatically import std.datetime for both the implementation above and the user? Or, too much beer at work on this Friday? :o)How does the compiler know that "std" is supposed to be an import rather than a misspelling of a previously declared "stc" ?
Feb 03 2017
On Saturday, 4 February 2017 at 00:43:14 UTC, Walter Bright wrote:On 2/3/2017 4:01 PM, Ali Çehreli wrote:std.datetime exists. I think, allowing fully qualified names and automatic importing of used fully qualified names will be much, much better, than introduction of string mixins literally everywhere. Woody_and_Buzz_jpg!"String mixins! They are everywhere!"void foo(std.datetime.Duration dur) { // ... } Can the compiler automatically import std.datetime for both the implementation above and the user? Or, too much beer at work on this Friday? :o)How does the compiler know that "std" is supposed to be an import rather than a misspelling of a previously declared "stc" ?
Feb 04 2017
On Friday, February 03, 2017 14:43:01 Dominikus Dittes Scherkl via Digitalmars-d wrote:Any thoughts?This is really cool, but I have a couple of concerns about this and how it seems deficient in comparison to DIP 1005. I mentioned it in Andrei's PR for this, but no one has responded to my comment. This first problem is UFCS. While this technique works great for parameters or on types you want to list in the template constraint, I don't see how it can work with UFCS. So, if you have something like auto func(alias pred, R)(R range) if(isInputRange!R && is(typeof(pred(range.front)) == bool)) {...} you can do auto func(alias pred, R)(R range) if(from!"std.range.primitives".isInputRange!R && is(typeof(pred(range.front)) == bool)) {...} but you can't import std.range.primitives.front to make dynamic arrays work, whereas with DIP 1005, you can just add the import to the function declaration, and it will work without having to tie the import to a specific symbol. And while in many cases, you can just forgo UFCS - e.g. auto func(P)(P param) if(is(typeof(from!"std.algorithm".find(param)) == param)) {...} that not only doesn't work with the range primitives, but it's bad practice to use UFCS in the function body and not the template constraint (since the semantics may not be the same), meaning that not using UFCS in the template constraint means not using it in the function body for anything that's in the template constraint (which many folks won't like and others won't understand, resulting in subtle bugs), and any case where you want a function to work with types that define a function as a member function as well as with types that use a free function, you need to use UFCS and thus cannot choose to not use it in the template constraint. So, unless there's something that I don't understand about this technique (which is definitely possible), it seems like it does not work with UFCS and thus makes it considerably worse than 1005 for a lot of templated code, much as it would work great for code that doesn't need UFCS. The other problem is how much more verbose it is. With DIP 1005, you can do with(import std.datetime) auto foo(SysTime st1, SysTime st2, Duration d); The import is only listed once, whereas with this technique, you have to list it for each symbol. e.g. auto foo(from!"std.datetime".SysTime st1, from!"std.datetime".SysTime st2, from!"std.datetime".Duration d); The result is much more verbose, and if you have several symbols that need imports between the return type, parameters, and template constraint, you quickly end up with a lot of extra text in the middle of your function signatures just because you want to tie the imports to the functions that use them. With DIP 1005, the imports are next to the function but separate where they avoid the need for repeating imports and don't get mixed into the middle of the function signature. So, while the proposed technique is really cool and clever in what it lets us do without actually altering the language, it seems like it's quite a bit worse than DIP 1005. As such, I'm inclined to argue that we should favor DIP 1005 over this proposal, as cool as it is. - Jonathan M Davis
Feb 08 2017
On Thursday, 9 February 2017 at 05:40:01 UTC, Jonathan M Davis wrote:The import is only listed once, whereas with this technique, you have to list it for each symbol. e.g. auto foo(from!"std.datetime".SysTime st1, from!"std.datetime".SysTime st2, from!"std.datetime".Duration d); The result is much more verbose, and if you have several symbols that need imports between the return type, parameters, and template constraint, you quickly end up with a lot of extra text in the middle of your function signatures just because you want to tie the imports to the functions that use them. - Jonathan M DavisThank you for your insightful feedback, I'm currently pressed for time but I can at least begin to address one of your concerns. Every proposed solution has some sort of tradeoff, verbosity could be dealt with for the price of turning a simple function into a template. In some cases this could be acceptable in others prohibitive. auto foo(alias dt = from!"std.datetime") (dt.SysTime st1, dt.SysTime st2, dt.Duration d) Our "Implicit Function-Template Instantiation" also unfortunately bails with default arguments... maybe there is an enhancement request for that already? (from!"std.datetime") almost works too(it would avoid turning the function into a template), but has other issues.
Feb 09 2017
On Thursday, 9 February 2017 at 05:40:01 UTC, Jonathan M Davis wrote:On Friday, February 03, 2017 14:43:01 Dominikus Dittes Scherkl via Digitalmars-d wrote:Of course. This is why I called it a "workaround". But in fact I would prefer to only use local imports instead of some impoting template. But with this workaround in mind, the implementation of DIP1005 would be much easier (see below).Any thoughts?This is really cool, but I have a couple of concerns about this and how it seems deficient in comparison to DIP 1005.[...] The other problem is how much more verbose it is. With DIP 1005, you can do with(import std.datetime) auto foo(SysTime st1, SysTime st2, Duration d); The import is only listed once, whereas with this technique, you have to list it for each symbol. e.g. auto foo(from!"std.datetime".SysTime st1, from!"std.datetime".SysTime st2, from!"std.datetime".Duration d);With my original proposal you would write auto foo(foo.M.SysTime st1, foo.M.SysTime st2, foo.M.Duration d) { import std.datetime; }
Feb 10 2017
On Friday, 10 February 2017 at 13:28:43 UTC, Dominikus Dittes Scherkl wrote: Sorry, accidentally hit some key-combination that immediately send the message that was not yet complete.With my original proposal you would write auto foo(foo.SysTime st1, foo.SysTime st2, foo.Duration d) { import std.datetime;alias SysTime = std.datetime.SysTime; alias Duration = std.datetime.Duration;}And the implementation of DIP1005 would only "create this alias automatically", so that you would write what you always did - if any type in a declaration is not found, the compiler tries to find it in the function definition and use that instead. Therefore I would call this "auto aliasing" It does not inquire a (visible) language change, only an enhanced lookup within the compiler.
Feb 10 2017
On Thursday, 9 February 2017 at 05:40:01 UTC, Jonathan M Davis wrote:auto func(alias pred, R)(R range) if(from!"std.range.primitives".isInputRange!R && is(typeof(pred(range.front)) == bool)) {...} but you can't import std.range.primitives.front to make dynamic arrays work, whereas with DIP 1005, you can just add the import to the function declaration, and it will work without having to tie the import to a specific symbol.That's a problem with UFCS in general. Workaround: import std.functional; import std.range; void main(){ auto r = [1]; auto i = r.unaryFun!(std.range.front); assert(i == 1); } As an enhancement, it seems supporting value.(callable) would work: range.(std.range.front) That would be generally useful: // print reciprocal of reduced range range.reduce!someFunc.(n => 1 / n).writeln;The other problem is how much more verbose it is. With DIP 1005, you can do with(import std.datetime) auto foo(SysTime st1, SysTime st2, Duration d); The import is only listed once, whereas with this technique, you have to list it for each symbol.That's a problem only when very few of a module's declarations need more than one symbol from the same module. How often does that occur when a module-level import should be avoided?
Feb 10 2017
On Thursday, 9 February 2017 at 05:40:01 UTC, Jonathan M Davis wrote:On Friday, February 03, 2017 14:43:01 Dominikus Dittes Scherkl via Digitalmars-d wrote:It's a hack on top of a hack, but you can do something like this: void test(T1, T2, alias isIntegral = from!"std.traits".isIntegral,(T1 t1, T2 t2) if (isIntegral!T1 && isIntegral!T2) { pragma(msg, isIntegral!T1); pragma(msg, isIntegral!T2); } void main() { } The only problem is that it allows a user to supply their own value for isIntegral, which can lead to some weird error messages and confusion if they don't know about this idiom. Also it can't quite do UFCS, because you cannot use UFCS with local symbols: void test(T1, T2, alias isIntegral = from!"std.traits".isIntegral, alias chop = from!"std.string".chop) (T1 t1, T2 t2) if (isIntegral!T1 && isIntegral!T2 && T1.stringof.chop() == "ubyt") //Error: no property 'chop' for type 'string' { }Any thoughts?This is really cool, but I have a couple of concerns about this and how it seems deficient in comparison to DIP 1005. I mentioned it in Andrei's PR for this, but no one has responded to my comment. This first problem is UFCS. While this technique works great for parameters or on types you want to list in the template constraint, I don't see how it can work with UFCS. So, if you have something like auto func(alias pred, R)(R range) if(isInputRange!R && is(typeof(pred(range.front)) == bool)) {...} you can do auto func(alias pred, R)(R range) if(from!"std.range.primitives".isInputRange!R && is(typeof(pred(range.front)) == bool)) {...} but you can't import std.range.primitives.front to make dynamic arrays work, whereas with DIP 1005, you can just add the import to the function declaration, and it will work without having to tie the import to a specific symbol. And while in many cases, you can just forgo UFCS - e.g. auto func(P)(P param) if(is(typeof(from!"std.algorithm".find(param)) == param)) {...} that not only doesn't work with the range primitives, but it's bad practice to use UFCS in the function body and not the template constraint (since the semantics may not be the same), meaning that not using UFCS in the template constraint means not using it in the function body for anything that's in the template constraint (which many folks won't like and others won't understand, resulting in subtle bugs), and any case where you want a function to work with types that define a function as a member function as well as with types that use a free function, you need to use UFCS and thus cannot choose to not use it in the template constraint. So, unless there's something that I don't understand about this technique (which is definitely possible), it seems like it does not work with UFCS and thus makes it considerably worse than 1005 for a lot of templated code, much as it would work great for code that doesn't need UFCS. The other problem is how much more verbose it is. With DIP 1005, you can do with(import std.datetime) auto foo(SysTime st1, SysTime st2, Duration d); The import is only listed once, whereas with this technique, you have to list it for each symbol. e.g. auto foo(from!"std.datetime".SysTime st1, from!"std.datetime".SysTime st2, from!"std.datetime".Duration d); The result is much more verbose, and if you have several symbols that need imports between the return type, parameters, and template constraint, you quickly end up with a lot of extra text in the middle of your function signatures just because you want to tie the imports to the functions that use them. With DIP 1005, the imports are next to the function but separate where they avoid the need for repeating imports and don't get mixed into the middle of the function signature. So, while the proposed technique is really cool and clever in what it lets us do without actually altering the language, it seems like it's quite a bit worse than DIP 1005. As such, I'm inclined to argue that we should favor DIP 1005 over this proposal, as cool as it is. - Jonathan M Davis
Feb 14 2017
On Thursday, 9 February 2017 at 05:40:01 UTC, Jonathan M Davis wrote:with(import std.datetime) auto foo(SysTime st1, SysTime st2, Duration d);There is of course the middle way: with(from!q{std.datetime}) I would love to be able to use `with` and have the language accept it in a wider range of places (e.g. module scope, on expressions), but in its current form it's a statement and introduces a scope, the inconsistency would be unpleasant.
Feb 14 2017