Respostas:
Além da resposta correta de fedorqui , gostaria de mostrar a diferença entre o comprimento da string e o comprimento de bytes:
myvar='Généralités'
chrlen=${#myvar}
oLang=$LANG oLcAll=$LC_ALL
LANG=C LC_ALL=C
bytlen=${#myvar}
LANG=$oLang LC_ALL=$oLcAll
printf "%s is %d char len, but %d bytes len.\n" "${myvar}" $chrlen $bytlen
renderizará:
Généralités is 11 char len, but 14 bytes len.
você pode até dar uma olhada nos caracteres armazenados:
myvar='Généralités'
chrlen=${#myvar}
oLang=$LANG oLcAll=$LC_ALL
LANG=C LC_ALL=C
bytlen=${#myvar}
printf -v myreal "%q" "$myvar"
LANG=$oLang LC_ALL=$oLcAll
printf "%s has %d chars, %d bytes: (%s).\n" "${myvar}" $chrlen $bytlen "$myreal"
Vai responder:
Généralités has 11 chars, 14 bytes: ($'G\303\251n\303\251ralit\303\251s').
Nota: De acordo com o comentário de Isabell Cowan , eu adicionei a configuração $LC_ALL
junto com $LANG
.
O argumento funciona da mesma forma que as variáveis regulares
strLen() {
local bytlen sreal oLang=$LANG oLcAll=$LC_ALL
LANG=C LC_ALL=C
bytlen=${#1}
printf -v sreal %q "$1"
LANG=$oLang LC_ALL=$oLcAll
printf "String '%s' is %d bytes, but %d chars len: %s.\n" "$1" $bytlen ${#1} "$sreal"
}
vai funcionar como
strLen théorème
String 'théorème' is 10 bytes, but 8 chars len: $'th\303\251or\303\250me'
printf
Ferramenta de correção útil :Se vocês:
for string in Généralités Language Théorème Février "Left: ←" "Yin Yang ☯";do
printf " - %-14s is %2d char length\n" "'$string'" ${#string}
done
- 'Généralités' is 11 char length
- 'Language' is 8 char length
- 'Théorème' is 8 char length
- 'Février' is 7 char length
- 'Left: ←' is 7 char length
- 'Yin Yang ☯' is 10 char length
Não é realmente bonito ... Para isso, há uma pequena função:
strU8DiffLen () {
local bytlen oLang=$LANG oLcAll=$LC_ALL
LANG=C LC_ALL=C
bytlen=${#1}
LANG=$oLang LC_ALL=$oLcAll
return $(( bytlen - ${#1} ))
}
Então agora:
for string in Généralités Language Théorème Février "Left: ←" "Yin Yang ☯";do
strU8DiffLen "$string"
printf " - %-$((14+$?))s is %2d chars length, but uses %2d bytes\n" \
"'$string'" ${#string} $((${#string}+$?))
done
- 'Généralités' is 11 chars length, but uses 14 bytes
- 'Language' is 8 chars length, but uses 8 bytes
- 'Théorème' is 8 chars length, but uses 10 bytes
- 'Février' is 7 chars length, but uses 8 bytes
- 'Left: ←' is 7 chars length, but uses 9 bytes
- 'Yin Yang ☯' is 10 chars length, but uses 12 bytes
Mas deixou um comportamento UTF-8 estranho, como caracteres com espaço duplo, caracteres com espaço zero, deslocamento reverso e outros que não poderiam ser tão simples ...
Dê uma olhada em diffU8test.sh ou diffU8test.sh.txt para obter mais limitações.
Para obter o comprimento de uma string armazenada em uma variável, diga:
myvar="some string"
size=${#myvar}
Para confirmar se foi salvo corretamente, echo
ele:
$ echo "$size"
11
$rulename
começa com o $RULE_PREFIX
prefixo: [ "${rulename:0:${#RULE_PREFIX}}" == "$RULE_PREFIX" ]
#myvar
e {#myvar}
?
${#parameter}
: O comprimento em caracteres do valor de parâmetro é expandido substituído .
Você pode usar:
MYSTRING="abc123"
MYLENGTH=$(printf "%s" "$MYSTRING" | wc -c)
wc -c
ou wc --bytes
para contagens de bytes = caracteres Unicode são contados com 2, 3 ou mais bytes.wc -m
ou wc --chars
para contagem de caracteres = caracteres Unicode são contados como únicos até que usem mais bytes.mylen=$(printf "%s" "$HOME/.ssh" | wc -c)
considerando que a solução aceita falha e você precisa myvar=$HOME/.ssh
primeiro.
Eu queria o caso mais simples, finalmente este é um resultado:
echo -n 'Tell me the length of this sentence.' | wc -m;
36
echo '' | wc -m
=> 1
. Você precisaria usar -n
: echo -n '' | wc -m
=> 0
... caso em que ele é uma boa solução :)
-n do not output the trailing newline
Se você quiser usar isso com argumentos de linha de comando ou função, certifique-se de usar em size=${#1}
vez de size=${#$1}
. O segundo pode ser mais instintivo, mas é uma sintaxe incorreta.
size=${#1}
é certamente válido.
#
não está substituindo o $
- a parte $
externa do aparelho ainda é o operador de expansão. O #
é o operador de comprimento, como sempre.
Em resposta ao início da postagem:
Se você deseja usar isso com argumentos de linha de comando ou função ...
com o código:
size=${#1}
Pode haver o caso em que você apenas deseja verificar um argumento de comprimento zero e não precisa armazenar uma variável. Eu acredito que você pode usar esse tipo de sintaxe:
if [ -z "$1" ]; then
#zero length argument
else
#non-zero length
fi
Veja GNU e wooledge para obter uma lista mais completa de expressões condicionais do Bash.
Usando seu exemplo fornecido
#KISS (Keep it simple stupid)
size=${#myvar}
echo $size
Aqui estão algumas maneiras de calcular o comprimento da variável:
echo ${#VAR}
echo -n $VAR | wc -m
echo -n $VAR | wc -c
printf $VAR | wc -m
expr length $VAR
expr $VAR : '.*'
e para definir o resultado em outra variável, apenas atribua o comando acima com aspas a outra variável da seguinte maneira:
otherVar=`echo -n $VAR | wc -m`
echo $otherVar
http://techopsbook.blogspot.in/2017/09/how-to-find-length-of-string-variable.html