Expressões computacionais
As expressões de computação em F# fornecem uma sintaxe conveniente para escrever cálculos que podem ser sequenciados e combinados usando construções e ligações de fluxo de controle. Dependendo do tipo de expressão computacional, eles podem ser pensados como uma forma de expressar mônadas, monoides, transformadores de mônadas e functores aplicadores. No entanto, ao contrário de outras linguagens (como a notação do em Haskell), elas não estão ligadas a uma única abstração e não dependem de macros ou outras formas de metaprogramação para realizar uma sintaxe conveniente e sensível ao contexto.
Visão geral
Os cálculos podem assumir muitas formas. A forma mais comum de computação é uma execução de único encadeamento, que é fácil de entender e modificar. No entanto, nem todas as formas de processamento são tão diretas quanto a execução em um único thread. Eis alguns exemplos:
- Cálculos não determinísticos
- Cálculos assíncronos
- Cálculos efetivos
- Cálculos generativos
Em geral, existem cálculos sensíveis ao contexto que devem ser realizados em determinadas partes de uma aplicação. Escrever código sensível ao contexto pode ser um desafio, pois é fácil "vazar" cálculos fora de um determinado contexto sem abstrações para impedir que você faça isso. Essas abstrações são muitas vezes desafiadoras para escrever por si mesmo, e é por isso que o F# tem uma maneira generalizada de fazer isso chamado expressões computacionais.
As expressões computacionais oferecem uma sintaxe uniforme e um modelo de abstração para codificar cálculos sensíveis ao contexto.
Cada expressão de computação é apoiada por um construtor de tipo
Visão geral da sintaxe
Todas as expressões computacionais têm a seguinte forma:
builder-expr { cexper }
Nesta forma, builder-expr
é o nome de um tipo de construtor que define a expressão de computação, e cexper
é o corpo de expressão da expressão de computação. Por exemplo, async
código de expressão computacional pode ter esta aparência:
let fetchAndDownload url =
async {
let! data = downloadData url
let processedData = processData data
return processedData
}
Há uma sintaxe especial e adicional disponível dentro de uma expressão de computação, como mostrado no exemplo anterior. As seguintes formas de expressão são possíveis com expressões computacionais:
expr { let! ... }
expr { and! ... }
expr { do! ... }
expr { yield ... }
expr { yield! ... }
expr { return ... }
expr { return! ... }
expr { match! ... }
Cada uma dessas palavras-chave e outras palavras-chave F# padrão só estão disponíveis em uma expressão de computação se tiverem sido definidas no tipo de construtor de suporte. A única exceção é match!
, que é em si açúcar sintático para o uso de let!
seguido por uma correspondência de padrão no resultado.
O tipo construtor é um objeto que define métodos especiais que governam a maneira como os fragmentos da expressão computacional são combinados; ou seja, seus métodos controlam como a expressão computacional se comporta. Outra maneira de descrever uma classe builder é dizer que ela permite que você personalize a operação de muitas construções F#, como loops e associações.
let!
A palavra-chave let!
vincula o resultado de uma chamada a outra expressão de computação a um nome:
let doThingsAsync url =
async {
let! data = getDataAsync url
...
}
Se você vincular a chamada a uma expressão de computação com let
, não obterá o resultado da expressão de computação. Em vez disso, você terá vinculado o valor da opção de compra não realizada à expressão de cálculo. Utilize let!
para vincular ao resultado.
let!
é definido pelo membro Bind(x, f)
no tipo de construtor.
and!
A palavra-chave and!
permite vincular os resultados de várias chamadas de expressão computacional de maneira eficiente.
let doThingsAsync url =
async {
let! data = getDataAsync url
and! moreData = getMoreDataAsync anotherUrl
and! evenMoreData = getEvenMoreDataAsync someUrl
...
}
O uso de uma série de let! ... let! ...
força a reexecução de ligações dispendiosas, portanto, o uso de let! ... and! ...
deve ser usado ao vincular os resultados de numerosas expressões computacionais.
and!
é definido principalmente pelo membro MergeSources(x1, x2)
no tipo de construtor.
Opcionalmente, MergeSourcesN(x1, x2 ..., xN)
pode ser definido para reduzir o número de nós de agrupamento, enquanto BindN(x1, x2 ..., xN, f)
ou BindNReturn(x1, x2, ..., xN, f)
pode ser usado para vincular os resultados da expressão de computação de forma eficiente, sem nós de agrupamento.
do!
A palavra-chave do!
é para chamar uma expressão de computação que retorna um tipo semelhante a unit
(definido pelo membro Zero
no construtor):
let doThingsAsync data url =
async {
do! submitData data url
...
}
Para o fluxo de trabalho assíncrono , esse tipo é Async<unit>
. Para outras expressões computacionais, é provável que o tipo seja CExpType<unit>
.
do!
é definido pelo membro Bind(x, f)
no tipo construtor, onde f
produz um unit
.
yield
A palavra-chave yield
é para retornar um valor da expressão de computação para que ele possa ser consumido como um IEnumerable<T>:
let squares =
seq {
for i in 1..10 do
yield i * i
}
for sq in squares do
printfn $"%d{sq}"
Na maioria dos casos, ele pode ser omitido pelos chamadores. A maneira mais comum de omitir yield
é com o operador ->
:
let squares =
seq {
for i in 1..10 -> i * i
}
for sq in squares do
printfn $"%d{sq}"
Para expressões mais complexas que podem gerar muitos valores diferentes, e talvez condicionalmente, simplesmente omitir a palavra-chave pode fazer:
let weekdays includeWeekend =
seq {
"Monday"
"Tuesday"
"Wednesday"
"Thursday"
"Friday"
if includeWeekend then
"Saturday"
"Sunday"
}
Tal como acontece com a palavra-chave yield no C#, cada elemento na expressão de computação é reproduzido à medida que é iterado.
yield
é definido pelo membro Yield(x)
no tipo construtor, onde x
é o item a ser devolvido.
yield!
A palavra-chave yield!
é para nivelar uma coleção de valores de uma expressão de computação:
let squares =
seq {
for i in 1..3 -> i * i
}
let cubes =
seq {
for i in 1..3 -> i * i * i
}
let squaresAndCubes =
seq {
yield! squares
yield! cubes
}
printfn $"{squaresAndCubes}" // Prints - 1; 4; 9; 1; 8; 27
Quando avaliada, a expressão computacional acionada por yield!
terá seus itens produzidos um a um, simplificando o resultado.
yield!
é definido pelo membro YieldFrom(x)
no tipo de construtor, onde x
é uma coleção de valores.
Ao contrário yield
, yield!
deve ser explicitamente especificado. Seu comportamento não está implícito em expressões computacionais.
return
A palavra-chave return
encapsula um valor no tipo correspondente à expressão computacional. Além de expressões computacionais usando yield
, é usado para "completar" uma expressão de computação:
let req = // 'req' is of type 'Async<data>'
async {
let! data = fetch url
return data
}
// 'result' is of type 'data'
let result = Async.RunSynchronously req
return
é definido pelo membro Return(x)
no tipo builder, onde x
é o item para encapsular. No uso de let! ... return
, BindReturn(x, f)
pode ser utilizado para melhorar o desempenho.
return!
A palavra-chave return!
avalia o valor de uma expressão de computação e envolve esse resultado no tipo correspondente à expressão de computação.
let req = // 'req' is of type 'Async<data>'
async {
return! fetch url
}
// 'result' is of type 'data'
let result = Async.RunSynchronously req
return!
é definido pelo membro ReturnFrom(x)
no tipo construtor, onde x
é outra expressão computacional.
match!
A palavra-chave match!
permite que você embuta uma chamada para outra expressão de computação e correspondência de padrão em seu resultado:
let doThingsAsync url =
async {
match! callService url with
| Some data -> ...
| None -> ...
}
Ao chamar uma expressão de computação com match!
, ele realizará o resultado da chamada como let!
. Isso é frequentemente usado ao chamar uma expressão de computação onde o resultado é um opcional.
Expressões computacionais incorporadas
A biblioteca principal do F# define quatro expressões computacionais internas: Sequence Expressions, Async expressions, Task expressionse Query Expressions.
Criando um novo tipo de expressão computacional
Você pode definir as características de suas próprias expressões de computação criando uma classe builder e definindo certos métodos especiais na classe. A classe builder pode, opcionalmente, definir os métodos conforme listados na tabela a seguir.
A tabela a seguir descreve métodos que podem ser usados em uma classe de construtor de fluxo de trabalho.
Método | Assinatura(s) típica(s) | Descrição |
---|---|---|
Bind |
M<'T> * ('T -> M<'U>) -> M<'U> |
Pediu let! e do! em expressões computacionais. |
BindN |
(M<'T1> * M<'T2> * ... * M<'TN> * ('T1 * 'T2 ... * 'TN -> M<'U>)) -> M<'U> |
Exigiu let! e and! eficientes em expressões computacionais sem mesclar entradas.por exemplo, Bind3 , Bind4 . |
Delay |
(unit -> M<'T>) -> Delayed<'T> |
Envolve uma expressão de computação em uma função.
Delayed<'T> pode ser de qualquer tipo, comumente M<'T> ou unit -> M<'T> são usados. A implementação padrão retorna um M<'T> . |
Return |
'T -> M<'T> |
Pediu return em expressões computacionais. |
ReturnFrom |
M<'T> -> M<'T> |
Pediu return! em expressões computacionais. |
BindReturn |
(M<'T1> * ('T1 -> 'T2)) -> M<'T2> |
Apelou a uma let! ... return eficiente nas expressões computacionais. |
BindNReturn |
(M<'T1> * M<'T2> * ... * M<'TN> * ('T1 * 'T2 ... * 'TN -> M<'U>)) -> M<'U> |
Exigiu let! ... and! ... return eficientes em expressões computacionais sem mesclar entradas.por exemplo, Bind3Return , Bind4Return . |
MergeSources |
(M<'T1> * M<'T2>) -> M<'T1 * 'T2> |
Chamado and! em expressões computacionais. |
MergeSourcesN |
(M<'T1> * M<'T2> * ... * M<'TN>) -> M<'T1 * 'T2 * ... * 'TN> |
Exigiu and! em expressões computacionais, mas melhora a eficiência reduzindo o número de nós tupling.por exemplo, MergeSources3 , MergeSources4 . |
Run |
Delayed<'T> -> M<'T> ouM<'T> -> 'T |
Executa uma expressão de computação. |
Combine |
M<'T> * Delayed<'T> -> M<'T> ouM<unit> * M<'T> -> M<'T> |
Chamado para sequenciamento em expressões computacionais. |
For |
seq<'T> * ('T -> M<'U>) -> M<'U> ouseq<'T> * ('T -> M<'U>) -> seq<M<'U>> |
Requeriu expressões for...do em expressões de cálculo. |
TryFinally |
Delayed<'T> * (unit -> unit) -> M<'T> |
Solicitou expressões try...finally em expressões computacionais. |
TryWith |
Delayed<'T> * (exn -> M<'T>) -> M<'T> |
Pediu expressões try...with em expressões computacionais. |
Using |
'T * ('T -> M<'U>) -> M<'U> when 'T :> IDisposable |
Pediu ligações use em expressões computacionais. |
While |
(unit -> bool) * Delayed<'T> -> M<'T> ou(unit -> bool) * Delayed<unit> -> M<unit> |
Pediu expressões while...do em expressões computacionais. |
Yield |
'T -> M<'T> |
Solicitadas expressões yield em expressões computacionais. |
YieldFrom |
M<'T> -> M<'T> |
Chamado para expressões yield! em expressões computacionais. |
Zero |
unit -> M<'T> |
Chamado para ramos else vazios de expressões if...then em expressões computacionais. |
Quote |
Quotations.Expr<'T> -> Quotations.Expr<'T> |
Indica que a expressão de computação é passada ao membro Run como uma citação. Ele traduz todas as instâncias de um cálculo em uma cotação. |
Muitos dos métodos em uma classe builder usam e retornam uma construção M<'T>
, que normalmente é um tipo definido separadamente que caracteriza o tipo de cálculos que estão sendo combinados, por exemplo, Async<'T>
para expressões assíncronas e Seq<'T>
para fluxos de trabalho de sequência. As assinaturas desses métodos permitem que sejam combinados e aninhados entre si, de modo que o objeto de fluxo de trabalho retornado de uma construção possa ser passado para a próxima.
Muitas funções usam o resultado de Delay
como argumento: Run
, While
, TryWith
, TryFinally
e Combine
. O tipo Delayed<'T>
é o tipo de retorno de Delay
e, consequentemente, o parâmetro para essas funções.
Delayed<'T>
pode ser um tipo arbitrário que não precisa estar relacionado com M<'T>
; Normalmente são utilizados M<'T>
ou (unit -> M<'T>)
. A implementação padrão é M<'T>
. Veja aqui para uma análise mais detalhada.
O compilador, quando analisa uma expressão de computação, traduz a expressão em uma série de chamadas de função aninhadas usando os métodos na tabela anterior e o código na expressão de computação. A expressão aninhada tem a seguinte forma:
builder.Run(builder.Delay(fun () -> {{ cexpr }}))
No código acima, as chamadas para Run
e Delay
são omitidas se não estiverem definidas na classe do construtor de expressões de computação. O corpo da expressão computacional, aqui denotado como {{ cexpr }}
, é traduzido em outras chamadas para os métodos da classe builder. Este processo é definido recursivamente de acordo com as traduções na tabela a seguir. Código entre parênteses duplos {{ ... }}
ainda precisa ser traduzido, expr
representa uma expressão F# e cexpr
representa uma expressão de computação.
Expressão | Tradução |
---|---|
{{ let binding in cexpr }} |
let binding in {{ cexpr }} |
{{ let! pattern = expr in cexpr }} |
builder.Bind(expr, (fun pattern -> {{ cexpr }})) |
{{ do! expr in cexpr }} |
builder.Bind(expr, (fun () -> {{ cexpr }})) |
{{ yield expr }} |
builder.Yield(expr) |
{{ yield! expr }} |
builder.YieldFrom(expr) |
{{ return expr }} |
builder.Return(expr) |
{{ return! expr }} |
builder.ReturnFrom(expr) |
{{ use pattern = expr in cexpr }} |
builder.Using(expr, (fun pattern -> {{ cexpr }})) |
{{ use! value = expr in cexpr }} |
builder.Bind(expr, (fun value -> builder.Using(value, (fun value -> {{ cexpr }})))) |
{{ if expr then cexpr0 }} |
if expr then {{ cexpr0 }} else builder.Zero() |
{{ if expr then cexpr0 else cexpr1 }} |
if expr then {{ cexpr0 }} else {{ cexpr1 }} |
{{ match expr with | pattern_i -> cexpr_i }} |
match expr with | pattern_i -> {{ cexpr_i }} |
{{ for pattern in enumerable-expr do cexpr }} |
builder.For(enumerable-expr, (fun pattern -> {{ cexpr }})) |
{{ for identifier = expr1 to expr2 do cexpr }} |
builder.For([expr1..expr2], (fun identifier -> {{ cexpr }})) |
{{ while expr do cexpr }} |
builder.While(fun () -> expr, builder.Delay({{ cexpr }})) |
{{ try cexpr with | pattern_i -> expr_i }} |
builder.TryWith(builder.Delay({{ cexpr }}), (fun value -> match value with | pattern_i -> expr_i | exn -> System.Runtime.ExceptionServices.ExceptionDispatchInfo.Capture(exn).Throw())) |
{{ try cexpr finally expr }} |
builder.TryFinally(builder.Delay({{ cexpr }}), (fun () -> expr)) |
{{ cexpr1; cexpr2 }} |
builder.Combine({{ cexpr1 }}, {{ cexpr2 }}) |
{{ other-expr; cexpr }} |
expr; {{ cexpr }} |
{{ other-expr }} |
expr; builder.Zero() |
Na tabela anterior, other-expr
descreve uma expressão que não está listada na tabela. Uma classe builder não precisa implementar todos os métodos e dar suporte a todas as traduções listadas na tabela anterior. As construções que não são implementadas não estão disponíveis em expressões computacionais desse tipo. Por exemplo, se você não quiser oferecer suporte à palavra-chave use
em suas expressões de computação, poderá omitir a definição de Use
em sua classe builder.
O exemplo de código a seguir mostra uma expressão de computação que encapsula uma computação como uma série de etapas que podem ser avaliadas uma etapa de cada vez. Um tipo de união discriminada, OkOrException
, codifica o estado de erro da expressão conforme avaliado até agora. Esse código demonstra vários padrões típicos que você pode usar em suas expressões de computação, como implementações clichês de alguns dos métodos do construtor.
/// Represents computations that can be run step by step
type Eventually<'T> =
| Done of 'T
| NotYetDone of (unit -> Eventually<'T>)
module Eventually =
/// Bind a computation using 'func'.
let rec bind func expr =
match expr with
| Done value -> func value
| NotYetDone work -> NotYetDone (fun () -> bind func (work()))
/// Return the final value
let result value = Done value
/// The catch for the computations. Stitch try/with throughout
/// the computation, and return the overall result as an OkOrException.
let rec catch expr =
match expr with
| Done value -> result (Ok value)
| NotYetDone work ->
NotYetDone (fun () ->
let res = try Ok(work()) with | exn -> Error exn
match res with
| Ok cont -> catch cont // note, a tailcall
| Error exn -> result (Error exn))
/// The delay operator.
let delay func = NotYetDone (fun () -> func())
/// The stepping action for the computations.
let step expr =
match expr with
| Done _ -> expr
| NotYetDone func -> func ()
/// The tryFinally operator.
/// This is boilerplate in terms of "result", "catch", and "bind".
let tryFinally expr compensation =
catch (expr)
|> bind (fun res ->
compensation();
match res with
| Ok value -> result value
| Error exn -> raise exn)
/// The tryWith operator.
/// This is boilerplate in terms of "result", "catch", and "bind".
let tryWith exn handler =
catch exn
|> bind (function Ok value -> result value | Error exn -> handler exn)
/// The whileLoop operator.
/// This is boilerplate in terms of "result" and "bind".
let rec whileLoop pred body =
if pred() then body |> bind (fun _ -> whileLoop pred body)
else result ()
/// The sequential composition operator.
/// This is boilerplate in terms of "result" and "bind".
let combine expr1 expr2 =
expr1 |> bind (fun () -> expr2)
/// The using operator.
/// This is boilerplate in terms of "tryFinally" and "Dispose".
let using (resource: #System.IDisposable) func =
tryFinally (func resource) (fun () -> resource.Dispose())
/// The forLoop operator.
/// This is boilerplate in terms of "catch", "result", and "bind".
let forLoop (collection:seq<_>) func =
let ie = collection.GetEnumerator()
tryFinally
(whileLoop
(fun () -> ie.MoveNext())
(delay (fun () -> let value = ie.Current in func value)))
(fun () -> ie.Dispose())
/// The builder class.
type EventuallyBuilder() =
member x.Bind(comp, func) = Eventually.bind func comp
member x.Return(value) = Eventually.result value
member x.ReturnFrom(value) = value
member x.Combine(expr1, expr2) = Eventually.combine expr1 expr2
member x.Delay(func) = Eventually.delay func
member x.Zero() = Eventually.result ()
member x.TryWith(expr, handler) = Eventually.tryWith expr handler
member x.TryFinally(expr, compensation) = Eventually.tryFinally expr compensation
member x.For(coll:seq<_>, func) = Eventually.forLoop coll func
member x.Using(resource, expr) = Eventually.using resource expr
let eventually = new EventuallyBuilder()
let comp =
eventually {
for x in 1..2 do
printfn $" x = %d{x}"
return 3 + 4
}
/// Try the remaining lines in F# interactive to see how this
/// computation expression works in practice.
let step x = Eventually.step x
// returns "NotYetDone <closure>"
comp |> step
// prints "x = 1"
// returns "NotYetDone <closure>"
comp |> step |> step
// prints "x = 1"
// prints "x = 2"
// returns "Done 7"
comp |> step |> step |> step |> step
Uma expressão de computação tem um tipo subjacente, que a expressão retorna. O tipo subjacente pode representar um resultado calculado ou um cálculo atrasado que pode ser executado, ou pode fornecer uma maneira de iterar através de algum tipo de coleção. No exemplo anterior, o tipo subjacente era Eventually<_>
. Para uma expressão de sequência, o tipo subjacente é System.Collections.Generic.IEnumerable<T>. Para uma expressão de consulta, o tipo subjacente é System.Linq.IQueryable. Para uma expressão assíncrona, o tipo subjacente é Async
. O objeto Async
representa o trabalho a ser executado para calcular o resultado. Por exemplo, você chama Async.RunSynchronously
para executar um cálculo e retornar o resultado.
Operações personalizadas
Você pode definir uma operação personalizada em uma expressão de computação e usar uma operação personalizada como um operador em uma expressão de computação. Por exemplo, você pode incluir um operador de consulta em uma expressão de consulta. Ao definir uma operação personalizada, você deve definir os métodos Yield e For na expressão de computação. Para definir uma operação personalizada, coloque-a em uma classe builder para a expressão de computação e, em seguida, aplique o CustomOperationAttribute
. Esse atributo usa uma cadeia de caracteres como um argumento, que é o nome a ser usado em uma operação personalizada. Esse nome entra em escopo no início da chave de abertura da expressão computacional. Portanto, você não deve usar identificadores que tenham o mesmo nome de uma operação personalizada neste bloco. Por exemplo, evite o uso de identificadores como all
ou last
em expressões de consulta.
Ampliando construtores de software existentes com novas operações personalizadas
Se você já tiver uma classe builder, suas operações personalizadas podem ser estendidas de fora dessa classe builder. As extensões devem ser declaradas em módulos. Os namespaces não podem conter membros de extensão, exceto no mesmo arquivo e no mesmo grupo de declaração de namespace onde o tipo é definido.
O exemplo a seguir mostra as extensões da classe FSharp.Linq.QueryBuilder
existente.
open System
open FSharp.Linq
type QueryBuilder with
[<CustomOperation>]
member _.any (source: QuerySource<'T, 'Q>, predicate) =
System.Linq.Enumerable.Any (source.Source, Func<_,_>(predicate))
[<CustomOperation("singleSafe")>] // you can specify your own operation name in the constructor
member _.singleOrDefault (source: QuerySource<'T, 'Q>, predicate) =
System.Linq.Enumerable.SingleOrDefault (source.Source, Func<_,_>(predicate))
As operações personalizadas podem ser sobrecarregadas. Para obter mais informações, consulte F# RFC FS-1056 - Permitir sobrecargas de palavras-chave personalizadas em expressões de computação.
Compilando expressões computacionais de forma eficiente
As expressões de computação F# que suspendem a execução podem ser compiladas em máquinas de estado altamente eficientes através do uso cuidadoso de um recurso de baixo nível chamado código retomável. O código retomável é documentado em F# RFC FS-1087 e usado para Expressões de Tarefa.
As expressões de computação F# que são síncronas (ou seja, não suspendem a execução) podem, alternativamente, ser compiladas em máquinas de estado eficientes usando funções embutidas incluindo o atributo InlineIfLambda
. Exemplos são dados em F# RFC FS-1098.
Expressões de lista, expressões de matriz e expressões de sequência recebem tratamento especial do compilador F# para garantir a geração de código de alto desempenho.