하나의 루프에서 두 개의 시퀀스를 통해 실행 시퀀스를 실행하려고합니다. #!/bin/bash for i in (1..15)

아래처럼 쉘에서 동일한 루프에서 두 개의 시퀀스를 실행하려고합니다.

#!/bin/bash
for i in (1..15) and (20..25) ;
do
     echo $i
     ......
     .....other process
done

어떻게 이것을 달성 할 수 있습니까?



답변

이를 위해 중괄호 확장 만 필요합니다.

$ for n in {1..3} {200..203}; do echo $n; done
1
2
3
200
201
202
203

for( )에 목록을 전달할 수 있습니다 .for i in x y z; do stuff "$i"; done

따라서 여기서 중괄호 { }는 쉘이 시퀀스를 목록으로 확장하도록합니다. 쉘은 인수 목록을 분할하기 때문에 공백을 둘 필요가 있습니다.


답변

또는 우리는 seq( 일련의 숫자 인쇄) 를 사용할 수 있습니다 .

for i in `seq 1 3` `seq 101 103`; do echo $i; done
for i in $(seq 1 3) $(seq 101 103); do echo $i; done

반복적 인 작업을위한 스크립트 인 경우 함수를 사용할 수 있습니다.

#!/bin/bash
my_function() { echo "$1"; }
for i in {1..3}; do my_function "$i"; done
for i in {101..103}; do my_function "$i"; done
#!/bin/bash
my_function() { for i in `seq $1 $2`; do echo "$i"; done; }
my_function "1" "3"
my_function "101" "103"

답변

Zanna의 대답pa4080의 대답 은 모두 좋으며 대부분의 상황에서 그중 하나와 함께 갈 것입니다. 아마도 말할 필요도 없지만 완전성을 위해 어쨌든 말할 것입니다. 각 값을 배열에로드 한 다음 배열을 반복 할 수 있습니다. 예를 들면 다음과 같습니다.

the_array=( 1 2 3 4 5 6 7 8 9 10 20 21 22 23 24 25 )
for i in "${the_array[@]}";
do
    echo $i
done

답변

루프없이 루핑

Zanna의 대답은 절대적으로 정확하고 bash에 적합하지만 루프를 사용하지 않고도 더 향상시킬 수 있습니다.

printf "%d\n"  {1..15} {20..25}

의 행동 printf의 수는 경우에하도록이다 ARGUMENTS에서 형식 컨트롤보다 큰 'FORMAT STRING'다음, printf모든 분할합니다 ARGUMENTS 동일한 조각으로하고 때까지 반복해서 형식 문자열에 끼워 보관이 부족 ARGUMENTS목록입니다.

이식성을 위해 노력하고 있다면 printf "%d\n" $(seq 1 15) $(seq 20 25)대신 활용할 수 있습니다

이것을 더 재미있게 보자. 단순히 숫자를 인쇄하는 것이 아니라 행동을 원한다고 가정 해 봅시다. 일련의 숫자 중에서 파일을 만들기 위해 쉽게 할 수 있습니다 touch {1..15}.txt {20..25}.txt. 여러 가지 일이 발생하면 어떻게해야합니까? 우리는 또한 다음과 같이 할 수 있습니다 :

$ printf "%d\n" {1..15} {20..25} | xargs -I % bash -c 'touch "$1.txt"; stat "$1.txt"' sh %

또는 우리가 그것을 오래된 학교 스타일로 만들고 싶다면 :

printf "%d\n" {1..15} {20..25} | while read -r line; do
    touch "$line".txt;
    stat "$line".txt;
    rm "$line".txt;
done

휴대용이지만 자세한 대안

중괄호 확장이없는 쉘 (이것이 {1..15} {20..25}의존 하는 쉘)에서 작동하는 스크립트 솔루션을 만들고 싶다면 간단한 while 루프를 작성할 수 있습니다.

#!/bin/sh
start=$1
jump=$2
new_start=$3
end=$4

i=$start
while [ $i -le $jump ]
do
    printf "%d\n" "$i"
    i=$((i+1))
    if [ $i -eq $jump ] && ! [ $i -eq $end ];then
        printf "%d\n" "$i"
        i=$new_start
        jump=$end
    fi
done

물론이 솔루션은 더 장황하고 일부는 단축 될 수 있지만 작동합니다. 테스트 ksh, dash, mksh, 물론 bash.


배쉬 C 스타일 루프

그러나 우리가 루프 배쉬 특정 (어떤 이유로 든 인쇄뿐만 아니라 그 숫자로 무언가를하는)을 만들고 싶다면 이것을 할 수도 있습니다 (기본적으로 휴대용 솔루션의 C 루프 버전).

last=15; for (( i=1; i<=last;i++ )); do printf "%d\n" "$i"; [[ $i -eq $last ]] && !  [[ $i -eq 25 ]] && { i=19;last=25;} ;done

또는 더 읽기 쉬운 형식으로 :

last=15
for (( i=1; i<=last;i++ ));
do
    printf "%d\n" "$i"
    [[ $i -eq $last ]] && !  [[ $i -eq 25 ]] && { i=19;last=25;}
done

다른 루핑 접근법의 성능 비교

bash-4.3$ time bash -c 'printf "%d\n" {0..50000}>/dev/null'

real    0m0.196s
user    0m0.124s
sys 0m0.028s
bash-4.3$ time bash -c 'for i in {1..50000}; do echo $i > /dev/null; done'

real    0m1.819s
user    0m1.328s
sys 0m0.476s
bash-4.3$ time bash -c ' i=0;while [ $i -le 50000 ]; do echo $i>/dev/null; i=$((i+1)); done'

real    0m3.069s
user    0m2.544s
sys 0m0.500s
bash-4.3$ time bash -c 'for i in $(seq 1 50000); do printf "%d\n" > /dev/null; done'

real    0m1.879s
user    0m1.344s
sys 0m0.520s

비쉘 대안

우리가 여기 파이썬 솔루션이 될 수 있기 때문에

$ python3 -c 'print("\n".join([str(i) for i in (*range(1,16),*range(20,26))]))'

또는 약간의 껍질로 :

bash-4.3$ python3 << EOF
> for i in (*range(16),*range(20,26)):
>    print(i)
> EOF