digitalmars.D.learn - Trying to use a template class with ranges
- mark (32/32) Feb 06 2020 I am starting on porting Python's difflib's sequence matcher to D.
- mark (7/7) Feb 06 2020 I forgot to mention: I want the class to work with:
- Steven Schveighoffer (24/64) Feb 06 2020 1. If one template parameter depends 100% on the other, it doesn't need
- mark (60/69) Feb 06 2020 On Thursday, 6 February 2020 at 15:21:46 UTC, Steven
- Steven Schveighoffer (3/5) Feb 06 2020 Hah, forgot that it's a class. Yes, I DID mean new Diff ;)
- mark (12/17) Feb 06 2020 Wow, that's all it needed to compile!
I am starting on porting Python's difflib's sequence matcher to D.
I want to have a class that will accept two ranges whose elements 
are of the same type and whose elements can be compared for 
equality.
How do I make a class declaration that specifies a (forward) 
range type and an equality-supporting element type?
Here's what doesn't work:
class Diff(T, E) {
	T a; // T should be a forward range of E elements
	T b; // E elements must support == and !=
         // This is a hash key=E element, value=slice of size_t
	size_t[][E] b2j;
	this(T a, T b) {
		this.a = a;
		this.b = b;
		chainB();
	}
	void chainB() {
		foreach (i, element; b)
			b2j[element] ~= i;
		// TODO
	}
}
unittest {
	import std.stdio: writeln;
     writeln("unittest for the diffrange library.");
	auto a = ["Tulips are yellow,", "Violets are blue,",
                   "Agar is sweet,", "As are you."];
	auto b = ["Roses are red,", "Violets are blue,",
                   "Sugar is sweet,", "And so are you."];
	auto diff = Diff(a, b);
}
 Feb 06 2020
I forgot to mention: I want the class to work with: Diff(aForwardRange, bForwardRange) where T = ForwardRange, E = anything that supports == A common use case is for two sequences of strings (i.e., lines read from two files). Diff(aString, bString) where T = char[] or wchar[] or dchar[] E = char or wchar or dchar
 Feb 06 2020
On 2/6/20 7:16 AM, mark wrote:
 I am starting on porting Python's difflib's sequence matcher to D.
 
 I want to have a class that will accept two ranges whose elements are of 
 the same type and whose elements can be compared for equality.
 
 How do I make a class declaration that specifies a (forward) range type 
 and an equality-supporting element type?
 
 Here's what doesn't work:
 
 class Diff(T, E) {
      T a; // T should be a forward range of E elements
      T b; // E elements must support == and !=
          // This is a hash key=E element, value=slice of size_t
      size_t[][E] b2j;
 
      this(T a, T b) {
          this.a = a;
          this.b = b;
          chainB();
      }
 
      void chainB() {
          foreach (i, element; b)
              b2j[element] ~= i;
          // TODO
      }
 }
 
 unittest {
      import std.stdio: writeln;
 
      writeln("unittest for the diffrange library.");
      auto a = ["Tulips are yellow,", "Violets are blue,",
                    "Agar is sweet,", "As are you."];
      auto b = ["Roses are red,", "Violets are blue,",
                    "Sugar is sweet,", "And so are you."];
      auto diff = Diff(a, b);
 }
 
1. If one template parameter depends 100% on the other, it doesn't need 
to be a parameter. i.e. I would do:
class Diff(T) {
    alias E = ElementType!T;
    ...
}
2. Class constructors do not support IFTI. You have to explicitly 
instantiate. To use IFTI, you need to create a factory function. i.e. to 
make the code above work, your ctor call should be Diff!(typeof(a)) 
(assuming you take the advice in 1).
But you can instead write a function to use IFTI:
auto makeDiff(T)(T r1, T r2) { return Diff!T(r1, r2); }
...
auto diff = makeDiff(a, b);
3. You should declare constraints signifying what types are valid. i.e.:
class Diff(T) if (
    isForwardRange!T // it's a forward range
    && is(typeof(T.init.front == T.init.front)) // elements are comparable
)
Might be good to put this constraint on the factory function too.
Note: is(typeof(...)) basically checks if the thing inside the typeof 
compiles.
-Steve
 Feb 06 2020
On Thursday, 6 February 2020 at 15:21:46 UTC, Steven 
Schveighoffer wrote:
[snip]
 3. You should declare constraints signifying what types are 
 valid. i.e.:
 class Diff(T) if (
    isForwardRange!T // it's a forward range
    && is(typeof(T.init.front == T.init.front)) // elements are 
 comparable
 )
 Might be good to put this constraint on the factory function 
 too.
I don't know how to do that syntactically. I tried the obvious 
and it wouldn't compile.
But even without that it won't compile although it is much closer 
now!
import std.range: ElementType, front, isForwardRange;
class Diff(T) if (
         isForwardRange!T && // T is a range
         is(typeof(T.init.front == T.init.front)) // Elements 
support ==
         ) {
     alias E = ElementType!T;
     T a;
     T b;
     size_t[][E] b2j;
     this(T a, T b) {
         this.a = a;
         this.b = b;
         chainB();
     }
     private final void chainB() {
         foreach (i, element; b)
             b2j[element] ~= i;
         // TODO
     }
}
auto differ(T)(T a, T b) { return Diff!T(a, b); }
unittest {
     import std.array;
     import std.stdio: writeln;
     writeln("unittest for the diffrange library.");
     auto d1 = differ("one two three four".array, "one too tree 
four".array);
     auto a = ["Tulips are yellow,", "Violets are blue,", "Agar is 
sweet,",
               "As are you."];
     auto b = ["Roses are red,", "Violets are blue,", "Sugar is 
sweet,",
               "And so are you."];
     auto d2 = differ(a, b);
}
Here's the error output:
Excluding package.d file from test due to 
https://issues.dlang.org/show_bug.cgi?id=11847
src/package.d(50,35): Error: no property opCall for type 
diffrange.Diff!(dchar[]), did you mean new Diff!(dchar[])?
src/package.d(57,21): Error: template instance 
diffrange.differ!(dchar[]) error instantiating
src/package.d(50,35): Error: no property opCall for type 
diffrange.Diff!(string[]), did you mean new Diff!(string[])?
src/package.d(62,21): Error: template instance 
diffrange.differ!(string[]) error instantiating
/home/mark/opt/bin/ldc2 failed with exit code 1.
Curiously the "Excluding package.d file" message isn't true.
Not that I mind, I just want to be able to get it going.
Thanks!
PS The line numbers don't match because I've deleted some structs 
& enums that are above the class but which aren't used yet.
 Feb 06 2020
On 2/6/20 11:05 AM, mark wrote:src/package.d(50,35): Error: no property opCall for type diffrange.Diff!(dchar[]), did you mean new Diff!(dchar[])?Hah, forgot that it's a class. Yes, I DID mean new Diff ;) -Steve
 Feb 06 2020
On Thursday, 6 February 2020 at 16:29:57 UTC, Steven Schveighoffer wrote:On 2/6/20 11:05 AM, mark wrote:Wow, that's all it needed to compile! And I've added the extra check you suggested: auto differ(T)(T a, T b) if ( isForwardRange!T && // T is a range is(typeof(T.init.front == T.init.front)) // Elements support == ) { return new Diff!T(a, b); } Thanks!src/package.d(50,35): Error: no property opCall for type diffrange.Diff!(dchar[]), did you mean new Diff!(dchar[])?Hah, forgot that it's a class. Yes, I DID mean new Diff ;) -Steve
 Feb 06 2020








 
  
  
 
 mark <mark qtrac.eu>
 mark <mark qtrac.eu> 