-
-
Notifications
You must be signed in to change notification settings - Fork 2.8k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Proposal: Interfaces #980
Comments
My intention is to write a std.meta package that will make things a lot smoother (functions like has_method(), has_self_method(), etc. etc.). Personally I'm not too keen on this proposal for two reasons:
Btw, for something quite similar (an interface implementation but in the runtime polymorphism sense), this is a proof of concept: https://gist.github.com/alexnask/1d39fbc01b42ce2b5b628828b6d1fb46 |
@alexnask, good work on your PR by the way; really what I needed for something I was doing :). Yeh I wasn't meaning for it to have runtime polymorphism, more like a way to have certain traits enforced onto your types, however I think handling it is much easier with the whole 'has_method' thing and so on :). So really this proposal falls apart with that idea of those functions existing. |
@BraedonWooding pub const Concept = struct {
pub const MethodType = enum {
Self,
ConstSelf,
Static,
Any,
};
pub const Rule = struct {
method_type: MethodType,
name: []const u8,
fn_type: type,
is_required: bool,
};
rules: []Rule,
};
const IteratorConcept = Concept {.rules = []Rule {
// Note how we ignore the self argument in the fn_type.
Rule { .method_type=Concept.MethodType.Self, .name="reset", .fn_type=fn()void, .is_required=false },
Rule { .method_type=Concept.MethodType.Self, .name="next", .fn_type=fn()i32, .is_required=true },
}};
pub fn matches_concept(comptime T: type, comptime concept: Concept) bool;
// Could use the std.meta functions instead, but this also checks the type's method matches the concept's
pub fn concept_has(comptime T: type, comptime concept: Concept, comptime name: []const u8) bool;
|
Other duck typing tools could also solve the problem. One is a test to see if a give bit of code compiles and the other is a conditional on the function signature:
With this you can build a concept test:
|
pub fn foo(comptime T: type, T: iter) if(isIterator(T)) void else unreachable {
Or rather the common pattern is something like; pub fn foo(comptime T: type, T: iter) void {
if (!isIterator()) {
@compileError("Class supplied does not have a '.reset' and '.next' method");
}
while (iter.next()) |val| {
....
}
....
} |
The idea of the conditional test on the function signature is to control the function matching. Basicly if T is not an iterater the function doesn't match nor compile, thus no function foo that takes an iterator. This opens up generic function overloading. I am not 100 on this syntax just the idea.
The @compiles is a very generic solution to test code constructs, ie covers the cases not covered by other comptime functions. typeOf and memberOf and others would probably be strong tests used. |
I love Dlang's compiles trait, however as far as I'm aware there are no plans for zig functions to directly read or generate bits of AST/unevaluated expressions. You should take a look at @typeinfo, it gives us access to all the information we need to create such tests and covers 99% of the cases where @compiles would be useful. Ofc, there still needs to be a nice wrapper over @typeinfo. std.meta will provide functions like has_field, has_child, has_method, has_self_method, is_slice_like, etc. |
A std.meta package would be great, but what about conditional compilation on function signature? It is a nice was to solve concepts without a clunky library solution or shoehorning it into the language. Something like this should be optional to use, enforced by the compiler, and easy to understand. This is one of the few things I think D got right. I have found suck typing and concepts are a better solution then interface. (I code a lot of "Enterprise" Java for work.. ) I am looking for a D replacement and hope zig is it, but there are a few language features I don't want to lose, strong generics/meta programming and concepts is one of them. |
@bheads By conditional compilation on function signature do you mean this example from above?:
If so, then I don't quite see the benefit.
With Zig's lazy analysis, only the block that is true is analyzed and compiled. The others are ignored by the compiler. |
With this function
You can no longer overload it for new types of T. If foo is in the std library then a user can no longer overload it. Also what happens when there are 3, 4, 10 different cases to handle? Functions like this can be come very large and unruly. but with this style:
a user or other lib can still add overloads, and the function only handles a single type case.
|
Function overloads aren't a thing in Zig (and editing an already built external API like std from your program is typically seen as bad practice as it invites all kind of bugs). I don't see them ever becoming a thing. I don't see how there is any difference between the two code examples you gave sorry. To me they have the same noise, there are other ways to accomplish the task, such as using structs, other functions to handle cases and so on; for example; fn foo(comptime T: type) void {
switch (getHandlingForThing(T)) {
Cases.IsThing => DoThing,
Cases.OtherThing => OtherThing;
else => @compileError("");
}
} Looks quite fine to me, you could even switch on the
What's a use case for 10 different cases? 3 or 4 would be the maximum, 5/6 would truly be rare, anything over that I couldn't see it being the best solution and that using another solution would be better. As with the other issue we need to see some 'real world cases', i.e. give us an example where it is needed (I struggle with this sometimes too, you get a really good idea but sadly it isn't really needed; despite it being a really good idea!!!). A good example for what we are looking for would be the following; Why I think we need switch cases in ZigSwitch cases are used universally across languages, with people often simulating them in languages like Python where they don't come 'naturally', they are very useful for things like emulators and drivers something that as low level as Zig should support; furthermore they also provide superior code to a series of if statements if the amount of cases is < 100 and are all sequential as it produces a lookup table, meaning that switching on enums often is much more efficient then checking them one by one.
Hopefully this helps :). |
No function overloading... thats archaic. Is there any posts on why not? I also noticed there are no default parameters. |
@bheads All rejected issues are here. I can't find a proposal specificly for function overloading, but related are #871, #427 and #148. For default parameters we have #484. I recommend reading through the rejected proposals. It'll give you an idea as to what kind of language Zig is. Just remember, things are rejected for good reason, and the userbase of Zig values the way Zig does things. |
@bheads 'archaic' is a point of contention; Zig has a few similarities to Go and not having function overloading is one of them, it is an archaic view like header files; there are valid reasons. From what I've gathered and my personal experience generally it seems that function overloading removes context in a lot of cases and generally when you have type ducking you cover the cases where it is very much needed; and in the other cases many people would say not to use it. Here are a few examples;
Default parameters is something that is a something you often get when having function overloads, I quite like them but again they remove information; those links that Hejsil (thanks) gave are really good to see some good counter arguments to that. I do potentially see them in the future though I don't what form they'll be in. |
Sorry if I come off strong or offensive, not my intentions at all. I just had nightmares of C and Java Enterprise code I deal with day to day that would be solved with function overloads and optional params. But I think duck typing with good comptime reflection, a std.meta library, along with guidelines and examples would remove the need for an interface in the language or STD lib. |
I like the idea of enforcing type constraints, but I really think stronger wording than "interface" or "trait" should be used, because there is a ton of overloaded language around this stuff. "interface" strictly means runtime dynamic types in some languages, for example. "trait" is even worse, because it can mean runtime OR compile-time dynamic type in Rust. The C++ term "concept" is pretty good because it is quite distinct and different, and well-specified, for example. I also think one of the things Go gets right about this is that the implementing type doesn't need to explicitly declare that it implements some "trait" or "interface"; this responsibility is in the caller, instead. So I could define my own interface that some stdlib type happens to implement, and use that type in my code, without modifying the stdlib type. This is the benefit of "duck typing". |
I don't think this is needed anymore, I agree @binary132 but with the latest changes to reflection it is quite easy to get this kind of behaviour :). Closing issue |
Is now everyone expected to roll his own interface implementation? I think this issue really should be solved uniformly otherwise this is just a big mess and I do not think basic features should be bolted onto zig with reflection hacks 👀 |
@monouser7dig How would an |
If it's not in zig, everyone will do it differently and when reading someone else code you/ (at least I) get confused very easily. |
I think reflection using indicates runtime reflection, but I imagine you are referring to the compile-time type introspection offered in Zig. Can you please provide an example @BraedonWooding. |
Currently duck typing is great, however there are a few issues with duck typing and these include the fact that the errors you get aren't always great, so I propose we add 'interfaces' which look very similar to structs however what they are meant for is as a supplied type in functions or wherever duck typing can be used.
I think an example is the best way to show it;
As you can see it is just duck typing but the key difference is in the use of it, and in the compile time errors you get! If you call it with an iterator that doesn't have a 'next' function then the compiler will give you a nice error; "<STRUCT_NAME> struct is missing required function/s: 'fn next(self: &Self) i32'". Another nice thing about this system is that it is nicer with optional requirements.
You can't currently do this at all, however with the new
@typeInfo
it'll be possible to simulate this from what I can see, however in doing it, you end up with much uglier and obfuscated code than something like this;Motivation
I've been building out a LINQ library in Zig (called Lazy, because well it is all about doing lazily executed queries on data structures, all around 'yield'ing getting values when you need them which remove the need for allocators completely in some cases and can provide a nice speed boost), and on of the annoyances was trying to get it to work with hash_maps and array lists using an .iterator that is provided by those things. Currently it isn't solvable to do a generic call like
Lazy.init(myObj)
and have it figure out what to do since it can't detect if functions exist, having something like above, would simplify a lot of my structures and would make errors a whole lot nicer.On a side note if you are interested in the idea of the library I've got a simple example version on my github here. 😄.
The text was updated successfully, but these errors were encountered: