Import conflict resoultion

Regan Heath regan at netwin.co.nz
Wed Jul 12 18:57:14 PDT 2006


Here is my current understanding of the problems and solutions proposed to  
date. Comments are welcome, including "enough already! the horse is dead!"  
;)

Regan


Import conflict resolution
--------------------------

Assumptions
-----------
Imports are private by default (Walter seems to like this now?)


Terminology
-----------
FQN = Fully Qualified Name


Problems
--------
A:
Importing 2 modules with the same symbol causes an immediate collision eg.
import std.string;
import std.regexp;

..find(.. //error std.string.find or std.regexp.find

B:
Changes to a module at a later date may cause a collision at a later date.
import a;
import b;

..foo(.. //calling a.foo.

At later date symbol 'foo' is added to b and now the collision occurs.


Current behaviour
-----------------
"How imports work is that first a name is searched for in the current  
namespace. If it is not found, then it is looked for in the import list.  
If it is
found uniquely among the imports, then that is used. If it is in more than  
one import, an error occurs"
  - Walter

"Import does not import names into the current namespace. It imports them  
into a secondary namespace, that is looked in *only* if the name isn't
found in the current namespace. Alias can then cherry-pick specific  
symbols out of that secondary namespace and put them in the current  
namespace,
possibly renaming them along the way."
  - Walter


Current solutions
-----------------
1:
import std.string;
import std.regexp;
alias std.string.find find;

..find(.. //no error std.string.find is called

2:
import std.string;
import std.regexp;

..std.string.find(.. //no error FQN is used


Opinions/pros/cons on the current solutions
-------------------------------------------

1:
  - PRO,Solves the import conflict for the intended symbol.
  - CON,Does NOT solve future symbol conflicts.
  - CON,The alias is 'physically' seperate from the import.

2:
  - PRO,Solves the import conflict for the intended symbol.
  - CON,Does NOT solve future symbol conflicts.
  - CON,Causes more verbose code.

Neither of these current solutions solves B.


Proposed solutions
------------------
3:
Prevent import into the "secondary namespace", only allow FQN access. (AKA  
"static import")
static import std.string; //exact syntax may vary
static import std.regexp; //exact syntax may vary

..find(.. //error
..std.string.find(.. //ok
..std.regexp.find(.. //ok

"All 'static import' does is eliminate the "if the name is not found in  
the current module, look in the imports" step. This has the effect of
requiring the use of FQN's when referencing imports."
  - Walter

4:
Import into a custom namespace, eg.
import std.string str; //exact syntax may vary
import std.regexp;

..str.find(.. //calls std.string.find
..find(.. //calls std.regexp.find

This method should also prevent import into the "secondary namespace".

5:
Allow selective import of the exact symbol which is required.
import std.string.find; //exact syntax may vary

..find(.. //calls std.string.find

No symbols from std.string would be present in the "secondary namespace".


Opinions/pros/cons on the various solutions
-------------------------------------------

3:
  - PRO,Solves the import conflict for the intended symbol.
  - PRO,Solves future symbol conflicts in the same module(s).
  - CON,Causes more verbose code, using FQN all the time.

4:
  - PRO,Solves the import conflict for the intended symbol.
  - PRO,Solves future symbol conflicts in the same module(s).

5:
  - PRO,Solves the import conflict for the intended symbol.
  - CON,Does NOT solve future symbol conflicts.


Conclusions, extra thoughts
---------------------------

The ideal solution IMO is #4, it solves both A and B in a simple, easy and  
elegant(IMO) fashion something that cannot
be said about the existing solutions #1 and #2.

To the people who want FQN all the time.. You may view the proposed  
solutions #3 and #4 as identical (in value to you).
The deciding factor to my mind is that other people do not want to use FQN  
all the time therefore #4 makes the most
people happy. The interesting thing, is that FQN can be achieved with  
soltion #4, note that:
   import std.string std.string;

would be FQN access, without! any further special syntax (like "static"),  
to me this makes #4 better than #3
even if you want to use FQN.

Yes, it requires more typing but I don't think you can complain about  
having to type the module name once more
when you plan to type it 20+ more times in the course of the file, can you?

FQN access is curently always present, but not required. However, with  
solution #4 FQN access to the symbol is no longer
ever required. It's presence results in code which can call the same  
function in more than one way. This allows the code to
be inconsistent for no gain (that I can imagine) therefore i think that  
solution #4 should prevent FQN access to symbols.



More information about the Digitalmars-d mailing list