【发布时间】:2018-08-30 19:07:38
【问题描述】:
object IO {
def getHtmlFromWebsiteViaHttp(link: String, apiKey: String = ""): String = {
Http(link)
.param("access_token", apiKey)
.asString
.body
}
}
class SongService {
private def retrieveSongId(songName: String): Option[JsValue] = {
val formattedSongName = songName.replace(" ", "%20")
val searchLink = "https://api.genius.com/search?q=" + formattedSongName
//impure call
val geniusStringResponse = IO.getHtmlFromWebsiteViaHttp(searchLink, apiKey)
//Extra processing on geniusStringResponse
}
}
我目前的设计是我将有一个服务类,负责通过外部 API 获取一些信息。现在我明白了,不可能有 100% 纯函数。
我的问题:处理需要连接到 Scala/FP 中的外部 API 的情况的最佳方法是什么?目的是通过最小化不纯函数来拥有最合适的“函数式编程风格”
目前,我将所有 API 调用封装在 IO 对象中。这够合适吗?我看到了单子的例子。在这种情况下我应该合并一个单子风格吗?
【问题讨论】:
-
“我将所有 API 调用封装在 IO 对象中” - 你的意思是,你只是把你的不纯函数移到了一个名为
"IO"的object中?我认为这有点过于简单化了...... -
对。因此,如果这不是解决此问题的最佳方法,那该怎么做呢?
-
我不确定这有多不纯。您只是从外部资源中获取价值。对我来说似乎是透明的。就像@AndreyTyukin 所说,将代码移动到一个单独的对象就是……移动代码。它不会(实际上)影响程序的运行。
-
是的,但是从外部资源中获取价值的行为不是天生不纯洁的吗?不能保证输出与指定的输入相同。 (404 错误或 301 正常)
-
@Lasf 它不是引用透明的,因为它很容易由于外部状态的变化而给出不同的结果,例如您尝试访问的服务器出现故障或机器失去互联网访问权限。更不用说 API 的歌单可能会定期更新。
标签: scala functional-programming pure-function