Mailing list for all users of the OCaml language and system.
 help / color / mirror / Atom feed
* module type...
@ 2004-11-11  8:35 Pietro Abate
  2004-11-11  8:56 ` [Caml-list] " skaller
  0 siblings, 1 reply; 10+ messages in thread
From: Pietro Abate @ 2004-11-11  8:35 UTC (permalink / raw)
  To: ocaml ml

Hi all,
I've a problem that boils down to this code below.

If I define
			
(* -------------- *)
module TermType =
    struct
        type t = int
        let copy t = t
        let to_string t = "this is a string"
    end
;;

let l : TermType.t list = [1;2];;

(* -------------- *)

everything is ok. and I get 
val l : TermType.t list = [1; 2]

but if I define:

(* -------------- *)
module type ElType =
    sig
        type t
        val copy : t -> t
        val to_string : t -> string
    end
;;
module TermType : ElType =
    struct
        type t = int
        let copy t = t
        let to_string t = "this is a string"
    end
;;
let l : TermType.t list = [1;2];;

(* -------------- *)

now the compiler tells me that:

This expression has type int but is here used with type TermType.t

but TermType.t should be unified with int...

why ? I can I solve this problem ?

and an other small question... why functor sigs are not defined in the std
library (ie for the Set.Make module ?)

thanks.

p

-- 
++ "All great truths begin as blasphemies." -George Bernard Shaw
++ Please avoid sending me Word or PowerPoint attachments.
   See http://www.fsf.org/philosophy/no-word-attachments.html


^ permalink raw reply	[flat|nested] 10+ messages in thread

* Re: [Caml-list] module type...
  2004-11-11  8:35 module type Pietro Abate
@ 2004-11-11  8:56 ` skaller
  2004-11-11  9:09   ` Pietro Abate
  0 siblings, 1 reply; 10+ messages in thread
From: skaller @ 2004-11-11  8:56 UTC (permalink / raw)
  To: Pietro Abate; +Cc: ocaml ml

On Thu, 2004-11-11 at 19:35, Pietro Abate wrote:
> Hi all,
> I've a problem that boils down to this code below.

> module type ElType =
>     sig
>         type t
>     end
> ;;
> module TermType : ElType =
>     struct
>         type t = int
>     end
> ;;
> let l : TermType.t list = [1;2];;
> 
> (* -------------- *)
> 
> now the compiler tells me that:
> 
> This expression has type int but is here used with type TermType.t
> 
> but TermType.t should be unified with int...
> 

By coercing TermType to ElType, you have coerced 'int' to 't',
where 't' is an abstract type, in other words the coercion is hiding
the representation. There are two ways around this

(a) don't hide the representation, put

	type t = int

into the module type ElType as well.

(b) add a type constructor function to the module  TermType

	let term_of_int i = i

with declaration in ElType:

	val term_of_int: int -> t


-- 
John Skaller, mailto:skaller@users.sf.net
voice: 061-2-9660-0850, 
snail: PO BOX 401 Glebe NSW 2037 Australia
Checkout the Felix programming language http://felix.sf.net




^ permalink raw reply	[flat|nested] 10+ messages in thread

* Re: [Caml-list] module type...
  2004-11-11  8:56 ` [Caml-list] " skaller
@ 2004-11-11  9:09   ` Pietro Abate
  2004-11-11  9:48     ` Virgile Prevosto
  0 siblings, 1 reply; 10+ messages in thread
From: Pietro Abate @ 2004-11-11  9:09 UTC (permalink / raw)
  To: ocaml ml

Thanks John.

On Thu, Nov 11, 2004 at 07:56:10PM +1100, skaller wrote:
> By coercing TermType to ElType, you have coerced 'int' to 't',
> where 't' is an abstract type, in other words the coercion is hiding
> the representation. There are two ways around this

ok, I got the problem... but neither of your solutions solve my more
general problem as I need ElType.t to be an abstract data type as I'm
using it in a functor...

now it says:
This expression has type int but is here used with type
  SetOfTerm.Set.elt = TermType.t

for the same reason as before, but I can't change the ElType.t
without breaking the functor

[....]
module MakeSet =
    functor (Q : ElType) -> functor (SetFunc : SetFuncType) -> struct

    module Set = SetFunc ( struct
        type t = Q.t
        let compare = compare
    end
    )

[....]    

below the extended version of the problem with the same error...

any advice on how to modify my modules/classes design to overcome this
problem ? what I'm trying to do is to have set, setofset, setofsetofset 
etc using the same code base and functors...

:)
p

--------------------------------

module type ElType =
    sig
        type t
        val copy : t -> t
        val to_string : t -> string
    end

module type OrderedType =
    sig
        type t
        val compare : t -> t -> int
    end
  
module type SetFuncType =
    functor (Ord : OrderedType) ->
      sig
        type elt = Ord.t
        type t
        val empty : t 
        val compare : t -> t -> int
        val add : elt -> t -> t
      end

module MakeSet =
    functor (Q : ElType) -> functor (SetFunc : SetFuncType) -> struct

    module Set = SetFunc ( struct
        type t = Q.t
        let compare = compare
    end
    )

    class set ?(el=[]) () =
        object(self)
            initializer self#add el
            val mutable s = Set.empty
            method add el =
                s <- List.fold_left ( fun s' e -> Set.add e s' ) s el
        end
end

module TermType : ElType =
    struct
        type t = int
        let copy t = t
        let to_string t = "this is a string"
    end
;;

    
module SetOfTerm = MakeSet (TermType) (Set.Make);;

(* ------------------ *)

module SetOfTermType : ElType =
    struct
        type t = SetOfTerm.set
        let copy t = t
        let to_string t = "this is an other string"
    end
;;

module SetOfSet = MakeSet (SetOfTermType) (Set.Make);;

let newset_naked ?(el=[]) () = new SetOfTerm.set ~el:el
let newset ?(el=[]) () = (`Set (new SetOfTerm.set ~el:el))
let newsetofset ?(el=[]) () = (`Set (newset ~el:el ()))

let a = newset_naked ~el:[1;2] ();;
let b = newsetofset ~el:[a] ();;

-- 
++ "All great truths begin as blasphemies." -George Bernard Shaw
++ Please avoid sending me Word or PowerPoint attachments.
   See http://www.fsf.org/philosophy/no-word-attachments.html


^ permalink raw reply	[flat|nested] 10+ messages in thread

* Re: [Caml-list] module type...
  2004-11-11  9:09   ` Pietro Abate
@ 2004-11-11  9:48     ` Virgile Prevosto
  2004-11-12 16:11       ` Specifying abstract type in a record josh
  0 siblings, 1 reply; 10+ messages in thread
From: Virgile Prevosto @ 2004-11-11  9:48 UTC (permalink / raw)
  To: caml-list

Hello,
Le 11.11.2004, à 20:09:31, Pietro Abate a écrit:
> 
> below the extended version of the problem with the same error...
> 
> any advice on how to modify my modules/classes design to overcome this
> problem ? what I'm trying to do is to have set, setofset,
> setofsetofset etc using the same code base and functors...
> 

You can use a variant of the solution (a), with t still being abstract
in ElType and the use of explicit type equalities (see the reference
manual,<http://pauillac.inria.fr/ocaml/htmlman/manual004.html>, section
2.4) for each of the implementations:

> module type ElType  =
>     sig
>         type t (* type t is abstract here. *)
>         val copy : t -> t
>         val to_string : t -> string
>     end

> module TermType : ElType 

                      with type t = int 
                      (* TermType.t is unifiable with int *)
  ... 
> module SetOfTerm = MakeSet (TermType) (Set.Make);;
> 
> module SetOfTermType : ElType 

                      with type t = SetOfTerm.set
                      (* SetOfTermType.t is unifiable with
                         Set.OfTerm.set *)
  ...
-- 
E tutto per oggi, a la prossima volta
Virgile


^ permalink raw reply	[flat|nested] 10+ messages in thread

* Specifying abstract type in a record
  2004-11-11  9:48     ` Virgile Prevosto
@ 2004-11-12 16:11       ` josh
  2004-11-12 16:33         ` [Caml-list] " Luc Maranget
                           ` (4 more replies)
  0 siblings, 5 replies; 10+ messages in thread
From: josh @ 2004-11-12 16:11 UTC (permalink / raw)
  To: caml-list

OK,  Here's what I'm trying to do:

# type doer = { file_name:string ; actor: ('a -> unit) };;

But when I do this, it tells me that I've got "Unbound type parameter 'a ".

However,  if I change the 'a to int like this:

# type doer = { file_name:string; actor: (int -> unit) };;

It works as expected, but is not polymorphic (which is what I was trying 
to do).
If I create an abstract type like:
# type t
# type doer = { file_name:string; actor (t -> unit) };;

It works until I try to use a created record:

# let b = {file_name = "one"; actor = (fun x -> () ) };;
# b.actor 10;;
The expression has type int but is used with type t

even if I try to do this

# type t = int;;

it doesn't work.  So, how _can_ I specify a record with an abstract 
field?  Or is this
the Wrong Way to Do Things?     Thank you for your help.

-jbs


^ permalink raw reply	[flat|nested] 10+ messages in thread

* Re: [Caml-list] Specifying abstract type in a record
  2004-11-12 16:11       ` Specifying abstract type in a record josh
@ 2004-11-12 16:33         ` Luc Maranget
  2004-11-12 16:37         ` Aleksey Nogin
                           ` (3 subsequent siblings)
  4 siblings, 0 replies; 10+ messages in thread
From: Luc Maranget @ 2004-11-12 16:33 UTC (permalink / raw)
  To: josh; +Cc: caml-list

> OK,  Here's what I'm trying to do:
> 
> # type doer = { file_name:string ; actor: ('a -> unit) };;
> 

Try this
type 'a doer = { file_name:string ; actor: ('a -> unit) };;

-- Luc


^ permalink raw reply	[flat|nested] 10+ messages in thread

* Re: [Caml-list] Specifying abstract type in a record
  2004-11-12 16:11       ` Specifying abstract type in a record josh
  2004-11-12 16:33         ` [Caml-list] " Luc Maranget
@ 2004-11-12 16:37         ` Aleksey Nogin
  2004-11-12 16:37         ` Richard Jones
                           ` (2 subsequent siblings)
  4 siblings, 0 replies; 10+ messages in thread
From: Aleksey Nogin @ 2004-11-12 16:37 UTC (permalink / raw)
  To: josh, Caml List

josh wrote:
> OK,  Here's what I'm trying to do:
> 
> # type doer = { file_name:string ; actor: ('a -> unit) };;
> 

What exactly are you trying to accomplish? Would something like

type 'a doer = { file_name:string ; actor: ('a -> unit) }

work for you?

-- 
Aleksey Nogin

Home Page: http://nogin.org/
E-Mail: nogin@cs.caltech.edu (office), aleksey@nogin.org (personal)
Office: Jorgensen 70, tel: (626) 395-2907


^ permalink raw reply	[flat|nested] 10+ messages in thread

* Re: [Caml-list] Specifying abstract type in a record
  2004-11-12 16:11       ` Specifying abstract type in a record josh
  2004-11-12 16:33         ` [Caml-list] " Luc Maranget
  2004-11-12 16:37         ` Aleksey Nogin
@ 2004-11-12 16:37         ` Richard Jones
  2004-11-12 16:40         ` Andrew Bagdanov
  2004-11-12 16:44         ` Matt Gushee
  4 siblings, 0 replies; 10+ messages in thread
From: Richard Jones @ 2004-11-12 16:37 UTC (permalink / raw)
  To: josh; +Cc: caml-list

[-- Attachment #1: Type: text/plain, Size: 1959 bytes --]

On Fri, Nov 12, 2004 at 10:11:51AM -0600, josh wrote:
> OK,  Here's what I'm trying to do:
> 
> # type doer = { file_name:string ; actor: ('a -> unit) };;
> 
> But when I do this, it tells me that I've got "Unbound type parameter 'a ".

You need to do this:

# type 'a doer = { file_name:string ; actor: ('a -> unit) };;
type 'a doer = { file_name : string; actor : 'a -> unit; }

because you're creating a polymorphic type ('a doer).  Note that this
won't let you store an actor function of type, say, int -> unit and
then replace it with another function of type, say, string -> unit.
For the same reason it won't let you pass an ``int doer'' to a
function expecting a ``string doer''.

> # type t
> # type doer = { file_name:string; actor (t -> unit) };;
> 
> It works until I try to use a created record:
> 
> # let b = {file_name = "one"; actor = (fun x -> () ) };;
> # b.actor 10;;
> The expression has type int but is used with type t
> 
> even if I try to do this
> 
> # type t = int;;
> 
> it doesn't work.

OK, this is another problem.  Here what you're doing is defining a
_new_ type ``t'', which is unrelated to your old type ``t''.

Here's another example:

# let f () = "hello";;
val f : unit -> string = <fun>
# f ();;
- : string = "hello"
# let g = f;;    
val g : unit -> string = <fun>
# let f () = "goodbye";;
val f : unit -> string = <fun>
# f ();;
- : string = "goodbye"
# g ();;
- : string = "hello"

Notice that ``g'' still calls the old ``f'', even after ``f'' has been
redefined.

There's a beginner's list:

> Beginner's list: http://groups.yahoo.com/group/ocaml_beginners

Rich.

-- 
Richard Jones.  http://www.annexia.org/  http://www.j-london.com/
>>>   http://www.team-notepad.com/ - collaboration tools for teams   <<<
Merjis Ltd. http://www.merjis.com/ - improving website return on investment
Use Perl libs in OCaml - http://www.merjis.com/developers/perl4caml

[-- Attachment #2: Digital signature --]
[-- Type: application/pgp-signature, Size: 189 bytes --]

^ permalink raw reply	[flat|nested] 10+ messages in thread

* Re: [Caml-list] Specifying abstract type in a record
  2004-11-12 16:11       ` Specifying abstract type in a record josh
                           ` (2 preceding siblings ...)
  2004-11-12 16:37         ` Richard Jones
@ 2004-11-12 16:40         ` Andrew Bagdanov
  2004-11-12 16:44         ` Matt Gushee
  4 siblings, 0 replies; 10+ messages in thread
From: Andrew Bagdanov @ 2004-11-12 16:40 UTC (permalink / raw)
  To: caml-list

josh writes:
 > OK,  Here's what I'm trying to do:
 > 
 > # type doer = { file_name:string ; actor: ('a -> unit) };;
 > 
 > But when I do this, it tells me that I've got "Unbound type parameter 'a ".
 > 

You need to add a type parameter to the doer type:

 # type 'a doer = {file_name:string; actor: ('a -> unit) };;
 type 'a doer = { file_name : string; actor : 'a -> unit; }

This binds the type parameter 'a in your original definition.  Now
things should work the way you want:

 # let b = {file_name = "one"; actor = (fun x -> () ) };;
 val b : 'a doer = {file_name = "one"; actor = <fun>}
 # b.actor 10;;
 - : unit = ()

Note that in this definition, b.actor remains polymorphic.  If you
were to do this:

 # let b = {file_name = "one"; actor = (fun x -> Printf.printf "%d\n" x) };;
 val b : int doer = {file_name = "one"; actor = <fun>}

then 'a becones concrete (int).

Hope this helps...

-Andy


^ permalink raw reply	[flat|nested] 10+ messages in thread

* Re: [Caml-list] Specifying abstract type in a record
  2004-11-12 16:11       ` Specifying abstract type in a record josh
                           ` (3 preceding siblings ...)
  2004-11-12 16:40         ` Andrew Bagdanov
@ 2004-11-12 16:44         ` Matt Gushee
  4 siblings, 0 replies; 10+ messages in thread
From: Matt Gushee @ 2004-11-12 16:44 UTC (permalink / raw)
  To: caml-list

On Fri, Nov 12, 2004 at 10:11:51AM -0600, josh wrote:

> # type doer = { file_name:string ; actor: ('a -> unit) };;
> 
> But when I do this, it tells me that I've got "Unbound type parameter 'a 
> ".

Your first impulse was on the right track, but see below.

> # type t
> # type doer = { file_name:string; actor (t -> unit) };;
> 
> It works until I try to use a created record:
> 
> # let b = {file_name = "one"; actor = (fun x -> () ) };;
> # b.actor 10;;
> The expression has type int but is used with type t

Right. You can't directly *use* an abstract type. Generally, you would
declare an abstract type in an interface--an .mli file, a module sig, or
a class type--then specify it ('type t = ...') in an implementation.
Abstract types are good for keeping implementation details hidden from
the user, but they don't, by themselves, give you polymorphism.

> it doesn't work.  So, how _can_ I specify a record with an abstract 
> field?

  type 'a doer = { file_name : string; actor : ('a -> unit) }

That's known as a parameterized record type.

You would also do the same thing for other data structures, and for
objects (though the syntax is a bit different in the latter case)

-- 
Matt Gushee                 When a nation follows the Way,
Haven Rock Press            Horses bear manure through
Englewood, Colorado, USA        its fields;   
books@havenrock.com         When a nation ignores the Way,
                            Horses bear soldiers through
                                its streets.
                                
                            --Lao Tzu (Peter Merel, trans.)


^ permalink raw reply	[flat|nested] 10+ messages in thread

end of thread, other threads:[~2004-11-12 16:44 UTC | newest]

Thread overview: 10+ messages (download: mbox.gz / follow: Atom feed)
-- links below jump to the message on this page --
2004-11-11  8:35 module type Pietro Abate
2004-11-11  8:56 ` [Caml-list] " skaller
2004-11-11  9:09   ` Pietro Abate
2004-11-11  9:48     ` Virgile Prevosto
2004-11-12 16:11       ` Specifying abstract type in a record josh
2004-11-12 16:33         ` [Caml-list] " Luc Maranget
2004-11-12 16:37         ` Aleksey Nogin
2004-11-12 16:37         ` Richard Jones
2004-11-12 16:40         ` Andrew Bagdanov
2004-11-12 16:44         ` Matt Gushee

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox