Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/gusty/fscontrol
[ARCHIVED] FsControl in now included in FSharpPlus https://fsprojects.github.io/FSharpPlus
https://github.com/gusty/fscontrol
Last synced: 2 months ago
JSON representation
[ARCHIVED] FsControl in now included in FSharpPlus https://fsprojects.github.io/FSharpPlus
- Host: GitHub
- URL: https://github.com/gusty/fscontrol
- Owner: gusty
- License: apache-2.0
- Created: 2012-12-09T08:17:19.000Z (about 12 years ago)
- Default Branch: master
- Last Pushed: 2016-12-25T05:50:01.000Z (about 8 years ago)
- Last Synced: 2024-05-02T00:38:30.910Z (8 months ago)
- Language: F#
- Homepage:
- Size: 842 KB
- Stars: 104
- Watchers: 25
- Forks: 17
- Open Issues: 5
-
Metadata Files:
- Readme: README.md
- License: LICENSE.md
Awesome Lists containing this project
README
FsControl [![Build Status](https://api.travis-ci.org/gmpl/FsControl.svg?branch=master)](https://travis-ci.org/gmpl/FsControl)
=========A library that enhances the F# coding experience by providing the following two innovations:
1. A mechanism for defining standalone generic functions within F# similar to Haskell's typeclasses (once their defining module is in scope). These generic functions are resolved at compile time to an implementation type using .Net's static class method overloading mechanism, .Net type extension facility, and F#'s type inferencing. For example, a generic definition of
Map
may be made so as to automatically resolve toList.map
,Array.map
,Seq.map
, or whatever the provided mappable (Functor) value's implementation ofMap
resolves to at compile time.2. The provision of a set of generic standalone function definitions together with their implementation over a set of .NET and F# core types. Some of these functions are abstractions ported from Haskell but adapted to the F#/.NET world. Other functions offer a solution to normalize common function calls over different Types which represent the same abstraction but mainly due to historical reasons have different names and signatures.
### NOTE: FsControl is part of [FSharpPlus](https://github.com/gmpl/FSharpPlus) now.
### This project is no longer maintained here.
### Getting Started
- Download binaries from [Nuget](https://www.nuget.org/packages/FsControl/), use the latest version (2.x)
- Open an F# script file or the F# interactive and reference the library
```fsharp
#r @"C:\Your path to the binaries\FsControl.dll";;
```
Ignore warnings about F# metadata.- Now you can create generic functions, here's an example with
map
([fmap](https://wiki.haskell.org/Functor) for Haskellers, [Select](http://www.dotnetperls.com/select) for C-sharpers):
```fsharp
let inline map f x = FsControl.Map.Invoke f x;;
```
Static constraints will be inferred automatically.- Test it with .NET / F# primitive types:
```fsharp
map string [|2;3;4;5|];;
// val it : string [] = [|"2"; "3"; "4"; "5"|]map ((+) 9) (Some 3);;
// val it : int option = Some 12
```
- You can also create your own type with a methodMap
:
```fsharp
type Tree<'t> =
| Tree of 't * Tree<'t> * Tree<'t>
| Leaf of 't
static member Map (x:Tree<'a>, f) =
let rec loop f = function
| Leaf x -> Leaf (f x)
| Tree (x, t1, t2) -> Tree (f x, loop f t1, loop f t2)
loop f x
```
By adding the static memberMap
we say that we're makingTree
an instance ofMap
.- Try mapping over your new type:
```fsharp
let myTree = Tree(6, Tree(2, Leaf 1, Leaf 3), Leaf 9);;
map ((*) 10) myTree;;
// val it : Tree = Tree (60,Tree (20,Leaf 10,Leaf 30),Leaf 90)
```
Generic functions may be seen as an exotic thing in F# that only saves a few key strokes (map
instead ofList.map
orArray.map
) still they allow you to reach a higher abstraction level, using ad-hoc polymorphism.But more interesting is the use of operators. You can't prefix them with the module they belong to, well you can but then it's no longer an operator. As an example many F# libraries define the bind operator
(>>=)
but it's not generic so if you use two different types which are both monads you will need to prefix it e.g.State.(>>=)
andReader.(>>=)
which defeats the purpose of having an operator.Here you can easily define a generic bind operator:
```fsharp
let inline (>>=) x f = Bind.Invoke x f
```
Or if you do [Railway Oriented Programming](https://www.google.ch/#q=railway+oriented+programming) you can finally have your generic Kleisli composition (fish) operator:
```fsharp
let inline (>=>) f g x = Bind.Invoke (f x) g
```
Also when working with combinators, the generic applicative functor (space invaders) operator is very handy:
```fsharp
let inline (<*>) x y = Apply.Invoke x y
```
Of course they are already defined in the FsControl.Operators module and they work with primitive and user defined types.
### Next steps:
- Have a look at the [sample files](https://github.com/gmpl/FsControl/blob/master/FsControl.Core/Samples/) adjust the path of the binaries and run the .fsx scripts.
- Before creating your own library of generic functions be aware that [FsControl.Operators](https://github.com/gmpl/FsControl/blob/master/FsControl.Core/Operators.fs) is a lightweight module with some operators and functions used mainly to test the project. Also take the time to visit [F#+](https://github.com/gmpl/FSharpPlus) which is a library that re-export all those functions and also provides more derived operators, builders and other interesting stuff.
- In the rare case that you are not interested in the generic stuff but want to re-use specific implementations many methods in FsControl are defined as extension methods and some have a C# friendly signature.
- Keep reading the doc.How does it works
-----------------Technically this is a base library with a collection of generic methods overloaded for .NET and F# core types but extensible to other types at the same time.
There are basically two Types involved in these overloads:
- The type that will implement the abstraction. This will be a “real” type, for example
List
orTree
. We may refer to this type as the type or as the instance-type, since it represents an instance of the abstraction. At the same time we can classify these types in primitive types and custom types. By primitive types we refer to existing types in the .NET framework.- The type that represent the abstraction: Examples of these types are
Map
,Bind
,Append
, etc. This will be a "dummy" type implemented as a static class with an overloaded method (usually with the same name as the type) and an entry point method called 'Invoke'. From now on and in order to differentiate from the type-instance we will call this type the method-class.For Haskellers this 'method-class' abstraction is similar to Haskell's Type-Classes but with a single method.
For OOP-ers it may compare to interfaces or abstract classes but with a single method, early binding (compile-time) and without dependencies on the assembly where the interface is defined.
FsControl contains overloads mainly for primitive types, but the generic functions will resolve to any type (a user-defined type) having a member with a matching signature. This makes possible to use some libraries that don't depend on FsControl, as long as the signature is the right one it will work.
How to use FsControl
--------------------You may find hard to understand how to use FsControl, the best is to have a look at the source code, if you just want to use the generic functions for primitive types open
FsControl.Operators
module.The purpose of the overloads is to associate primitive types with method-classes, here we can have three different scenarios:
1) Add a new method-class and instance-types for existing types.
This is the most complex scenario, to define a new method-class is not straightforward, there will be some guidelines but at the moment the best is to have a look at the source code.
2) Add a new type and make it an instance of an existing method-class.
There are 2 ways:a) You can have a look at the signature of the method you want to implement in the source code, which will follow this convention:
```fsharp
static member [inline] [MethodName] (arg1:Type, [more args], output[:ReturnType], mthd[:MethodClassName]) =
Implementation
```
To find the exact signature you need to look at the source code of the method-class you are interested.Here's an example:
In the source code for
Map
(in Functor.fs) theoption
instance is defined like this:
```fsharp
[]static member Map (x:option<_>, f, []impl:Map) = Option.map f x
```
So you can create a typeTree
and add an instance for the existing method-classMap
this way:
```fsharp
// Define a type Tree
type Tree<'a> =
| Tree of 'a * Tree<'a> * Tree<'a>
| Leaf of 'a// add an instance for Map (Functor)
static member Map (x:Tree<_>, f, impl) =
let rec loop f (t:Tree<'a>) =
match t with
| Leaf x -> Leaf (f x)
| Tree (x, t1, t2) -> Tree (f x, loop f t1, loop f t2)
loop f x
```
b) Some methods accept also a 'clean signature' without the unused parametersoutput
andimpl
. You can find a list of these methods below, in the section "How can I make my classes FsControl-ready?". This way it doesn't require to reference FsControl binaries.3) Add an instance for an existing Type of an existing method-class:
We can’t do this. This is only possible if we have control over the source code of either the instance-type or the method-class.
The fact that the association must be done either in the instance-class or in the method-class is due to both a technical limitation(1)
and a conceptual reason(2)
.-
(1)
Extensions methods are not taken into account in overload resolution.
-(2)
It may lead to a bad design practice, something similar happens in Haskell with Type Classes (see [orphan instances](http://www.haskell.org/haskellwiki/Orphan_instance)).Anyway if you find a situation like this you can either wrap the type you're interested in or "shadow" the generic function.
How can I make my classes FsControl-ready?
------------------------------------------An easy way to make classes in your project callable from FsControl without referencing FsControl DLLs at all is to use standard signatures for your methods. Here's a list of the standard signatures available at the moment, this list is not exhaustive:
Functors:
```fsharp
static member Map (x:MyFunctor<'T>, f:'T->'U) : MyFunctor<'U> = {my map impl.}
```
Applicatives:
```fsharp
static member Return (x:'T) : MyApplicative<'T> = {my Return impl.}
static member (<*>) (f:MyApplicative<'T->'U>, x:MyApplicative<'T>) : MyApplicative<'U> = {my Apply impl.}
```
Monads:
```fsharp
static member Return (x:'T) : MyMonad<'T> = {my Return impl.} // similar to Applicatives
static member Bind (x:MyMonad<'T>, f:'T->MyMonad<'U>) : MyMonad<'U> = {my Bind impl.}
```
Monoids:
```fsharp
static member Empty : MyMonoid = {my Empty impl.} // get_Empty() = ... may be used alternatively.
static member Append (x:MyMonoid, y:MyMonoid) : MyMonoid = {my Append impl.}
static member Concat (x:seq) : MyMonoid = {my Concat impl.} // optional: it can be automatically derived from Append
```
Foldables:
```fsharp
static member FoldBack (source:MyFoldable<'T>, folder:'T->'State->'State, state:'State) : 'State = {my FoldBack impl.}
static member ToSeq (source:MyFoldable<'T>) : seq<'T> = {my ToSeq impl.}
static member FromSeq (source: seq<'T>) : MyFoldable<'T> = {my FromSeq impl.}
```If you find problems (typically insane compile times) you can still define it as described in 2).
FAQ
---Q: Is there a performance penalty in using this library?
A: Normally not, because all these constraints are resolved at compile time and code is inlined so on the contrary there might be eventually some speed up at run-time. On the other hand, the more overloads the more pressure on the compiler, this project will take several minutes to compile but this doesn't mean that by linking the FsControl dll to an application the compile time of the application will slow down. It will slow down depending on how much generic code (inlined) uses.
Q: What about the generic abstractions? Do they mean that by having a generic solution the code will be less efficient?
A: In many cases yes, but in FsControl the most generic code is in 'default methods' and normally the overloads define specialized methods which are optimized for specific instances.
Q: Where can I find more information about the abstractions provided?
A: There are many posts on Haskell everywhere, of those some are very formal and others are more intuitive. Apart from that in the last years there were some F# intuitive explanations in different blogs about the same abstractions implemented in F# in a non-generic way.
Q: Is this a Haskell emulator?
A: No, there are some abstractions specifics to F#, however it's true that this library (as many others F# libs) is heavily inspired in concepts coming from Haskell but as F# is another language with another type system, strict evaluation and some different conventions there are many differences in names, types and implementations. Also there are some specific F# abstractions. Anyway by knowing those differences you may be able to translate Haskell code to F#. There is a [Haskell Compatibility module in F#+](https://github.com/gmpl/FSharpPlus/blob/master/FSharpPlus/Compatibility.fs#L4) which is another project based in FsControl, and it contains binds to mimic Haskell functions, operator and types.
Q: How can I contribute to this library?
A: You can review the code, find better implementations of specific instances, add missing instances for primitive types, add/propose new method-classes, add sample files an so on. Finding issues, making suggestions, giving feedback, discussion in general is also welcome.