Warning: Cannot use a scalar value as an array in /home/admin/public_html/forum/include/fm.class.php on line 757
Warning: Invalid argument supplied for foreach() in /home/admin/public_html/forum/include/fm.class.php on line 770
Warning: Invalid argument supplied for foreach() in /home/admin/public_html/forum/topic.php on line 737 Форумы портала PHP.SU :: Уроки № 3 - Знакомсво с функциями
Покинул форум
Сообщений всего: 4350
Дата рег-ции: Авг. 2008 Откуда: Москва
Помог: 57 раз(а)
Уроки № 3 - Знакомство с функциями
Теперь, когда вы уже знакомы с переменными, операциями над ними, познакомились с управляющими структурами языка, можно поговорить и о функциях.
Бывают такие ситуации, что какую-то последовательность инструкций вы повторяете много раз. И эта последовательность выполняет в конечном итоге что-то вполне завершенное.
Функция это кучка действий. Она принимает аргументы, делает с ними некие операции и возвращает результат. К примеру Приготовить пиццу это функция принимающая параметры - тесто, яйца, сыр, кетчуп, анчоусы, плиту, и возвращает Пиццу.
Возвращаемое значение у функций может быть любого типа данных.
Еще одним преимуществом функций, является упрощение чтения кода программы.
Имя функции может состоять из латинских буков, цифр и знаков подчеркивания. Начинаться с цифры оно не должно. Всё точно так же как с названиями переменных.
Возможно вы заметили, что уже с первого урока мы путаем вас понятиями
параметр и аргумент.
Аргументами называются неопределенные входящие данные.
Параметрами же являются известные данные.
Т.е. когда мы описываем функцию, в скобках аргументы, когда вызываем - параметры. Но застопориваться на этом, думаю, не обязательно.
В функции которая готовит пиццу - аргументом будет "количество требуемых блюд"
а параметром будет цифра 5.
Во всяком случае во всех случаях речь идёт о данных передаваемых функции.
Champion
Отправлено: 16 Января, 2009 - 11:36:11
Активный участник
Покинул форум
Сообщений всего: 4350
Дата рег-ции: Авг. 2008 Откуда: Москва
Помог: 57 раз(а)
Рассмотрим создание самодельной функции на примере.
Задача функции найти самый большой элемент массива.
echo'$ar1[1]'.$ar1[1];// выведет 6 (а не 95, помните ту загадочную строку функции?). Получается значение не изменилось
// Напомним: в одинарных кавычках значения переменных не подставляются
// подумайте над результатами такого вызова:
echo arr_max($ar2);
?>
Область видимости
Как же так получается что операция присваивания $arr[1] = 95
не изменила значение переменной вне функции?
Существует такое понятие - Область видимости. Понять это понятие легко и понятно =)
Функция представляет из себя своего рода маленькую подпрограмму.
У этой подпрограммы есть свои переменные, а переменные из главной программы она просто не видит.
Так как в эту программку одним из параметров поступил массив $arr[1]
это значит что функция вполне его видела и могла с ним работать.
Но на самом деле в функцию передаётся на переменная, а её точная копия. Ее значение.
Поэтому когда наша подпрограмма завершает свое выполнение - все её переменные
(копии наших) удаляются.
Передача параметров по ссылке
Напомню тебе немного первый урок, в котором я объяснял как хранит данные в памяти компьютер, и что $variable на самом деле хранит вовсе не значение а идентификатор ячейки в памяти, в которой как и находится наше значение.
Как ты заметил в примере выше, все переменные используемые в функциях являются локальными, то-есть создаются используются и удаляются в самой функции, а за её пределами не доступны.
Для того, чтобы передать в функцию не значение переменной, а саму переменную которую функция сможет изменить и за её пределами мы сможем ею пользоваться - существует особый способ передачи параметров - По ссылке. Выглядит вызов функции с параметрами по ссылке с добавлением символа & вот так: myFunction(&$myvar);
Когда в вызове функции переходит передача переменной по ссылке происходит передача идентификатора ячейки, а не значения. Когда ты вызываешь функцию func($a) интерпретатор в первую очередь достает из памяти значение переменной $a и передает это значение в функцию. После этого в самой функции создается новая переменная которой присваивается это значение.
Когда функция вызывается с параметром по ссылке func(&$a) интерпретатор передает в функцию не значение переменной, а идентификатор ячейки в памяти
При вызове функции с параметрами обычным способом все изменения происходят в другой, новой ячейке памяти и за пределами функции работа продолжается со старой ячейкой. А при передаче по ссылке все изменения происходят в той же ячейке и поэтому изменения видны и из вне самой функции.
Champion
Отправлено: 16 Января, 2009 - 12:07:27
Активный участник
Покинул форум
Сообщений всего: 4350
Дата рег-ции: Авг. 2008 Откуда: Москва
Помог: 57 раз(а)
Оператор return
Наша функция не обязана что либо возвращать конструкцией return.
Возможно, она просто выполняет ряд операций, например,
вывод нескольких строк или еще что-либо.
Например, уже знакомый нам код из предыдущей главы можно оформить так:
elseif($sName=="Valenok")echo("Теперь я точно уверен, это Valenok");
elseif($sName=="Champion")echo("Это тоже наш автор, Champion");
elseecho("Я запутался..");
}
// основная программа
$val='Valenok'; who_is_it($val);
$eu='EuGen'; who_is_it($eu);
$ch='Champion'; who_is_it($ch);
$unknown='Гость'; who_is_it($unknown);
?>
Мы не просили функцию что либо возвращать.. В таком случае интерпретатор возвращает значение Null.
Но бывает, что одно возвращенного значения не хватает. Также возможно, что вы хотите, чтобы изменения, произошедшие с переменными в функции отразились и в основной программе .
Для этого можно воспользоваться двумя решениями.
Либо заставить нашу подпрограмму увидеть переменную извне, либо передать ей оригинал параметра вместо копии.
Поставив значок & перед переменной, мы укажем интерпретатору что мы передаем
не саму переменную, а ссылку на переменную. По сути это можно понимать как тип Resource.
Функция будет уже работать не с копией данных в памяти, а непосредственно с определенной ячейкой в памяти. Таким образов все изменения будут записаны прямиком
в нужный нам участок оперативной памяти и изменения сохранятся.
Пусть функция возвращает сумму 2 слогаемых, в первый параметр запишется произведение, а второй параметр просто удвоится.
Конструкция global позволяет функции увидеть и использовать переменную из внешнего мира.
В данном случае даже не придётся передавать ей параметры, так как она их увидит сама.
Покинул форум
Сообщений всего: 4350
Дата рег-ции: Авг. 2008 Откуда: Москва
Помог: 57 раз(а)
Рекурсия.
Давайте возьмем функцию, которая ищет какой-то элемент ряда Фибоначии(0,1,1,2,3,5,8... - первые два элемента 0 и 1, остальные - сумма двух предыдущих). Напишем ее без рекурсии, потому что пока не знаем что это такое.
function fibonacci($num)// $num - номер интересующего нас элемента
{
if($num<1){// номера элемента меньше 1 не существует, заканчиваем функцию
returnfalse;
}
if($num<=2){// если это один из первых элементов, нетрудно увидеть как они определяются
return($num- 1);
}
// общий случай. Идем от 3го до требуемого номера
$pre_pre=0;// элемент, скажем так, предпредыдущий.
$current=1;// текущий
for($i=3;$i<=$num;$i++){
$pre=$current;// бывший текущий становится предыдущим
$current=$pre+$pre_pre;// определяем текущий элемент
$pre_pre=$pre;// бывший предыдущий становится предпредыдущим
}
return$current;
}
/*** Основная часть программы ***/
$n=5;
echo fibonacci($n);// 0,1,1,2,3 - получается 3
?>
Функция достаточно большая. Давайте посмотрим на нее и увидим, что в этой функции используется цикл. У цикла есть условие выхода и в каждой итерации мы работаем с теми значениями, которые остались после предыдущей итерации. Тогда бывает удобно применять рекурсию. Функция называется рекурсивной, если она вызывает саму себя. Вот как! Давайте перепишем ту функцию с использованием рекурсии.
// $pre и $pre_pre, как и в тот раз - предыдущий и предпредыдущий элемент.
// $n номер элемента, который мы ищем, НО реально смысл в этой переменной несколько другой. Она хранит в себе количество элементов, которое осталось посчитать
// Считать сумму начинаем с 3го элемента.
function fib($n,$pre= 1,$pre_pre= 0)
{
if($n== 1)return1;
if($n< 1)returnfalse;
if($n==2)// начинали с 3го, поэтому выходим, когда осталось посчитать два.
return$pre;
return fib($n- 1,$pre+$pre_pre,$pre);
}
echo fib(5);
?>
Как видите, получилось гораздо меньше строк.
Но рекурсию используют не только вместо циклов. Так поступают нечасто, потому что обычно цикл проще. Бывают задачи, которые решить без рекурсии не возможно. Например, построение дерева файлов и подкаталогов в каталоге. С помощью рекурсии, мы бы написали функцию сименем папки в качестве входного параметра. Функция эта берет элемент и, если это файл, просто выводит его имя, но если это директория, то функция вызывает саму себя для этой директории. Когда вы узнаете функции работы с файловой системой, вы ее напишете)
Как реализовать это без рекурсии... затрудняюсь придумать.
При описании рекурсивных функций необходимо предусмотреть условие выхода из функции, чтобы она не была бесконечной. Как и в цикле.
Да, кстати, видите, что параметров у функции 3, а вызывал ее я в программе с одним параметром? Заметьте, при объявлении функции два последних параметра объявлены следующим видом: $variable = value. Им задано значение по умолчанию. Т.е. при вызове передавать эти параметры не обязательно, если, конечно, они не отличаются от значений по умолчанию. Если отличаются, надо передать. Вы также можете передать только один из таких параметров, можете все. Но если вы передаете не все параметры, как php определяет, какие перменные вы имели в виду? Очень просто слева на право. Если передано меньше значений, чем нужно функции, самые правые переменные примут значение по умолчанию.
Champion
Отправлено: 17 Января, 2009 - 15:36:34
Активный участник
Покинул форум
Сообщений всего: 4350
Дата рег-ции: Авг. 2008 Откуда: Москва
Помог: 57 раз(а)
Теперь вопросы по главе!
1. В каком варианте (вариантах) функция объявлена синтаксически не верно? Почему?
function my_func() {}
function _qwerty($a) {}
function 4func($a = 4) {}
function func($a = 4) {}
function func.my($a) {}
function func-my($a = 4) {}
function int($a = 4) {}
function return($a = 4) {}
2. В каком варианте функция function func($a, $b = 4, &$c, $d = 5, $e = 6) вызвана неправильно? Почему? (все переменные имеют значение, функция возвращает значение)
call func(1, 3, $d)
$d = func(1, 3, 5)
$d = func(1, 3, $f, 6)
echo func($s)
func(1, func(1, 3, $d), $d)
func(1, $d, func(1, 3, $d))
3. Напишите функцию, которая ищет минимум массива и его индекс, чтобы и с индексом и с самим минимумом можно было работать в основной программе.
Покинул форум
Сообщений всего: 82
Дата рег-ции: Июнь 2010 Откуда: Харківська обл. с. Криштопівка
Помог: 0 раз(а)
Существует ли возможность передать переменной имя функции, а потом подставлять эту переменную в двух разных качествах:
1, как текст;
2, как функцию.
Сомневаюсь, что это возможно, но решил спросить...
Пример:
Покинул форум
Сообщений всего: 4350
Дата рег-ции: Авг. 2008 Откуда: Москва
Помог: 57 раз(а)
like_you пишет:
(function_exists(substr_replace ($f,'',-2,2))
Это что?
И что подразумевается здесь под значением функции? (Добавление)
Советую нажать на ссылочки в коде и прочитать описание функций, чтобы уяснить непонятности, которые, по-моему, имеют здесь место.
like_you
Отправлено: 18 Июня, 2010 - 20:20:15
Гость
Покинул форум
Сообщений всего: 82
Дата рег-ции: Июнь 2010 Откуда: Харківська обл. с. Криштопівка
Помог: 0 раз(а)
Champion пишет:
(function_exists(substr_replace ($f,'',-2,2))
функция substr_replace($f,'',-2,2) убирает со значения переменной $f две скопки () справа. Это сделал для функции function_exists, я просто думал, что она со скобками будет плохо работь, но сейчас проверил и удачно работает и так:
(Добавление)
Но всёже интересно узнать возможно ли в переменную вставить функцию, а потом эту переменную использовать где-то в коде.
Да - и под значением функции имеется ввиду булевое значение возвращяемое функцией.
Покинул форум
Сообщений всего: 82
Дата рег-ции: Июнь 2010 Откуда: Харківська обл. с. Криштопівка
Помог: 0 раз(а)
Бомба! Спасибо большое. То что мне было нужно. Логично до этого врядли можно дойти, разве что методом "тыка" или имея солидный опыт с PHP.
Вообще респект Вам и уважуга! Что сайт суперский, что на форумах можно почитать содержательные посты. Супер!
Покинул форум
Сообщений всего: 8
Дата рег-ции: Июнь 2010
Помог: 0 раз(а)
А если, например, для функции func($a = 8, $b = 4, $c = 9, $d = 5, $e = 6) мне необходимо заменить значение третьего аргумента $c, а остальные оставить по умолчанию. Как будет вызываться функция в этом случае?
RomAndry
Отправлено: 30 Июня, 2010 - 16:11:44
Частый посетитель
Покинул форум
Сообщений всего: 913
Дата рег-ции: Янв. 2008
Покинул форум
Сообщений всего: 9095
Дата рег-ции: Июнь 2007 Откуда: Berlin
Помог: 707 раз(а)
Champion пишет:
Функция называется рекурсивной, если она вызывает саму себя
Это утверждение не совсем правильно.
Если быть точным, то это - явная рекурсия. Существует еще так же и неявная, как например, функция func1, вызывающая func2, func2, вызывающая func3 и func3, вызывыющая func1 (элементов такого "кольца" может быть сколько угодно)
----- Есть в мире две бесконечные вещи - это Вселенная и человеческая глупость. Но насчет первой .. я не уверен.
Druid
Отправлено: 10 Октября, 2010 - 10:20:03
Новичок
Покинул форум
Сообщений всего: 14
Дата рег-ции: Окт. 2010
Помог: 0 раз(а)
Хотелось бы показать, что у меня получилось по заданиям в этом уроке. Есть ли какие замечания?
Все гости форума могут просматривать этот раздел. Только зарегистрированные пользователи могут создавать новые темы в этом разделе. Только зарегистрированные пользователи могут отвечать на сообщения в этом разделе.