Опубликован: 05.07.2006 | Доступ: свободный | Студентов: 4679 / 885 | Оценка: 4.12 / 3.74 | Длительность: 18:59:00
Лекция 6:

Указатели и массивы

5.2. Указатели и аргументы функций

Так как в "с" передача аргументов функциям осуществляется "по значению", вызванная процедура не имеет непосредственной возможности изменить переменную из вызывающей программы. Что же делать, если вам действительно надо изменить аргумент? Например, программа сортировки захотела бы поменять два нарушающих порядок элемента с помощью функции с именем swap. Для этого недостаточно написать

swap(a, b);

определив функцию swap при этом следующим образом:

swap(x, y)      /* wrong */
int x, y;
{
   int temp;

   temp = x;
   x = y;
   y = temp;
}

из-за вызова по значению swap не может воздействовать на аргументы a и b в вызывающей функции.

К счастью, все же имеется возможность получить желаемый эффект. Вызывающая программа передает указатели подлежащих изменению значений: swap(&a, &b) ; так как операция & выдает адрес переменной, то &a является указателем на a. В самой swap аргументы описываются как указатели и доступ к фактическим операндам осуществляется через них.

swap(px, py)    /* interchange *px and *py */
int *px, *py;
{
   int temp;

   temp = *px;
   *px = *py;
   *py = temp;
}

Указатели в качестве аргументов обычно используются в функциях, которые должны возвращать более одного значения. (Можно сказать, что swap возвращает два значения, новые значения ее аргументов ). В качестве примера рассмотрим функцию getint, которая осуществляет преобразование поступающих в свободном формате данных, разделяя поток символов на целые значения, по одному целому за одно обращение. функция getint должна возвращать либо найденное значение, либо признак конца файла, если входные данные полностью исчерпаны. Эти значения должны возвращаться как отдельные объекты, какое бы значение ни использовалось для EOF, даже если это значение вводимого целого.

Одно из решений, основывающееся на описываемой в "лекции №7" функции ввода scanf, состоит в том, чтобы при выходе на конец файла getint возвращала EOF в качестве значения функции ; любое другое возвращенное значение говорит о нахождении нормального целого. Численное же значение найденного целого возвращается через аргумент, который должен быть указателем целого. Эта организация разделяет статус конца файла и численные значения.

Следующий цикл заполняет массив целыми с помощью обращений к функции

getint: 
int n, v, array[size]; 

for (n = 0; n < size && getint(&v) != EOF; n++) 
      array[n] = v;

В результате каждого обращения v становится равным следующему целому значению, найденному во входных данных. Обратите внимание, что в качестве аргумента getint необходимо указать &v а не v. Использование просто v скорее всего приведет к ошибке адресации, поскольку getint полагает, что она работает именно с указателем.

Сама getint является очевидной модификацией написанной нами ранее функции atoi:

getint(pn)    /* get next integer from input */
int *pn;
{
  int c,sign;

  while ((c = getch()) == ' ' || c == '\n'
  || c == '\t'); /* skip white space */
  sign = 1;
  if (c == '+' || c == '-') { /* record
     sign */
     sign = (c == '+') ? 1 : -1;
     c = getch();
  }
  for (*pn = 0; c >= '0' && c <= '9'; c = getch())
     *pn = 10 * *pn + c - '0';
  *pn *= sign;
  if (c != EOF)
     ungetch(c);
  return(c);
}

Выражение *pn используется всюду в getint как обычная переменная типа int. Мы также использовали функции getch и ungetch ( описанные в "лекции №4" ) , так что один лишний символ, который приходится считывать, может быть помещен обратно во ввод.

Упражнение 5-1

Напишите функцию getfloat, аналог getint для чисел с плавающей точкой. Какой тип должна возвращать getfloat в качестве значения функции?