Преобразовать строку в целочисленный тип в Go?

236

Я пытаюсь преобразовать строку, возвращенную из flag.Arg(n)в int. Какой идиоматический способ сделать это в Go?

Мэтт Джойнер
источник

Ответы:

299

Например,

package main

import (
    "flag"
    "fmt"
    "os"
    "strconv"
)

func main() {
    flag.Parse()
    s := flag.Arg(0)
    // string to int
    i, err := strconv.Atoi(s)
    if err != nil {
        // handle error
        fmt.Println(err)
        os.Exit(2)
    }
    fmt.Println(s, i)
}
peterSO
источник
14
func main() { ... }не принимает аргументов и не возвращает значения. Используйте функцию osпакета, Exitнапримерos.Exit(2).
peterSO
2
В качестве альтернативы просто сделайте роковое, например,panic(err)
Питер Бенгтссон,
70

Преобразование простых строк

Самый простой способ - использовать strconv.Atoi()функцию.

Обратите внимание, что есть много других способов. Например, fmt.Sscan()и strconv.ParseInt()которые дают большую гибкость, так как вы можете указать базу и битовый размер, например. Также как отмечено в документации strconv.Atoi():

Atoi эквивалентен ParseInt (s, 10, 0), преобразованному в тип int.

Вот пример использования упомянутых функций (попробуйте на Go Playground ):

flag.Parse()
s := flag.Arg(0)

if i, err := strconv.Atoi(s); err == nil {
    fmt.Printf("i=%d, type: %T\n", i, i)
}

if i, err := strconv.ParseInt(s, 10, 64); err == nil {
    fmt.Printf("i=%d, type: %T\n", i, i)
}

var i int
if _, err := fmt.Sscan(s, &i); err == nil {
    fmt.Printf("i=%d, type: %T\n", i, i)
}

Вывод (если вызывается с аргументом "123"):

i=123, type: int
i=123, type: int64
i=123, type: int

Разбор пользовательских строк

Существует также удобство, fmt.Sscanf()которое дает еще большую гибкость, так как с помощью строки формата вы можете указать числовой формат (например, ширину, основание и т. Д.) Вместе с дополнительными дополнительными символами на входе string.

Это отлично подходит для разбора пользовательских строк, содержащих число. Например, если ваш ввод предоставлен в форме, "id:00123"где у вас есть префикс, "id:"а число фиксировано 5 цифрами, дополненными нулями, если короче, это очень легко разобрать, как это:

s := "id:00123"

var i int
if _, err := fmt.Sscanf(s, "id:%5d", &i); err == nil {
    fmt.Println(i) // Outputs 123
}
icza
источник
О чем говорит второй аргумент ParseInt?
kaushik94
1
@ kaushik94 Нажмите на strconv.ParseInt()ссылку , и вы увидите сразу: ParseInt(s string, base int, bitSize int). Итак, это основа: «ParseInt интерпретирует строку s в заданной базе (от 2 до 36)»
icza
Обратите внимание, что аргумент bitSize для strconv.ParseInt () не будет преобразовывать строку в выбранный вами тип, а вместо этого используется только для ограничения результата определенной «битностью». Смотрите также: stackoverflow.com/questions/55925894/...
Львов
@viv Да, это правильно. Если значение типа intтребуется и strconv.ParseInt()используется, необходимо ручное преобразование типа (из int64в int).
icza
16

Вот три способа разбить строки на целые числа, от самого быстрого времени выполнения до самого медленного:

  1. strconv.ParseInt(...) быстрый
  2. strconv.Atoi(...) все еще очень быстро
  3. fmt.Sscanf(...) не очень быстро, но наиболее гибкий

Вот тест, который показывает использование и пример времени для каждой функции:

package main

import "fmt"
import "strconv"
import "testing"

var num = 123456
var numstr = "123456"

func BenchmarkStrconvParseInt(b *testing.B) {
  num64 := int64(num)
  for i := 0; i < b.N; i++ {
    x, err := strconv.ParseInt(numstr, 10, 64)
    if x != num64 || err != nil {
      b.Error(err)
    }
  }
}

func BenchmarkAtoi(b *testing.B) {
  for i := 0; i < b.N; i++ {
    x, err := strconv.Atoi(numstr)
    if x != num || err != nil {
      b.Error(err)
    }
  }
}

func BenchmarkFmtSscan(b *testing.B) {
  for i := 0; i < b.N; i++ {
    var x int
    n, err := fmt.Sscanf(numstr, "%d", &x)
    if n != 1 || x != num || err != nil {
      b.Error(err)
    }
  }
}

Вы можете запустить его, сохранив как atoi_test.goи запустив go test -bench=. atoi_test.go.

goos: darwin
goarch: amd64
BenchmarkStrconvParseInt-8      100000000           17.1 ns/op
BenchmarkAtoi-8                 100000000           19.4 ns/op
BenchmarkFmtSscan-8               2000000          693   ns/op
PASS
ok      command-line-arguments  5.797s
maerics
источник
2

Попробуй это

import ("strconv")

value : = "123"
number,err := strconv.ParseUint(value, 10, 32)
Доктор медицины Махеди Хасан
источник
0

Если вы контролируете входные данные, вы можете использовать мини-версию

package main

import (
    "testing"
    "strconv"
)

func Atoi (s string) int {
    var (
        n uint64
        i int
        v byte
    )   
    for ; i < len(s); i++ {
        d := s[i]
        if '0' <= d && d <= '9' {
            v = d - '0'
        } else if 'a' <= d && d <= 'z' {
            v = d - 'a' + 10
        } else if 'A' <= d && d <= 'Z' {
            v = d - 'A' + 10
        } else {
            n = 0; break        
        }
        n *= uint64(10) 
        n += uint64(v)
    }
    return int(n)
}

func BenchmarkAtoi(b *testing.B) {
    for i := 0; i < b.N; i++ {
        in := Atoi("9999")
        _ = in
    }   
}

func BenchmarkStrconvAtoi(b *testing.B) {
    for i := 0; i < b.N; i++ {
        in, _ := strconv.Atoi("9999")
        _ = in
    }   
}

самый быстрый вариант (при необходимости напишите чек). Результат:

Path>go test -bench=. atoi_test.go
goos: windows
goarch: amd64
BenchmarkAtoi-2                 100000000               14.6 ns/op
BenchmarkStrconvAtoi-2          30000000                51.2 ns/op
PASS
ok      path     3.293s
Jenyokcoder
источник
1
Какой ? В самом деле ? Люди, которые написали «иди» сделали многое легко. Не
крутите
А как насчет Atoi ("- 9999")?
Алексей Чечель