私有函数vs嵌套函数

时间:2013-04-07 04:42:26

标签: function f# functional-programming

问题是:

何时使用私有函数以及何时使用嵌套函数? (我问的是 F#,但答案可能与其他函数语言相关)

一个小例子

namespace SomeName

module BinaryReaderExt =
    open System.IO

    let seek (reader : BinaryReader) positions =
        reader.BaseStream.Seek(positions, SeekOrigin.Current) |> ignore

module Mod =
    open System.IO

    let private prFun reader:BinaryReader =
        //do something
        BinaryReaderExt.seek reader 10L


    let outerFun (stream :System.IO.Stream) =
        let reader = new System.IO.BinaryReader(stream)
        let seek = BinaryReaderExt.seek reader

        let nestedFun () =
            seek 10L
            //do something

        nestedFun()
        prFun reader

嵌套函数可以使用更高范围的数据,这是一个很大的好处。它也不会污染周围的模块。但是看起来很笨拙,不是吗?特别是当有一些大型嵌套函数时

相反,私人功能可以公开并进行测试。似乎它们看起来更具可读性

你有什么看法?

2 个答案:

答案 0 :(得分:4)

我经常在模块中使用private函数 - 通常用于" helper"模块中其他函数使用的函数,但不需要暴露给外部代码。

private函数的另一个用例是简单地使代码更具可读性。如果一个函数嵌套在另一个函数中,但是它太长而无法读取 - 例如,如果嵌套函数的代码构成函数长度的一半以上--I&包含它#39; ll通常将其移出到模块级别并使其private,以便调用函数的代码更容易理解。

答案 1 :(得分:0)

  

嵌套函数可以使用更高范围的数据,这是一个很大的好处。它也不会污染周围的模块。

我同意你的观点。 我的建议是将功能保持在合适的范围。例如,如果函数仅在一个地方使用,则最好是嵌套函数。例如,将loop向上移动并使其成为private函数是没有意义的。

let length xs =
   let rec loop acc = function
      | [] -> acc
      | _::xs -> loop (acc + 1) xs
   loop 0 acc
  但是看起来很笨拙,不是吗?特别是当有一些大型嵌套函数时

如果您需要大型嵌套函数,则可能是您做错了。它们应该分成多个小的嵌套函数,或者最外面的函数应该转换为一个类型。

  

相反,私人功能可以公开并进行测试。似乎它们看起来更具可读性。

可读性是一个主观问题。我认为组织问题更重要。嵌套函数的要点是它们很简单,可以通过测试最外面的函数来测试。

当函数具有更多适用性时,您可以将它们放入实用程序模块并在需要时打开该模块。请注意,除了标记函数private之外,还有其他隐藏函数的技术。例如,您可以使用fsi文件来指示所公开的接口。