回答:
チェックInteger.parse/1
してFloat.parse/1
。
Integer.parse/1
以上使用する理由はありますString.to_integer/1
か?
Integer.parse/1
は、:error
失敗するとアトムを返します。String.to_integer/1
をスローし(FunctionClauseError)
ます。
このページの人々に感謝します、ここで答えを単純化するだけです:
{intVal, ""} = Integer.parse(val)
文字列全体が(接頭辞だけでなく)解析されたことを検証するためです。
文字列から数値を作成する4つの関数があります
String.to_integer
うまく動作しますがString.to_float
、より難しいです:
iex()> "1 2 3 10 100" |> String.split |> Enum.map(&String.to_integer/1)
[1, 2, 3, 10, 100]
iex()> "1.0 1 3 10 100" |> String.split |> Enum.map(&String.to_float/1)
** (ArgumentError) argument error
:erlang.binary_to_float("1")
(elixir) lib/enum.ex:1270: Enum."-map/2-lists^map/1-0-"/2
(elixir) lib/enum.ex:1270: Enum."-map/2-lists^map/1-0-"/2
String.to_float
唯一例えば、よくフォーマットされたフロートを扱うことができます1.0
、ではない1
(整数)。記載されたことString.to_float
のドキュメント
テキスト表現が文字列であるフロートを返します。
stringは、小数点を含むfloatの文字列表現でなければなりません。小数点のない文字列を浮動小数点数として解析するには、Float.parse / 1を使用する必要があります。それ以外の場合は、ArgumentErrorが発生します。
ただしFloat.parse
、必要な数ではなく、2つの要素のタプルを返すため、それをパイプラインに入れることは「クール」ではありません。
iex()> "1.0 1 3 10 100" |> String.split \
|> Enum.map(fn n -> {v, _} = Float.parse(n); v end)
[1.0, 1.0, 3.0, 10.0, 100.0]
elem
タプルから最初の要素を取得するために使用すると、短くて甘くなります。
iex()> "1.0 1 3 10 100" |> String.split \
|> Enum.map(fn n -> Float.parse(n) |> elem(0) end)
[1.0, 1.0, 3.0, 10.0, 100.0]
これをchar_listに変換してから、Erlang to_integer/1
またはを使用できますto_float/1
。
例えば
iex> {myInt, _} = :string.to_integer(to_char_list("23"))
{23, []}
iex> myInt
23
fn q -> {v, _} = Float.parse(q); v end
、私が気に入らないことです。私はそれをで使用したいのですがEnum.map
、list |> Enum.map(&String.to_float/1)
string.to_floatは整数では機能しませんか?
Decimal.new("1") |> Decimal.to_integer
Decimal.new("1.0") |> Decimal.to_float
使用の問題Integer.parse/1
は、文字列の末尾にある限り、文字列の数値以外の部分を解析することです。例えば:
Integer.parse("01") # {1, ""}
Integer.parse("01.2") # {1, ".2"}
Integer.parse("0-1") # {0, "-1"}
Integer.parse("-01") # {-1, ""}
Integer.parse("x-01") # :error
Integer.parse("0-1x") # {0, "-1x"}
同様String.to_integer/1
に次の結果があります。
String.to_integer("01") # 1
String.to_integer("01.2") # ** (ArgumentError) argument error :erlang.binary_to_integer("01.2")
String.to_integer("0-1") # ** (ArgumentError) argument error :erlang.binary_to_integer("01.2")
String.to_integer("-01") # -1
String.to_integer("x-01") # ** (ArgumentError) argument error :erlang.binary_to_integer("01.2")
String.to_integer("0-1x") # ** (ArgumentError) argument error :erlang.binary_to_integer("01.2")
代わりに、最初に文字列を検証します。
re = Regex.compile!("^[+-]?[0-9]*\.?[0-9]*$")
Regex.match?(re, "01") # true
Regex.match?(re, "01.2") # true
Regex.match?(re, "0-1") # false
Regex.match?(re, "-01") # true
Regex.match?(re, "x-01") # false
Regex.match?(re, "0-1x") # false
^[0-9]*$
ユースケースによっては、正規表現の方が単純な場合があります(など)。
文字列を文字列内の数値型に変換して他のすべての文字を削除したい場合、これはやり過ぎですが、floatの場合はfloat、intの場合はint、文字列が含まれていない場合はnilを返します。数値型。
@spec string_to_numeric(binary()) :: float() | number() | nil
def string_to_numeric(val) when is_binary(val), do: _string_to_numeric(Regex.replace(~r{[^\d\.]}, val, ""))
defp _string_to_numeric(val) when is_binary(val), do: _string_to_numeric(Integer.parse(val), val)
defp _string_to_numeric(:error, _val), do: nil
defp _string_to_numeric({num, ""}, _val), do: num
defp _string_to_numeric({num, ".0"}, _val), do: num
defp _string_to_numeric({_num, _str}, val), do: elem(Float.parse(val), 0)
String.to_integer/1