ฉันต้องการตัวพิมพ์ใหญ่เพียงตัวอักษรตัวแรกในสตริงของฉันด้วยการทุบตี
foo="bar";
//uppercase first character
echo $foo;
ควรพิมพ์ "บาร์";
ฉันต้องการตัวพิมพ์ใหญ่เพียงตัวอักษรตัวแรกในสตริงของฉันด้วยการทุบตี
foo="bar";
//uppercase first character
echo $foo;
ควรพิมพ์ "บาร์";
คำตอบ:
foo="$(tr '[:lower:]' '[:upper:]' <<< ${foo:0:1})${foo:1}"
notQuiteCamel
NotQuiteCamel
ชุดตัวเลือกของคุณมีผลข้างเคียงหรือบังคับให้ส่วนที่เหลือเป็นตัวพิมพ์เล็ก - ด้วยค่าใช้จ่ายในการเพิ่มจำนวนของ sub shells และกระบวนการ tr
bash
นี้
วิธีหนึ่งในการทุบตี (เวอร์ชัน 4+):
foo=bar
echo "${foo^}"
พิมพ์:
Bar
"${foo,}"
เป็นตัวพิมพ์เล็กตัวอักษรตัวแรกใช้ "${foo,,}"
เป็นตัวพิมพ์เล็กทุกตัวอักษรที่ใช้ "${foo^^}"
เป็นตัวพิมพ์ใหญ่ทั้งหมดตัวอักษรที่ใช้
${foo~}
และ${foo~~}
มีผลเช่นเดียวกับ และ${foo^}
${foo^^}
แต่ฉันไม่เคยเห็นทางเลือกที่กล่าวถึงทุกที่
bad substitution
ทางเดียวกับsed
:
echo "$(echo "$foo" | sed 's/.*/\u&/')"
พิมพ์:
Bar
brew install coreutils gnu-sed
g
สำหรับสิ่งที่คุ้มค่าฉันไม่ต้องการเรียกใช้คำสั่งเหล่านี้ในเวอร์ชัน OSX ตั้งแต่รับรุ่น GNU
sed
ในกรณีนี้
sed 's/./\U&/
และมันจะทำงานบน POSIX sed
$ foo="bar";
$ foo=`echo ${foo:0:1} | tr '[a-z]' '[A-Z]'`${foo:1}
$ echo $foo
Bar
นี่คือวิธีใช้เครื่องมือข้อความ "ดั้งเดิม":
#!/bin/bash
string="abcd"
first=`echo $string|cut -c1|tr [a-z] [A-Z]`
second=`echo $string|cut -c2-`
echo $first$second
tr [:lower:] [:upper:]
เป็นตัวเลือกที่ดีกว่าหากต้องการทำงานเป็นภาษา / โลแคลที่รวมตัวอักษรที่ไม่อยู่ในa-z
หรือA-Z
ช่วง
ใช้ awk เท่านั้น
foo="uNcapItalizedstrIng"
echo $foo | awk '{print toupper(substr($0,0,1))tolower(substr($0,2))}'
มันสามารถทำได้ใน 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}"
foo = $1
แต่ฉันเพิ่งได้รับ-bash: foo: command not found
=
ในงานที่ได้รับมอบหมาย นี่คือสิ่งที่shellcheck.netจะค้นหาให้คุณตามโปรแกรม
=
ข้อโต้แย้งไปยังโปรแกรม (มันจะรู้ได้อย่างไรว่าsomeprogram =
กำลังทำงานอยู่someprogram
หรือมอบหมายให้ตัวแปรชื่อsomeprogram
)
มันก็ใช้ได้เช่นกัน ...
FooBar=baz
echo ${FooBar^^${FooBar:0:1}}
=> Baz
FooBar=baz
echo ${FooBar^^${FooBar:1:1}}
=> bAz
FooBar=baz
echo ${FooBar^^${FooBar:2:2}}
=> baZ
และอื่น ๆ
แหล่งที่มา:
Inroductions / Tutorials:
หากต้องการใช้คำแรกให้พิมพ์ใหญ่เท่านั้น:
foo='one two three'
foo="${foo^}"
echo $foo
โอ้สองสาม
ในการใช้ประโยชน์ทุก ๆ คำในตัวแปร:
foo="one two three"
foo=( $foo ) # without quotes
foo="${foo[@]^}"
echo $foo
O ne T wo T hree
(ทำงานใน bash 4+)
foo='one two three'; foo=$(for i in $foo; do echo -n "${i^} "; done)
ความละเอียดที่สั้นกว่าสำหรับทุกคำ fooตอนนี้คือ "One Two Three"
ทางเลือกและวิธีการแก้ปัญหาที่สะอาดสำหรับทั้ง Linux และ OSX มันยังสามารถใช้กับตัวแปร bash
python -c "print(\"abc\".capitalize())"
ส่งคืน Abc
อันนี้ใช้ได้สำหรับฉัน:
ค้นหาไฟล์ * php ทั้งหมดในไดเรกทอรีปัจจุบันและแทนที่อักขระตัวแรกของชื่อไฟล์แต่ละชื่อเป็นตัวพิมพ์ใหญ่:
เช่น: test.php => Test.php
for f in *php ; do mv "$f" "$(\sed 's/.*/\u&/' <<< "$f")" ; done
เพื่อความสนุกที่นี่คุณคือ:
foo="bar";
echo $foo | awk '{$1=toupper(substr($1,0,1))substr($1,2)}1'
# or
echo ${foo^}
# or
echo $foo | head -c 1 | tr [a-z] [A-Z]; echo $foo | tail -c +2
# or
echo ${foo:1} | sed -e 's/^./\B&/'
ไม่ตรงกับสิ่งที่ถาม แต่มีประโยชน์มาก
declare -u foo #When the variable is assigned a value, all lower-case characters are converted to upper-case.
foo=bar
echo $foo
BAR
และตรงกันข้าม
declare -l foo #When the variable is assigned a value, all upper-case characters are converted to lower-case.
foo=BAR
echo $foo
bar
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
}
ตัวอักษรตัวแรกถึงล่าง xc () {v-first-letter-to-lower-xc | คลิปบอร์ด xclip - เลือก}
ถ้าอักขระตัวแรกไม่ใช่ตัวอักษร (แต่เป็นแท็บเว้นวรรคและเครื่องหมายคำพูดคู่ที่หลบหนี) เราควรทดสอบมันจนกว่าเราจะพบจดหมาย! ดังนั้น:
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=
$[...]
) คุณใช้วงเล็บที่ไร้ประโยชน์มากมาย คุณสมมติว่าสตริงนั้นประกอบด้วยตัวอักษรจากตัวอักษรละติน (ตัวอักษรเน้นเสียงหรือตัวอักษรในตัวอักษรอื่น ๆ ) คุณมีงานจำนวนมากที่จะทำให้ข้อมูลโค้ดนี้ใช้งานได้! (และเนื่องจากคุณใช้ regex คุณอาจใช้ regex เพื่อค้นหาตัวอักษรตัวแรกแทนที่จะวนซ้ำ)
tr
:if [[ $S =~ ^([^[:alpha:]]*)([[:alpha:]].*) ]]; then out=${BASH_REMATCH[1]}${BASH_REMATCH[2]^}; else out=$S; fi; printf '%s\n' "$out"
echo $foo
เป็นตัวอย่างของสถานการณ์ที่หนึ่งแยกวิเคราะห์คาดว่ารายการของคำ ; ดังนั้นในecho ${F}
เนื้อหาของคุณF
คือการแยกสตริงและการขยายแบบกลม นั่นเป็นความจริงที่คล้ายกันสำหรับecho ${S:$N:1}
ตัวอย่างและส่วนที่เหลือทั้งหมด ดูBashPitfalls #