AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • 主页
  • 系统&网络
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • 主页
  • 系统&网络
    • 最新
    • 热门
    • 标签
  • Ubuntu
    • 最新
    • 热门
    • 标签
  • Unix
    • 最新
    • 标签
  • DBA
    • 最新
    • 标签
  • Computer
    • 最新
    • 标签
  • Coding
    • 最新
    • 标签
主页 / coding / 问题 / 79341484
Accepted
Vlam
Vlam
Asked: 2025-01-09 13:49:16 +0800 CST2025-01-09 13:49:16 +0800 CST 2025-01-09 13:49:16 +0800 CST

此表达式的类型为 unit,但预期表达式的类型为 'a Client.io

  • 772

我正在尝试用下面的 OCaml 编写一个简单的可执行代码。

open Printf
open Lwt
open Cohttp
open Cohttp_lwt_unix
open Yojson

let () =
  let ip = "8.8.8.8" in
  let key = "" in
  let uri =
    Uri.of_string
      ("https://api.ip2location.io/?format=json&key=" ^ key ^ "&ip=" ^ ip)
  in
  Lwt_main.run
    ( Client.get uri >>= fun (resp, body) ->
      let code = resp |> Response.status |> Code.code_of_status in
      let json_promise = body |> Cohttp_lwt.Body.to_string in
      json_promise >>= fun json_string ->
      let json = Basic.from_string json_string in
      let open Yojson.Basic.Util in
      if code == 200 then
        if member "usage_type" json <> `Null then
          let usage_type = json |> member "usage_type" |> to_string in
          printf "usage_type: %s\n" usage_type
        else
          printf
            "ERROR: The usage_type field requires a paid subscription to the \
             Starter plan or higher."
      else if (code == 400 || code == 401) && member "error" json <> `Null then
        let error_message =
          json |> member "error" |> member "error_message" |> to_string
        in
        printf "ERROR: " ^ error_message
      else printf "HTTP Code: " ^ Int.to_string code )

但是当我跑步时我一直看到下面的内容dune build。

File "bin/main.ml", line 24, characters 10-46:
24 |           printf "usage_type: %s\n" usage_type
               ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
Error: This expression has type unit but an expression was expected of type
         'a Client.io

从其他 StackOverflow 帖子来看,它似乎与返回的不同类型有关,if/else但我已确保所有类型都if/else在使用printf。

如果有人能让我知道我还做错了什么,我将不胜感激。

ocaml
  • 2 2 个回答
  • 37 Views

2 个回答

  • Voted
  1. Andreas Rossberg
    2025-01-09T14:45:20+08:002025-01-09T14:45:20+08:00

    我怀疑这是由于>>=Lwt 中的类型是

    (>>=) : 'a Lwt.t -> ('a -> 'b Lwt.t) -> 'b Lwt.t
    

    在左侧,您将 的结果传递给它Client.get url,因此右侧必须是返回相同类型的函数。同样,Lwt.run期望Lwt.t。(我并不熟悉所涉及的所有库,但Client.io可能是类型同义词扩展到Lwt.t此处。)

    此示例的修复方法是使用>|=而不是第二个>>=,其类型为

    (>|=) : 'a Lwt.t -> ('a -> 'b) -> 'b Lwt.t
    
    • 2
  2. Best Answer
    Chris
    2025-01-09T14:59:11+08:002025-01-09T14:59:11+08:00

    Andreas Rossberg 已经发现了一个问题,但您还看到一些运算符优先级问题。

    这种事:

    printf "ERROR: " ^ error_message
    

    还有这个:

    printf "HTTP Code: " ^ Int.to_string code
    

    看起来真的很可疑,因为无法将printf "HTTP code: "的求值()提供给^。看起来你想要写类似这样的内容:

    printf ("HTTP Code: " ^ Int.to_string code)
    

    但是您不妨利用printf下面的方法编写,它更干净并且没有运算符优先级问题。

    printf "HTTP Code: %d" code
    

    还要注意打开多个模块。如果多个模块定义了相同的运算符,或者它们重新定义了标准库运算符,则可能会出现一些意外行为。

    除了完全限定名称外,您还可以本地打开模块,从而缓解此问题。但由于您已经在代码中使用过一次,因此您似乎知道此选项。

    从风格上看,我可能只会重构这一点:

            let error_message =
              json |> member "error" |> member "error_message" |> to_string
            in
            printf "ERROR: " ^ error_message
    

    到:

            json 
            |> member "error" 
            |> member "error_message" 
            |> to_string
            |> printf "ERROR: %s"
    

    也有意见,但有机会重新修改:

          if code == 200 then
            if member "usage_type" json <> `Null then
              let usage_type = json |> member "usage_type" |> to_string in
              printf "usage_type: %s\n" usage_type
            else
              printf
                "ERROR: The usage_type field requires a paid subscription to the \
                 Starter plan or higher."
          else if (code == 400 || code == 401) && member "error" json <> > `Null then
            let error_message =
              json |> member "error" |> member "error_message" |> to_string
            in
            printf "ERROR: " ^ error_message
          else printf "HTTP Code: " ^ Int.to_string code )
    

    使用模式匹配和条件保护:

          match code with
          | 200 when member "usage_type" json <> `Null ->
            json
            |> member "usage_type" 
            |> to_string
            |> printf "usage_type: %s\n"
          | 200 ->
            printf
              "ERROR: The usage_type field requires a paid subscription to the \
               Starter plan or higher."
          | (400 | 401) when member "error" json <> `Null ->
            json 
            |> member "error" 
            |> member "error_message" 
            |> to_string
            |> printf "ERROR: %s"
          | _ -> printf "HTTP Code: %d" code
    
    • 1

相关问题

  • 找出在模式匹配中使用 catch all 时这些类型不同的原因

  • 返回未命名的总和类型

  • List.of_seq的实现

  • 使用 ocamlformat 的 OCaml 多行字符串会产生丑陋的结果

  • 排除 OCaml 中 lwt_ppx 的编译错误:无法编译文件

Sidebar

Stats

  • 问题 205573
  • 回答 270741
  • 最佳答案 135370
  • 用户 68524
  • 热门
  • 回答
  • Marko Smith

    重新格式化数字,在固定位置插入分隔符

    • 6 个回答
  • Marko Smith

    为什么 C++20 概念会导致循环约束错误,而老式的 SFINAE 不会?

    • 2 个回答
  • Marko Smith

    VScode 自动卸载扩展的问题(Material 主题)

    • 2 个回答
  • Marko Smith

    Vue 3:创建时出错“预期标识符但发现‘导入’”[重复]

    • 1 个回答
  • Marko Smith

    具有指定基础类型但没有枚举器的“枚举类”的用途是什么?

    • 1 个回答
  • Marko Smith

    如何修复未手动导入的模块的 MODULE_NOT_FOUND 错误?

    • 6 个回答
  • Marko Smith

    `(表达式,左值) = 右值` 在 C 或 C++ 中是有效的赋值吗?为什么有些编译器会接受/拒绝它?

    • 3 个回答
  • Marko Smith

    在 C++ 中,一个不执行任何操作的空程序需要 204KB 的堆,但在 C 中则不需要

    • 1 个回答
  • Marko Smith

    PowerBI 目前与 BigQuery 不兼容:Simba 驱动程序与 Windows 更新有关

    • 2 个回答
  • Marko Smith

    AdMob:MobileAds.initialize() - 对于某些设备,“java.lang.Integer 无法转换为 java.lang.String”

    • 1 个回答
  • Martin Hope
    Fantastic Mr Fox msvc std::vector 实现中仅不接受可复制类型 2025-04-23 06:40:49 +0800 CST
  • Martin Hope
    Howard Hinnant 使用 chrono 查找下一个工作日 2025-04-21 08:30:25 +0800 CST
  • Martin Hope
    Fedor 构造函数的成员初始化程序可以包含另一个成员的初始化吗? 2025-04-15 01:01:44 +0800 CST
  • Martin Hope
    Petr Filipský 为什么 C++20 概念会导致循环约束错误,而老式的 SFINAE 不会? 2025-03-23 21:39:40 +0800 CST
  • Martin Hope
    Catskul C++20 是否进行了更改,允许从已知绑定数组“type(&)[N]”转换为未知绑定数组“type(&)[]”? 2025-03-04 06:57:53 +0800 CST
  • Martin Hope
    Stefan Pochmann 为什么 {2,3,10} 和 {x,3,10} (x=2) 的顺序不同? 2025-01-13 23:24:07 +0800 CST
  • Martin Hope
    Chad Feller 在 5.2 版中,bash 条件语句中的 [[ .. ]] 中的分号现在是可选的吗? 2024-10-21 05:50:33 +0800 CST
  • Martin Hope
    Wrench 为什么双破折号 (--) 会导致此 MariaDB 子句评估为 true? 2024-05-05 13:37:20 +0800 CST
  • Martin Hope
    Waket Zheng 为什么 `dict(id=1, **{'id': 2})` 有时会引发 `KeyError: 'id'` 而不是 TypeError? 2024-05-04 14:19:19 +0800 CST
  • Martin Hope
    user924 AdMob:MobileAds.initialize() - 对于某些设备,“java.lang.Integer 无法转换为 java.lang.String” 2024-03-20 03:12:31 +0800 CST

热门标签

python javascript c++ c# java typescript sql reactjs html

Explore

  • 主页
  • 问题
    • 最新
    • 热门
  • 标签
  • 帮助

Footer

AskOverflow.Dev

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve