Difference between revisions of "User:Enamex/DIP: Extension Interfaces"
m |
|||
Line 44: | Line 44: | ||
''This proposal allows for disciplined definition of'' '''extension functions''' ''(as is provided today via UFCS) that'' '''obey namespacing rules''', ''are available to call wherever the name of a supported type is passed to a template'' ('''locally, they bind to the implementing type's name; are available wherever it is'''), ''and'' '''simplify using D constraints while improving the situation of extending <code>structs</code> without inheritance woes'''. | ''This proposal allows for disciplined definition of'' '''extension functions''' ''(as is provided today via UFCS) that'' '''obey namespacing rules''', ''are available to call wherever the name of a supported type is passed to a template'' ('''locally, they bind to the implementing type's name; are available wherever it is'''), ''and'' '''simplify using D constraints while improving the situation of extending <code>structs</code> without inheritance woes'''. | ||
− | It optionally allows for using any < | + | It optionally allows for using any <code>interface</code>s as 'extension interfaces' that would be remotely implementable by <code>class</code>/<code>struct</code>s. |
== Description == | == Description == |
Revision as of 19:49, 10 September 2015
Title: | Extension Interfaces |
---|---|
DIP: | xx |
Version: | 0.1 |
Status: | Draft |
Created: | 2015-09-04 |
Last Modified: | 2015-09-10 |
Author: | Enamex(:real-name) |
Links: |
Contents
Abstract
The existing constraints system depends on manually-written 'tests' instead of compiler-assisted comparisons of signatures of the template with a prototype, for example (a la `sig` in ML).
A system is here proposed to allow the declaration of 'extension interfaces' or 'lazy interfaces' that are ad-hoc attachable to existing types (structs and classes) and thus usable as native parts of a type within the package that declares the extension interfaces.
Rationale
Idiomatic modern D in its current state prefers free functions for any added 'interface functionality' to types instead of adding (static) methods directly to types or having to inherit from interfaces with classes.
This presents several key problems:
1. The currently idiomatic D way to use constraints is by testing the availability of certain methods to be called on a type without regard for any nominative 'bundle' it might belong to in another language (like the names for Haskell type-classes or Rusts's traits or the earlier versions of C++ Concepts proposal).
2. D template constraints, because of template hygiene, can only recognize native methods of a type, effectively ignoring a substantial amount of functionality idiomatically provided through free functions.
3. Extending classes exclusively through free-standing functions has significant namespacing issues.
This proposal allows for disciplined definition of extension functions (as is provided today via UFCS) that obey namespacing rules, are available to call wherever the name of a supported type is passed to a template (locally, they bind to the implementing type's name; are available wherever it is), and simplify using D constraints while improving the situation of extending structs
without inheritance woes.
It optionally allows for using any interface
s as 'extension interfaces' that would be remotely implementable by class
/struct
s.
Description
Allow the declaration of extension interfaces that accept function prototypes (with optional default implementations) and associated constants (both `static immutable` and manifest constants allowed) and the implementation of such interfaces for a `struct` or `class` after its definition.
Extension interfaces defined under the user's most super package would be implementable for types provided in third-party packages/libraries. Types defined under the user's most super package would be able to implement extension interfaces provided in third-party package/libraries.
It is impossible for a package to implement a third-party extension interface for a third-party type. It must be that either the extension interface or the type or both are local to the client code.
In any sub-module of the most super package of a project, any local type or extension interface that has been coupled with a local or foreign extension interface or type would behave as if the type natively possesses all functions and associated constants defined in the implemented extension interfaces. All functions in the type-implemented extension interfaces would be callable through method syntax `object.method()` and visible throughout the package no matter where the implementation block lies (it must still be under the same most super package, however).
Usage
module superb.a;
struct Tile {
void scan() {}
}
module superb.call.util;
import superb.a;
// anonymous extension interface, attached to struct superb.a.Tile
interface (Tile) {
bool detectChell() { return true; }
}
// named extension interface 'prototype'; later implemented by types and tested against;
// could also support using any interface this way, and dropping the `()` in declaration
interface () HydraulicMobileEnvironment
: path.SomeOtherInterfaceInherited // inheritance of extension interfaces allowed
{
static immutable int startingHealth;
bool smash_opposite_tiles() { return true; }
}
module superb.core;
import superb.a: Tile;
import superb.call.util: HydraulicMobileEnvironment;
// implement extension interface 'HydraulicMobileEnvironment' for type Tile
interface (Tile)
: HydraulicMobileEnvironment
{
// provided required implementation, uses default of 'smash_opposite_tiles'
static immutable int startingHealth = 95;
}
module superb.mainfunc;
import superb.a;
import superb.core;
void main() {
Tile t;
t.smash_opposite_tiles(); // works; 'superb.core.HydraulicMobileEnvironment' implementation for 'superb.a.Tile';
// error if 'superb.core.HydraulicMobileEnvironment' is not imported;
// You import the interface declaration,
// not the implementation (which is automatically available unless declared private)
t.scan(); // native definition
t.detectChell(); // function in anonymous extension interface; doesn't require importing anything for it
}
Alternative syntax:
alias struct ForeignStructToBeExtended {
/* add methods/static functions and constants here;
they become a part of `ForeignStructToBeExtended`
wherever `ForeignStructToBeExtended` is in scope
in the super of the package that defines the struct/class.
*/
}
alias interface ExtensionInterface {
/* An 'extension interface' declared as `alias interface`
is 'inherited' from by either a local type if the alias
interface is foreign; or a foreign type if the alias
interface is local to the super package.
*/
}
// `alias interface` is implemented for a struct/class by the syntax:
alias struct ForeignStructToBeExtended
: ExtensionInterface
{
/*methods and stuff*/
}
Recommendations
When writing a DIP, try not to express your opinion. DIPs should provide facts and be as objective as possible. Even when that's pretty hard, you can make the DIP look more objective by not using, for example, "I prefer XXX because YYY". If YYY is an objective advantage, write that in a more objective way, like "XXX can be a better option because YYY". Try to leave non-technical personal preferences aside; "XXX can be a better option because the syntax is nicer" is not good enough even when you don't say "I prefer".
Try not to include half-baked ideas. If you are not sure about something, leave it outside the DIP and write it on the NG announcement instead for further discussion. The idea can be added to the DIP in the future when it is in a better shape.
NG Announcement
When posting the DIP announcement to the NG, please copy the abstract, so people can easily know what is it about and follow the link if they are interested.
Copyright
This document has been placed in the Public Domain.