Каков правильный способ преобразования первого символа строки из верхнего в нижний регистр в сценарии оболочки в Mac OS? [Дубликат]

Если у вас есть законная необходимость запускать некоторые js из partial, вот как вы могли это сделать, требуется jQuery:

<script type="text/javascript">        
    function scriptToExecute()
    {
        //The script you want to execute when page is ready.           
    }

    function runWhenReady()
    {
        if (window.$)
            scriptToExecute();                                   
        else
            setTimeout(runWhenReady, 100);
    }
    runWhenReady();
</script>
106
задан MinhTri 19 September 2016 в 15:33
поделиться

11 ответов

foo="$(tr '[:lower:]' '[:upper:]' <<< ${foo:0:1})${foo:1}"
97
ответ дан Michael Hoffman 17 August 2018 в 16:03
поделиться
  • 1
    Несмотря на то, что он более сложный, чем лучший набранный ответ, он фактически выполняет именно это: «верхний регистр первого символа в переменной». Лучший полученный ответ не имеет таких результатов. Похоже, что простые ответы предпочтительнее, чем правильные? – Krzysztof Jabłoński 26 February 2016 в 16:48
  • 2
    @ KrzysztofJabłoński: Собственно, «лучший набранный ответ» ниже, будут давать те же результаты, что и этот ответ под Bash 4. Этот ответ имеет накладные расходы на вызов подоболочки (другая оболочка), накладные расходы на вызов утилиты tr (другой процесс, а не Bash), накладные расходы на использование здесь -doc / string (временное создание файла), и он использует две замены по сравнению с лучшими набранными ответами. Если вы абсолютно должны писать устаревший код, рассмотрите возможность использования функции вместо подоболочки. – Steve 13 April 2016 в 23:57
  • 3
    @ItaiHanski OP не указал, что должно случиться с остальными персонажами, только с первым. Предположим, что они хотели изменить notQuiteCamel на NotQuiteCamel. Ваш вариант имеет побочный эффект или принуждение остальных к нижнему регистру - за счет удвоения количества подклассов и tr процессов. – Jesse Chisholm 27 July 2016 в 20:54
  • 4
    Работает только на ASCII – Bell 27 February 2017 в 01:23
  • 5
    @DanieleOrlando, правда, но у этого вопроса нет тегов, кроме bash. – Charles Duffy 10 March 2017 в 04:48

для вашей ситуации

a='one two three'
a="${a^}"
printf '%s\n' "$a"
# One two three

для каждого слова заглавные буквы первой буквы

a='one two three'
b=( $a ) # $a without quotes
for word in "${b[@]}"; do
    c+=( "${word^}" )
done
a="${c[*]}"
printf '%s\n' "$a"
# One Two Three
0
ответ дан Andrey Karpov 17 August 2018 в 16:03
поделиться

Вот способ «родных» текстовых инструментов:

#!/bin/bash

string="abcd"
first=`echo $string|cut -c1|tr [a-z] [A-Z]`
second=`echo $string|cut -c2-`
echo $first$second
9
ответ дан Equin0x 17 August 2018 в 16:03
поделиться
  • 1
    наконец, что-то, что работает с переменной и на Mac! Спасибо! – OZZIE 20 January 2017 в 17:02
  • 2
    @DanieleOrlando, не такой портативный, как можно было бы надеяться. tr [:lower:] [:upper:] - лучший выбор, если он должен работать в языке / локалях, включая буквы, которые не находятся в диапазонах a-z или A-Z. – Charles Duffy 8 March 2017 в 19:01
  • 3
    @CharlesDuffy true, ваша точка должна быть частью ответа. – Daniele Orlando 9 March 2017 в 11:26
$ foo="bar";
$ foo=`echo ${foo:0:1} | tr  '[a-z]' '[A-Z]'`${foo:1}
$ echo $foo
Bar
16
ответ дан Majid Laissi 17 August 2018 в 16:03
поделиться

Один способ с bash (версия 4 +):

foo=bar
echo "${foo^}"

печатает:

Bar
216
ответ дан Michael Myers 17 August 2018 в 16:03
поделиться
  • 1
    Есть ли противоположность этому? – CMCDragonkai 16 December 2015 в 13:06
  • 2
    @CMCDragonkai: для ввода первой буквы используйте "${foo,}". Для ввода всех букв используйте "${foo,,}". Для прописных букв используйте буквы "${foo^^}". – Steve 16 December 2015 в 22:37
  • 3
    Как это работает для ${foo:?}? – A-B-B 21 October 2016 в 00:53
  • 4
    @ A-B-B: Я не уверен, что вы имеете в виду, но вам, вероятно, придется назначить другую переменную, если вы пытаетесь подстроить foo. – Steve 21 October 2016 в 00:59
  • 5
    Nice, работает и на UTF-8 – Bell 27 February 2017 в 01:24

Это можно сделать и в чистом bash с bash-3.2:

# First, get the first character.
fl=${foo:0:1}

# Safety check: it must be a letter :).
if [[ ${fl} == [a-z] ]]; then
    # Now, obtain its octal value using printf (builtin).
    ord=$(printf '%o' "'${fl}")

    # Fun fact: [a-z] maps onto 0141..0172. [A-Z] is 0101..0132.
    # We can use decimal '- 40' to get the expected result!
    ord=$(( ord - 40 ))

    # Finally, map the new value back to a character.
    fl=$(printf '%b' '\'${ord})
fi

echo "${fl}${foo:1}"
4
ответ дан Michał Górny 17 August 2018 в 16:03
поделиться
  • 1
    не работает с буквами, отличными от ascii, как ó – Jasen 6 May 2016 в 23:08
  • 2
    как определить foo? Я пробовал foo = $1, но я получаю только -bash: foo: command not found – OZZIE 20 January 2017 в 17:01
  • 3
    @OZZIE, нет пробелов вокруг = в назначениях. Это то, что shellcheck.net найдет для вас программно. – Charles Duffy 8 March 2017 в 19:00
  • 4
    Я всегда забываю о сценариях оболочки .. только язык, где имеет место одно пространство (до / после) ... sigh (python по крайней мере 4, если я правильно помню) – OZZIE 9 March 2017 в 08:19
  • 5
    @OZZIE, it имеет значение, потому что, если это не имеет значения, вы не можете передать = в качестве аргумента в программу (как это должно знать, работает ли someprogram = someprogram или присвоение переменной с именем someprogram?) – Charles Duffy 10 March 2017 в 04:38

Что делать, если первый символ не является буквой (но есть вкладка, пробел и скрытая двойная кавычка)? Нам лучше проверить это, пока мы не найдем письмо! Итак:

S='  \"ó foo bar\"'
N=0
until [[ ${S:$N:1} =~ [[:alpha:]] ]]; do N=$[$N+1]; done
#F=`echo ${S:$N:1} | tr [:lower:] [:upper:]`
#F=`echo ${S:$N:1} | sed -E -e 's/./\u&/'` #other option
F=`echo ${S:$N:1}
F=`echo ${F} #pure Bash solution to "upper"
echo "$F"${S:(($N+1))} #without garbage
echo '='${S:0:(($N))}"$F"${S:(($N+1))}'=' #garbage preserved

Foo bar
= \"Foo bar=
-4
ответ дан Roger 17 August 2018 в 16:03
поделиться
  • 1
    Пожалуйста, напишите более чистый код! вам не хватает кавычек повсюду. Вы используете устаревший синтаксис (backticks); вы используете устаревший синтаксис ($[...]). Вы используете много бесполезных круглых скобок. Вы предполагаете, что строка состоит из символов латинского алфавита (как насчет акцентированных букв или букв в других алфавитах?). У вас есть много работы, чтобы сделать этот снипп полезным! (и поскольку вы используете регулярное выражение, вы можете также использовать регулярное выражение, чтобы найти первую букву вместо цикла). – gniourf_gniourf 8 March 2017 в 19:05
  • 2
    Я думаю, вы ошибаетесь. Это обязательный комментарий, который сопровождает downvote, объясняя все неправильные шаблоны и неправильные представления ответа. Пожалуйста, учитесь на своих ошибках (и до этого, попытайтесь понять их) вместо того, чтобы быть упрямым. Быть хорошим человеком также несовместимо с распространением плохих практик. – gniourf_gniourf 9 March 2017 в 10:44
  • 3
    Или, если вы используете Bash ≥4, вам не нужно tr: if [[ $S =~ ^([^[:alpha:]]*)([[:alpha:]].*) ]]; then out=${BASH_REMATCH[1]}${BASH_REMATCH[2]^}; else out=$S; fi; printf '%s\n' "$out" – gniourf_gniourf 9 March 2017 в 10:51
  • 4
    Этот код по-прежнему сильно нарушен. Он по-прежнему использует backticks вместо современного синтаксиса подстановки; он все еще имеет непревзойденные кавычки; он по-прежнему не имеет котировок в тех местах, где они действительно имеют значение ; и т. д. Попытайтесь поместить в тестовые данные некоторые пробельные символы в виде пробелов, если вы не считаете, что эти недостающие котировки имеют значение, и исправить проблемы shellcheck.net , пока этот код не станет чистым , – Charles Duffy 10 March 2017 в 04:40
  • 5
    Что касается языка, который вы указали из unix.stackexchange.com/questions/68694/… , то, что вы пропустите, заключается в том, что echo $foo является примером ситуации, когда парсер ожидает список слов ; поэтому в вашем echo ${F} содержимое F является разделенным по строкам и glob-расширенным. Это аналогично для примера echo ${S:$N:1} и всего остального. См. BashPitfalls # 14 . – Charles Duffy 10 March 2017 в 04:45

Это работало для меня:

Поиск всего * php-файла в текущем каталоге и замена первого символа каждого имени файла на заглавную букву:

например: test. php => Test.php

for f in *php ; do mv "$f" "$(\sed 's/.*/\u&/' <<< "$f")" ; done
0
ответ дан Shemeshey 17 August 2018 в 16:03
поделиться
  • 1
    Первоначальный вопрос был строкой и не ссылался на переименование файлов. – AndySavage 20 February 2017 в 19:45

Один из способов с sed:

echo "$(echo "$foo" | sed 's/.*/\u&/')"

Отпечатки:

Bar
24
ответ дан Steve 17 August 2018 в 16:03
поделиться
  • 1
    не работает на MacOS10 Lion sed, вздох, \ u расширяется до u вместо того, чтобы быть оператором. – vwvan 7 June 2014 в 02:36
  • 2
    @vwvan brew install coreutils gnu-sed и следуйте инструкциям, чтобы использовать команды без префикса g. Для чего это стоит, я никогда не хотел запускать OSX-версию этих команд с момента получения версий GNU. – Dean 26 August 2014 в 20:01
  • 3
    @vwvan: Да, извините, вам понадобится GNU sed в этом случае – Steve 26 August 2014 в 23:40
  • 4
    @Dean: FWIW, я никогда не хотел запускать OSX :-) – Steve 26 August 2014 в 23:41
  • 5
    Просто измените его на sed 's/./\U&/, и он будет работать на POSIX sed. – David Conrad 26 May 2017 в 15:29

Использование awk только

foo="uNcapItalizedstrIng"
echo $foo | awk '{print toupper(substr($0,0,1))tolower(substr($0,2))}'
4
ответ дан toske 17 August 2018 в 16:03
поделиться
first-letter-to-lower () {
        str="" 
        space=" " 
        for i in $@
        do
                if [ -z $(echo $i | grep "the\|of\|with" ) ]
                then
                        str=$str"$(echo ${i:0:1} | tr  '[A-Z]' '[a-z]')${i:1}$space" 
                else
                        str=$str${i}$space 
                fi
        done
        echo $str
}
first-letter-to-upper-xc () {
        v-first-letter-to-upper | xclip -selection clipboard
}
first-letter-to-upper () {
        str="" 
        space=" " 
        for i in $@
        do
                if [ -z $(echo $i | grep "the\|of\|with" ) ]
                then
                        str=$str"$(echo ${i:0:1} | tr  '[a-z]' '[A-Z]')${i:1}$space" 
                else
                        str=$str${i}$space 
                fi
        done
        echo $str
}

first-letter-to-lower-xc () {v-first-letter-to-lower | xclip -selection clipboard}

-1
ответ дан user123456 17 August 2018 в 16:03
поделиться
Другие вопросы по тегам:

Похожие вопросы: