Як порівняти версію програми в сценарії оболонки?


14

Припустимо, я хочу порівняти gccверсію, щоб побачити, чи встановлена ​​в системі мінімальна версія чи ні.

Щоб перевірити gccверсію, я виконав наступне

gcc --version | head -n1 | cut -d" " -f4

Вихід був

4.8.5

Отже, я написав просту ifзаяву, щоб перевірити цю версію на якусь іншу цінність

if [ "$(gcc --version | head -n1 | cut -d" " -f4)" -lt 5.0.0 ]; then
    echo "Less than 5.0.0"
else
    echo "Greater than 5.0.0"
fi

Але це видає помилку:

[: integer expression expected: 4.8.5

Я зрозумів свою помилку в тому, що я використовував рядки для порівняння та -ltвимагає цілого числа. Отже, чи є інший спосіб порівняння версій?


@ 123 Нічого не відбувається
Абхіманья Сахаран

1
Також є запитання щодо переповнення стека з купою різних пропозицій щодо порівняння рядків версій.
пр

1
Набагато простіше, ніж використовувати труби:gcc -dumpversion
Віктор Ламойн

Відповіді:


22

Я не знаю, чи це красиво, але він працює для кожного відомого формату версії.

#!/bin/bash
currentver="$(gcc -dumpversion)"
requiredver="5.0.0"
 if [ "$(printf '%s\n' "$requiredver" "$currentver" | sort -V | head -n1)" = "$requiredver" ]; then 
        echo "Greater than or equal to 5.0.0"
 else
        echo "Less than 5.0.0"
 fi

( Примітка: краща версія користувача "wildcard": https://unix.stackexchange.com/users/135943/wildcard , видалено додаткову умову)


3
Спочатку я вважав це жахливим, а потім зрозумів, що краса сценаріїв оболонок полягає саме у зловживанні подібними інструментами. +1
Бойкот SE для Моніки Селліо

2
Це руйнується, якщо в операторі друку є знаки "%". Краще замінити printf "$requiredver\n$currentver"на printf '%s\n' "$requiredver" "$currentver".
phk

1
-Vє розширенням GNU, sort(1)тому це рішення не є портативним.
stefanct

1
sort -nпрацює майже так само, як у випадку чисельних версій.
Rockallite

1
@LucianoAndressMartini, подивись, що ти думаєш про мою редакцію.
Wildcard

2

Тут я даю рішення для порівняння версій Unix Kernel. І це має працювати для інших, таких як gcc. Мене цікавить лише номер першої 2 версії, але ви можете додати ще один рівень логіки. Це один лайнер, і я написав його в декількох рядках для розуміння.

check_linux_version() {
    version_good=$(uname -r | awk 'BEGIN{ FS="."}; 
    { if ($1 < 4) { print "N"; } 
      else if ($1 == 4) { 
          if ($2 < 4) { print "N"; } 
          else { print "Y"; } 
      } 
      else { print "Y"; }
    }')

    #if [ "$current" \< "$expected" ]; then
    if [ "$version_good" = "N" ]; then
        current=$(uname -r)
        echo current linux version too low
        echo current Linux: $current
        echo required 4.4 minimum
        return 1
    fi
}

Ви можете змінити це та використовувати його для перевірки версії gcc.


+1 він сумісний з іншими подібними до Unix? (Моє рішення - не)
Лучано Андресс Мартіні

1

Раніше ми перевіряли версії у файлі GNU. Ми виділялися через засоби Makefile. Нам довелося виявляти старі компілятори Binutils та баггі та обходити їх на ходу.

Ми використовували схему:

#!/usr/bin/env bash

CC=$(command -v gcc)
GREP=$(command -v grep)

# Fixup CC and GREP as needed. It may be needed on AIX, BSDs, and Solaris
if [[ -f "/usr/gnu/bin/grep" ]]; then
    GREP="/usr/gnu/bin/grep"
elif [[ -f "/usr/linux/bin/grep" ]]; then
    GREP="/usr/linux/bin/grep"
elif [[ -f "/usr/xpg4/bin/grep" ]]; then
    GREP="/usr/xpg4/bin/grep"
fi

# Check compiler for GCC 4.8 or later
GCC48_OR_LATER=$("$CXX" -v 2>&1 | "$GREP" -i -c -E "gcc version (4\.[8-9]|[5-9]\.)")
if [[ "$GCC48_OR_LATER" -ne 0 ]];
then
   ...
fi

# Check assembler for GAS 2.19 or later
GAS219_OR_LATER=$("$CXX" -xc -c /dev/null -Wa,-v -o/dev/null 2>&1 | "$GREP" -c -E "GNU assembler version (2\.19|2\.[2-9]|[3-9])")
if [[ "$GAS219_OR_LATER" -ne 0 ]];
then
   ...
fi

Приємно! Це працює і дуже легко розширюється, і дуже портативно!
Бред Паркс

1

Коротша версія:

version_greater_equal()
{
    printf '%s\n%s\n' "$2" "$1" | sort -V -C
}

version_greater_equal "${gcc_version}" 8.2 || die "need 8.2 or above"

(1) Це незначна варіація вже заданих відповідей. Ви можете додати значення, додавши пояснення, яке ще не було розміщено. (2)  printf '%s\n'досить хороший; printfповторить рядок формату, якщо потрібно.
G-Man каже: "Відновіть Моніку"

Я зазвичай вважаю за краще редагувати наявні відповіді, але видалити половину з них складно: інші можуть бачити значення там, де я цього не роблю. Те саме для багатослівних пояснень. Менше значить більше.
Марч

Я знаю, що printfповторює рядок формату, але мені (відсутність!) Синтаксису для цього є IMHO незрозумілий; тому я використовую це лише тоді, коли потрібно = коли кількість аргументів велика або змінна.
Марч

1
function version_compare () {
  function sub_ver () {
    local len=${#1}
    temp=${1%%"."*} && indexOf=`echo ${1%%"."*} | echo ${#temp}`
    echo -e "${1:0:indexOf}"
  }
  function cut_dot () {
    local offset=${#1}
    local length=${#2}
    echo -e "${2:((++offset)):length}"
  }
  if [ -z "$1" ] || [ -z "$2" ]; then
    echo "=" && exit 0
  fi
  local v1=`echo -e "${1}" | tr -d '[[:space:]]'`
  local v2=`echo -e "${2}" | tr -d '[[:space:]]'`
  local v1_sub=`sub_ver $v1`
  local v2_sub=`sub_ver $v2`
  if (( v1_sub > v2_sub )); then
    echo ">"
  elif (( v1_sub < v2_sub )); then
    echo "<"
  else
    version_compare `cut_dot $v1_sub $v1` `cut_dot $v2_sub $v2`
  fi
}

### Usage:

version_compare "1.2.3" "1.2.4"
# Output: <

Кредит належить @Shellman

Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.