diff --git a/modules/10-basics/10-first-program/description.ru.yml b/modules/10-basics/10-first-program/description.ru.yml deleted file mode 100644 index 5fd094d..0000000 --- a/modules/10-basics/10-first-program/description.ru.yml +++ /dev/null @@ -1,42 +0,0 @@ ---- - -name: Первая программа на С++ -theory: | - Изучать язык программирования, по традиции, начинают с программы 'Hello, World!'. - -
- Hello, World! -- - В языке C++ эта программа будет выглядеть так: - - ```cpp - main() { - std::cout << "Hello, World!"; - } - ``` - - Текст `Hello, World!` появится на экране благодаря команде `std::cout <<`. - Такая команда выводит на экран информацию, которая указана после оператора << `'Hello, World!'`. - Оператор `<<` позволяет строить цепочки, например: `std::cout << "Hello, World!" << "\n";` - То есть теперь информация после _Hello, World!_ будет выводиться на экране на следующей строке. - - То, что присутствует на экране помимо этой команды, нужно для работы любой программы на языке C++, мы разберём это позднее. -instructions: | - Наберите в редакторе код из задания символ в символ и нажмите «Проверить». - - ```cpp - #include
- Mother of Dragons - Dracarys! -- - Почему это важно знать? Инструкция — это единица исполнения. Программа которая запускает код на C++, выполняет инструкции строго по очереди. И мы, как разработчики, должны понимать этот порядок и уметь мысленно разделять программу на независимые части, удобные для анализа. - Теоретически инструкции можно написать последовательно друг за другом без переноса на новую строчку: - - ```cpp - std::cout << "Mother of Dragons\n"; std::cout << "Dracarys!\n"; - ``` - - Результат на экране будет таким же, но на практике такой подход считается плохим. - -instructions: | - Выведите на экран друг за другом три имени: *Robert*, *Stannis*, *Renly*. В результате на экране должно отобразиться: - -
- Robert - Stannis - Renly -- - Для каждого имени можете используйте свой собственный вызов `std::cout <<`. - - Вывод в поток `std::cout` не умеет самостоятельно делать перевод строки, по этому это надо указать явно управляющим символом "\n" или воспользоваться функцией `endl`. - - ```cpp - std::cout << "Mother of Dragons\n"; - std::cout << "Mother of Dragons" << "\n"; - std::cout << "Mother of Dragons" << std::endl; - ``` - - Подсказка: - - С помощью оператора вставки `<<` можно выстраивать цепочки: - - ```cpp - std::cout << "Mother of Dragons\n" << "Dracarys!\n"; - ``` - -tips: - - | - [Функция endl](https://learn.microsoft.com/ru-ru/cpp/standard-library/ostream-functions?view=msvc-170#endl) - [Что такое поток](https://learn.microsoft.com/ru-ru/cpp/standard-library/what-a-stream-is?view=msvc-170) - [Использование printf в современном С++](https://learn.microsoft.com/ru-ru/archive/msdn-magazine/2015/march/windows-with-c-using-printf-with-modern-c) diff --git a/modules/10-basics/30-statemets/ru/data.yml b/modules/10-basics/30-statemets/ru/data.yml index 58b2947..1156db0 100644 --- a/modules/10-basics/30-statemets/ru/data.yml +++ b/modules/10-basics/30-statemets/ru/data.yml @@ -1,3 +1,4 @@ +--- name: Инструкции (Statements) tips: - > diff --git a/modules/10-basics/40-program-structure/description.ru.yml b/modules/10-basics/40-program-structure/description.ru.yml deleted file mode 100644 index f60b454..0000000 --- a/modules/10-basics/40-program-structure/description.ru.yml +++ /dev/null @@ -1,79 +0,0 @@ ---- - -name: Структура программы в С++ -theory: | - Давайте посмотрим на листинг программы из прошлого урока. - - ```cpp - #include
- Assertion `result == expected` failed in test.cpp line 26: - Expected: "Hello, World" to be: "Hello, World!" -- - Читать это следует так: в ответе ожидалось строка с "Hello, World!" (первая строчка после двойного '='), но в вместо него пришла строка "Hello, World". Строки отличаются символом под номером 12. В данном примере пропущен восклицательный знак. - - Иногда в процессе решения будет казаться, что вы сделали все правильно, но система "капризничает" и не принимает решение. Подобное поведение практически исключено. Нерабочие тесты просто не могут попасть на сайт, они автоматически запускаются после каждого изменения. В подавляющем большинстве таких случаев, (а все наши проекты в сумме провели миллионы проверок за много лет), ошибка содержится в коде решения. Она может быть очень незаметной, вместо английской буквы случайно ввели русскую, вместо верхнего регистра использовали нижний или забыли вывести запятую. Другие случаи сложнее. Возможно ваше решение работает для одного набора входных данных, но не работает для другого. Поэтому всегда внимательно читайте условие задачи и вывод тестов. Там почти наверняка есть указание на ошибку. - - Однако, если вы уверены в ошибке или нашли какую-то неточность, то вы всегда можете указать на нее. В конце каждой теории есть ссылка на содержимое урока на гитхабе (этот проект полностью открытый!). Перейдя туда, вы можете написать issue, посмотреть содержимое тестов (там видно, как вызывается ваш код) и даже отправить pullrequest. Если для вас это пока темный лес, то подключитесь в наше сообщество [Telegram Hexlet](https://t.me/hexletcommunity/12), там в канале Волонтеры мы всегда поможем. - - Кроме наших тестов, будет крайне полезно экспериментировать с кодом в вашем собственном консольном приложении. [Установите Visual Studio](https://visualstudio.microsoft.com/ru/downloads/) и попробуйте создать [простое консольное приложение](https://learn.microsoft.com/ru-ru/cpp/build/vscpp-step-1-create?view=msvc-170). Оцените мощь IDE, которая может сама автодополнять выражения и подчеркивать места с ошибками. - - Если вы используете Linux, то у вас уже установлен компилятор g++ и вы можете скомпилировать программу с помощью него, а код набирать в любом текстовом редакторе, например [VScode](https://code.visualstudio.com/). - -instructions: | - Просто тренировка. Выведите на экран число 420262531. - -
- 420262531 -- - Поэкспериментируйте с выводом. Передайте туда другое число или строку. Посмотрите на ответ системы, попробуйте его перевести и понять. - -definitions: - - name: Тесты - description: | - специальный код, проверяющий программы на корректность, сверяя правильный результат с реальным. - -tips: - - | - [g++](https://gcc.gnu.org/onlinedocs) - - | - [TDD](https://ru.wikipedia.org/wiki/Разработка_через_тестирование) - - | - [Сообщество Хекслета в Telegram](https://t.me/hexletcommunity/12) diff --git a/modules/10-basics/50-testing/ru/data.yml b/modules/10-basics/50-testing/ru/data.yml index f98ffba..87f69da 100644 --- a/modules/10-basics/50-testing/ru/data.yml +++ b/modules/10-basics/50-testing/ru/data.yml @@ -1,3 +1,4 @@ +--- name: Как мы проверяем ваши решения definitions: - name: Тесты diff --git a/modules/10-basics/60-syntax-error/description.ru.yml b/modules/10-basics/60-syntax-error/description.ru.yml deleted file mode 100644 index eec8ca1..0000000 --- a/modules/10-basics/60-syntax-error/description.ru.yml +++ /dev/null @@ -1,29 +0,0 @@ ---- -name: Ошибки оформления (синтаксиса) -theory: | - В человеческих языках грамматика важна, но текст с ошибками чаще всего можно понять и прочитать. В программировании всё строго. Любое мельчайшее нарушение, и программа не запустится. Примером может быть забытая `;`, неправильно расставленные скобки и другие детали. Подобные ошибки называются синтаксическими, потому что они нарушают правила синтаксиса языка. Если программа на C++ написана синтаксически некорректно, то компилятор выводит на экран соответствующее сообщение, а также указание на файл и строчку в нём, где по его мнению произошла ошибка. Ниже пример кода с синтаксической ошибкой: - - ```cpp - std::cout << "alala - ``` - - Если попробовать запустить код выше, то мы увидим следующее сообщение: - -
- ./main.cpp:5:16: error: expected expression - std::cout << "alala - ^ - 1 error generated. -- - С одной стороны, ошибки синтаксиса — самые простые, потому что они связаны исключительно с грамматическими правилами написания кода, а не с самим смыслом кода. Их легко исправить: нужно лишь найти нарушение в записи. С другой стороны, компилятор не всегда может чётко указать на это нарушение. Поэтому бывает, что забытую скобку нужно поставить не туда, куда указывает сообщение об ошибке. - -instructions: | - Это задание не связано с уроком напрямую. Но будет полезным потренироваться с выводом на экран. Выведите на экран *What Is Dead May Never Die*. - -definitions: - - name: Компилятор - description: Программа выполняющая преобразование исходного кода в низкоуровневый машинный код подходящий для выполнения - - - name: Синтаксическая ошибка - description: Нарушение грамматических правил языка программирования diff --git a/modules/10-basics/60-syntax-error/ru/data.yml b/modules/10-basics/60-syntax-error/ru/data.yml index 1f501cb..2cc434b 100644 --- a/modules/10-basics/60-syntax-error/ru/data.yml +++ b/modules/10-basics/60-syntax-error/ru/data.yml @@ -1,3 +1,4 @@ +--- name: Ошибки оформления (синтаксиса) definitions: - name: Компилятор diff --git a/modules/10-basics/70-compiler/description.ru.yml b/modules/10-basics/70-compiler/description.ru.yml deleted file mode 100644 index 436e0ad..0000000 --- a/modules/10-basics/70-compiler/description.ru.yml +++ /dev/null @@ -1,52 +0,0 @@ ---- - -name: Компиляция и компоновка (линковка) -theory: | - С++ — компилируемый язык. Поэтому прежде чем запускать написанную программу, нам нужно превратить текстовые файлы с исходным кодом в машинный код, который понятен компьютеру. - - В этом уроке мы узнаем, что такое компилятор. Также разберем, как происходит процесс компиляции, который можно разбить на две стадии: **компиляция** и **компоновка**. - - ## Компиляция - - Чтобы скомпилировать программу на C++, мы используем специальную программу. Она последовательно просматривает каждый файл исходного кода (.cpp) и выполняет две важные задачи: - - 1. Проверяет код на соответствие правилам языка C++. В противном случае компилятор выдаст ошибку и номер соответствующей строки, чтобы помочь точно определить, что нужно исправить. Процесс компиляции будет прерван, пока ошибка не будет исправлена - - 2. Переводит исходный код C++ в файл машинного кода, называемый объектным файлом - - Объектные файлы обычно имеют имена `name.o` или `name.obj`, где name совпадает с именем файла `.cpp`, из которого он был создан. - - Если бы в вашей программе было бы три файла `.cpp`, компилятор сгенерировал бы три объектных файла. - - Компиляторы C++ доступны для многих операционных систем. Например, в стандартной поставке многих дистрибутивов Linux есть компилятор **gcc**. В Windows можно пользоваться IDE Visual Studio — в нее уже встроен компилятор и система сборки. - - ## Компоновка - - После того, как компилятор создал один или несколько объектных файлов, включается другая программа — **компоновщик** или **линкер**. Работа компоновщика состоит из трех частей: - - 1. Берет все объектные файлы, сгенерированные компилятором, и объединяет их в единую исполняемую программу - - 2. Помимо возможности связывать объектные файлы компоновщик также может связывать файлы библиотек. Файл библиотеки — это набор предварительно скомпилированного кода, который был упакован для повторного использования в других программах - - 3. Обеспечивает правильное разрешение всех межфайловых зависимостей. Например, если мы определяем что-то в одном файле `.cpp`, а затем используем это в другом файле `.cpp`, компоновщик соединит их вместе. Если компоновщик не может связать ссылку с чем-то с ее определением, мы получим ошибку компоновщика, и процесс линковки прервется - - ## Системы сборки - - Когда проект содержит десятки и даже сотни файлов с исходным кодом, процесс его сборки надо автоматизировать. Здесь на помощь приходят системы сборки, которые автоматически запускают все нужные команды, чтобы скомпилировать и скомпоновать все файлы проекта. В итоге на выходе получается один исполняемый файл. - - Одной из таких систем является утилита `Make` и `Makefile`. В `Makefile` описываются все цели и зависимости проекта, а утилита `Make` смотрит в этот файл и запускает компилятор с соответствующими командами. - - Еще одна популярная система сборки проектов — утилита `CMake`, которая работает поверх `Make`. Она отличается своей кроссплатформенностью и позволяет делать сборки под различные операционные системы. - - Компиляции и сборка программы не менее важный процесс, чем написание самой программы. - -instructions: | - Выведите на экран число 9780262531962. - -tips: - - | - "Если в редакторе есть запись `// BEGIN` и `// END`, то код нужно писать между этими строчками." - - | - [Установка gcc в Windows](https://www.digitalocean.com/community/tutorials/c-compiler-windows-gcc) - - | - [gcc для Linux](https://gcc.gnu.org/onlinedocs) diff --git a/modules/10-basics/70-compiler/ru/data.yml b/modules/10-basics/70-compiler/ru/data.yml index eabbe8a..9a88a2a 100644 --- a/modules/10-basics/70-compiler/ru/data.yml +++ b/modules/10-basics/70-compiler/ru/data.yml @@ -1,3 +1,4 @@ +--- name: Компиляция и компоновка (линковка) tips: - > diff --git a/modules/10-basics/description.ru.yml b/modules/10-basics/description.ru.yml index d00f141..723f60f 100644 --- a/modules/10-basics/description.ru.yml +++ b/modules/10-basics/description.ru.yml @@ -2,4 +2,3 @@ name: Основы description: | C++ – язык программирования созданный програмистом Bell Labs Бьёрн Страуструп в начале 80-х годов. Он стабильно входит в Top 10 самых популярных языков. На нем можно разрабатывать realtime или высокопроизводительные приложения, игры, мультимедиа, системное и индустриальное ПО. Изучать C++ мы будем с нуля, с самых азов. Первый модуль – плацдарм для написания осмысленных программ. В нем мы разберем, как написать свой первый код на C++. Расскажем, что такое комментарии и зачем они нужны. На примере проверки ваших решений рассмотрим, что такое тестирование и как читать вывод тестов. - diff --git a/modules/20-arithmetics/10-basics/description.ru.yml b/modules/20-arithmetics/10-basics/description.ru.yml deleted file mode 100644 index 36e86b0..0000000 --- a/modules/20-arithmetics/10-basics/description.ru.yml +++ /dev/null @@ -1,57 +0,0 @@ ---- -name: Арифметические операции -theory: | - На базовом уровне компьютеры оперируют только числами. Даже в прикладных программах на высокоуровневых языках внутри много чисел и операций над ними. К счастью, для старта достаточно знать обычную арифметику — с нее и начнем. - - Для сложения двух чисел в математике мы пишем, например, *3 + 4*. В программировании — то же самое. Вот программа, складывающая два числа: - - ```cpp - // Не забываем точку с запятой в конце, - // так как каждая строчка в коде - инструкция - int main() { - 3 + 4; - return 0; - } - ``` - - Если запустить эту программу на выполнение, то она тихо отработает и завершиться. На экран ничего не будет выведено. Операция сложения, как и остальные операции, сама по себе ничего не делает кроме сложения. Чтобы воспользоваться результатом сложения, его нужно, например, вывести на экран. - - ```cpp - int main() { - std::cout << 3 + 4; - return 0; - } - ``` - - После запуска на экране появится результат: - -
7- - Кроме сложения доступны следующие операции: - * `*` — умножение - * `/` — деление - * `-` — вычитание - * `%` — [остаток от деления](https://ru.wikipedia.org/wiki/Деление_с_остатком) - - Теперь давайте выведем на экран результат деления, а потом результат возведения в степень: - - ```cpp - int main() { - std::cout << 8 / 2; - std::cout << 3 * 3 * 3; - } - ``` - -
- 4 - 27 -- -instructions: | - Выведите на экран результат деления числа *81* на *9*. Не забудьте сделать перевод строки. - -tips: - - | - Всегда отбивайте арифметические операторы пробелами от самих чисел (операндов) – это хороший стиль программирования. Поэтому в наших примерах `std::cout << 3 + 4`, а не `std::cout << 3+4`. - - | - Остаток от деления отбрасывается с округлением вниз. Таким образом результат деления – всегда целое число. Как работать с вещественными числами будет показано в следующих уроках. diff --git a/modules/20-arithmetics/10-basics/ru/data.yml b/modules/20-arithmetics/10-basics/ru/data.yml index 929aa9f..bbd9fb5 100644 --- a/modules/20-arithmetics/10-basics/ru/data.yml +++ b/modules/20-arithmetics/10-basics/ru/data.yml @@ -1,3 +1,4 @@ +--- name: Арифметические операции tips: - > diff --git a/modules/20-arithmetics/20-operators/description.ru.yml b/modules/20-arithmetics/20-operators/description.ru.yml deleted file mode 100644 index c3f8423..0000000 --- a/modules/20-arithmetics/20-operators/description.ru.yml +++ /dev/null @@ -1,37 +0,0 @@ ---- -name: Операторы -theory: | - Перед тем, как двигаться дальше, разберем базовую терминологию. Знак операции, такой как `+`, называют **оператором**. Оператор — просто символ, который выполняет операцию, например сложение. - - ```cpp - std::cout << (8 + 2); - ``` - - В этом примере `+` это оператор, а числа *8* и *2* — это **операнды**. Скобки конечно можно и опустить, но так выразительнее. - - В случае сложения у нас есть два операнда: один слева, другой справа от знака *+*. Операции, которые требуют наличия двух операндов, называются **бинарными**. Если пропустить хотя бы один операнд, например, так `3 + ;`, то программа завершится с синтаксической ошибкой. - - Операции (не операторы) бывают не только бинарными, но и унарными (с одним операндом), и даже тернарными (с тремя операндами)! Причем операторы могут выглядеть одинаково, но обозначать разные операции. - - ```cpp - std::cout << (-3); // => -3 - ``` - - Выше пример применения унарной операции к числу *3*. Оператор минус перед тройкой говорит компилятору взять число *3* и найти противоположное, то есть *-3*. - - Это немного может сбить с толку, потому что *-3* — это одновременно и число само по себе, и оператор с операндом, но у языков программирования такая структура. - -instructions: | - Напишите программу, которая посчитает разность между числами `6` и `-81` и выведет ответ на экран. Не забудьте сделать перевод строки. - -definitions: - - name: "Арифметическая операция" - description: "сложение, вычитание, умножение и деление." - - name: "Оператор" - description: "специальный символ, создающий операцию. Например, `+` создает операцию сложения." - - name: "Операнд" - description: "объект, который участвует в операции. `3 * 6`: здесь 3 и 6 — операнды." - - name: "Унарная операция" - description: "операция с одним операндом. Например, `-3` — унарная операция для получения числа, противоположного числу три." - - name: "Бинарная операция" - description: "операция с двумя операндами. Например, `3 + 9`." diff --git a/modules/20-arithmetics/20-operators/ru/data.yml b/modules/20-arithmetics/20-operators/ru/data.yml index e0e8a5c..6521cbd 100644 --- a/modules/20-arithmetics/20-operators/ru/data.yml +++ b/modules/20-arithmetics/20-operators/ru/data.yml @@ -1,3 +1,4 @@ +--- name: Операторы definitions: - name: Арифметическая операция diff --git a/modules/20-arithmetics/30-commutativity/description.ru.yml b/modules/20-arithmetics/30-commutativity/description.ru.yml deleted file mode 100644 index 9a84bd0..0000000 --- a/modules/20-arithmetics/30-commutativity/description.ru.yml +++ /dev/null @@ -1,17 +0,0 @@ ---- -name: Коммутативная операция -theory: | - Мы все помним со школы: «от перемены мест слагаемых сумма не меняется». Это один из базовых и интуитивно понятных законов арифметики, он называется **коммутативным законом**. - - Бинарная операция считается коммутативной, если, поменяв местами операнды, вы получаете тот же самый результат. Очевидно, что сложение — коммутативная операция: *3 + 2 = 2 + 3*. А вот является ли коммутативной операция вычитания? Нет: *2 - 3 ≠ 3 - 2*. В программировании этот закон работает точно так же, как в арифметике. Более того, большинство операций, с которыми мы будем сталкиваться в реальной жизни, не являются коммутативными. Отсюда вывод: всегда обращайте внимание на порядок того, с чем работаете. - -instructions: | - Это задание напрямую не связано с темой урока. Но будет полезным попрактиковаться с арифметическими операциями и выводом на экран. - - Напишите программу, которая считает и последовательно выводит на экран значения следующих математических выражений: «3 умножить на 5» и «-8 разделить на -4». - - Не забывайте что `std::cout` не умеет сам делать перевод строки. - -definitions: - - name: "Коммутативность" - description: "свойство операции, когда изменения порядка операндов не влияет на результат. Например, сложение — коммутативная операция: от перемены мест слагаемых сумма не меняется." diff --git a/modules/20-arithmetics/30-commutativity/ru/data.yml b/modules/20-arithmetics/30-commutativity/ru/data.yml index 4a856c1..669d02f 100644 --- a/modules/20-arithmetics/30-commutativity/ru/data.yml +++ b/modules/20-arithmetics/30-commutativity/ru/data.yml @@ -1,3 +1,4 @@ +--- name: Коммутативная операция definitions: - name: Коммутативность diff --git a/modules/20-arithmetics/40-composition/description.ru.yml b/modules/20-arithmetics/40-composition/description.ru.yml deleted file mode 100644 index 325ee6e..0000000 --- a/modules/20-arithmetics/40-composition/description.ru.yml +++ /dev/null @@ -1,32 +0,0 @@ ---- -name: Композиция операций -theory: | - А что, если понадобится вычислить такое выражение: *3 + 5 - 2*? Именно так мы и запишем: - - ```cpp - std::cout << (3 + 5 - 2); // 3 + 5 - 2 => 8 - 2 => 6 - ``` - - Обратите внимание, что компьютер производит арифметические вычисления в правильном порядке: сначала деление и умножение, потом сложение и вычитание. Иногда этот порядок нужно изменить — об этом следующий урок. - - Или другой пример: - - ```cpp - std::cout << (2 * 4 * 5 * 10); - // 2 * 4 * 5 * 10 => 8 * 5 * 10 => 40 * 10 => 400 - ``` - - Как видно, операции можно соединять друг с другом, получая возможность вычислять все более сложные составные выражения. Чтобы представить себе то, как программа производит вычисления, давайте разберем пример: `2 * 4 * 5 * 10`. - - 1. Сначала вычисляется *2 * 4* и получается выражение *8 * 5 * 10*. - 2. Затем *8 * 5*. В итоге имеем *40 * 10*. - 3. В конце концов происходит последнее умножение, и получается результат *400*. - - Операции можно соединять друг с другом, получая возможность вычислять все более сложные составные выражения. - -instructions: | - Реализуйте программу, которая вычисляет значение выражения `8 / 2 + 5 - -3 / 2`. Не вычисляйте ничего самостоятельно, ваша программа должна производить все вычисления сама. - - Обратите внимание, что программа производит арифметические вычисления в правильном порядке: сначала деление и умножение, потом сложение и вычитание. Иногда этот порядок нужно изменить — об этом следующий урок. - - Также обратите внимание на то, что в C++ по умолчанию используется целочисленное деление, `3 / 2` будет `1`. В выводе не забудьте сделать перевод строки. diff --git a/modules/20-arithmetics/40-composition/ru/data.yml b/modules/20-arithmetics/40-composition/ru/data.yml index ea206ad..f721c13 100644 --- a/modules/20-arithmetics/40-composition/ru/data.yml +++ b/modules/20-arithmetics/40-composition/ru/data.yml @@ -1 +1,2 @@ +--- name: Композиция операций diff --git a/modules/20-arithmetics/50-priority/description.ru.yml b/modules/20-arithmetics/50-priority/description.ru.yml deleted file mode 100644 index 3b069c1..0000000 --- a/modules/20-arithmetics/50-priority/description.ru.yml +++ /dev/null @@ -1,38 +0,0 @@ ---- -name: Приоритет операций -theory: | - Посмотрите внимательно на выражение *2 + 2 * 2* и посчитайте в уме ответ. - - Правильный ответ: *6*. - - Если у вас получилось *8*, то этот урок для вас. В школьной математике мы изучали понятие «приоритет операции». Приоритет определяет то, в какой последовательности должны выполняться операции. Например, умножение и деление имеют больший приоритет, чем сложение и вычитание: *2 + 3 * 2* вычислится в *8*. - - Но нередко вычисления должны происходить в порядке, отличном от стандартного приоритета. В сложных ситуациях приоритет можно (и нужно) задавать круглыми скобками, точно так же, как в школе, например: `(2 + 2) * 2`. - - Скобки можно ставить вокруг любой операции. Они могут вкладываться друг в друга сколько угодно раз. Вот пара примеров: - - ```cpp - std::cout << (3 * (4 - 2)); // => 6 - std::cout << (7 * 3 + (4 / 2) - (8 + (2 - 1))); // => 14 - ``` - - Иногда выражение сложно воспринимать визуально. Тогда можно расставить скобки, не повлияв на приоритет. Например, задание из прошлого урока можно сделать немного понятнее, если расставить скобки. - - Было: - - ```cpp - std::cout << (8 / 2 + 5 - -3 / 2); // => 10 - ``` - - Стало: - - ```cpp - std::cout << (((8 / 2) + 5) - (-3 / 2)); // => 10 - ``` - - Запомните: код пишется для людей, потому что код будут читать люди, а машины будут только исполнять его. Для машин нет «более» понятного или «менее» понятного кода, независимо от того, является ли код корректным или нет. - -instructions: | - Дано выражение `70 * 3 + 4 / 8 + 2`. - - Расставьте скобки так, чтобы оба сложения (`3 + 4` и `8 + 2`) высчитывались в первую очередь. Выведите результат на экран. Не забудьте сделать перевод строки. diff --git a/modules/20-arithmetics/50-priority/ru/data.yml b/modules/20-arithmetics/50-priority/ru/data.yml index 8b35617..da15a1d 100644 --- a/modules/20-arithmetics/50-priority/ru/data.yml +++ b/modules/20-arithmetics/50-priority/ru/data.yml @@ -1 +1,2 @@ +--- name: Приоритет операций diff --git a/modules/20-arithmetics/60-float/description.ru.yml b/modules/20-arithmetics/60-float/description.ru.yml deleted file mode 100644 index d0325a1..0000000 --- a/modules/20-arithmetics/60-float/description.ru.yml +++ /dev/null @@ -1,35 +0,0 @@ ---- -name: Числа с плавающей точкой - -theory: | - В математике существуют разные виды чисел, например, натуральные – это целые числа от одного и больше, или рациональные – это числа с точкой, например 0.5. С точки зрения устройства компьютеров, между этими видами чисел пропасть. Попробуйте ответить на простой вопрос, сколько будет *0.2 + 0.1*? А теперь посмотрим, что на это скажет C++: - - ```cpp - std::cout << (0.2 + 0.1); - // => 0.30000000000000004 - ``` - - Операция сложения двух рациональных чисел внезапно привела к неточному вычислению результата. Тот же самый результат выдадут и другие языки программирования. Такое поведение обуславливается ограничениями вычислительных мощностей. Объем памяти, в отличие от чисел, конечен (бесконечное количество чисел требует бесконечного количества памяти для своего хранения). И если с натуральными числами эта проблема решается простым ограничением по верхней границе (есть некоторое максимальное число, которое можно ввести), то с рациональными такой финт не пройдет. - - ```cpp - #include
- The num_out = 0 - The num_in = 0 -- - Хорошей практикой считается определять переменную ближе к тому месту в коде где она будет использоваться и инициализировать сразу нейтральным значением. - - Например, если это счетчик цикла, то 0, если в переменную будет собираться строка, то пустой строкой. - - ```cpp - int main() { - string acc = ""; - acc += "Hello, "; - acc += "World"; - std::cout << acc; - } - ``` - -
- Hello, World! -- - C++ поддерживает три основных способа инициализации переменных. Во-первых, мы можем выполнить копирующую инициализацию, используя знак равенства: - - ```cpp - int age = 18; // копирующая инициализация значения 18 в переменную age - ``` - - Подобно копирующему присваиванию, этот код копирует значение с правой стороны знака равно в переменную, создаваемую с левой стороны. Во-вторых, мы можем выполнить прямую инициализацию с помощью скобок: - - ```cpp - int age(18); // прямая инициализация значения 18 в переменную age - ``` - - Для простых типов данных копирующая и прямая инициализации, по сути, одинаковы. Различия между копирующей инициализацией и прямой инициализацией мы разберем далее в курсе. - - И третий тип инициализации - списковая. Списковая инициализация - это более унифицированный механизм инициализации, подходит как для простых типов, так и для составных, которые мы будем рассматривать дальше по курсу. - - Инициализация списком бывает двух форм: - - ```cpp - int width { 5 }; // прямая инициализация списком значения 5 в переменную width (предпочтительно) - int height = { 6 }; - ``` - - Эти две формы функционируют почти одинаково, но обычно предпочтительнее прямая форма. - -instructions: | - Внутри функции `main` определите переменную типа `int`, присвойте ей значение 42 и выведите на экран. Не забудьте сделать перевод строки. - -tips: - - | - [Именование в программировании](https://ru.hexlet.io/blog/posts/naming-in-programming) - [Ключевые слова в С++](https://learn.microsoft.com/ru-ru/cpp/cpp/keywords-cpp?view=msvc-170) - -definitions: - - name: Переменная - description: способ сохранить информацию и дать ей имя для последующего использования в коде. - - name: Ключевые слова - description: зарезервированные слова, которые имеют особое значение для компилятора. Например, `int` или `float` — ключевое слово для объявления переменных. diff --git a/modules/30-variables/10-definition/ru/data.yml b/modules/30-variables/10-definition/ru/data.yml index 3ed3e5e..42c5ac7 100644 --- a/modules/30-variables/10-definition/ru/data.yml +++ b/modules/30-variables/10-definition/ru/data.yml @@ -1,3 +1,4 @@ +--- name: Что такое переменная? tips: - > diff --git a/modules/30-variables/20-change/description.ru.yml b/modules/30-variables/20-change/description.ru.yml deleted file mode 100644 index bdceabc..0000000 --- a/modules/30-variables/20-change/description.ru.yml +++ /dev/null @@ -1,43 +0,0 @@ ---- -name: Изменение переменной - -theory: | - Само слово «переменная», говорит о том, что ее можно менять. И действительно, с течением времени внутри программы, значения переменных могут изменяться. - - ```cpp - int num { 1 }; - std::cout << num << std::endl; - // тип данных указывать не надо, так как переменная была определена выше - num = 2; - std::cout << num << std::endl; - ``` - -
- 1 - 2 -- - C++ статически типизированный язык, это значит что тип переменной задается при определении и больше не меняется. - - Прежде всего это связанно с тем, что данные разных типов, по-разному хранятся в памяти компьютера. - - В примере выше, при создании переменной, мы присвоили ей число. Компилятор запоминает тип и проверяет все последующие изменения переменной. Если попробовать этой же переменной присвоить строку, то мы получим следующую ошибку: - - ```cpp - num { "Hello" }; - // error: assigning to 'int' from incompatible type - ``` - - Интересно то, что компилятор делает такую проверку без запуска кода на выполнение, именно поэтому такой вид типизации называют статическим (статика – без запуска). В динамических языках, таких как Javascript, Ruby, PHP или Python, подобное поведение не является ошибкой, переменная может легко изменить свой тип в процессе работы. - -instructions: | - - В коде определена переменная со значением `10`. Переопределите значение этой переменной и присвойте ей значение на единицу больше. Не забудьте сделать перевод строки. - -tips: - - | - Если в редакторе есть запись `// BEGIN` и `// END`, то код нужно писать между этими строчками. - -definitions: - - name: Переменная - description: Способ сохранить информацию и дать ей имя для последующего использования в коде. diff --git a/modules/30-variables/20-change/ru/data.yml b/modules/30-variables/20-change/ru/data.yml index cd21591..28ea8c9 100644 --- a/modules/30-variables/20-change/ru/data.yml +++ b/modules/30-variables/20-change/ru/data.yml @@ -1,3 +1,4 @@ +--- name: Изменение переменной tips: - > diff --git a/modules/30-variables/30-variables-naming/description.ru.yml b/modules/30-variables/30-variables-naming/description.ru.yml deleted file mode 100644 index 2b68505..0000000 --- a/modules/30-variables/30-variables-naming/description.ru.yml +++ /dev/null @@ -1,45 +0,0 @@ ---- -name: Выбор имени переменной -theory: | - Представим себе, что мы пишем программу которая принимает из консоли возраст пользователя и выводит его на экран: - - ```cpp - #include
- 18 -- - Область видимости глобальной переменной весь файл, но мы не можем обратиться к ней внутри функции, так как ее перекрывает локальная переменная. Такие ошибки трудно уловимы и конечно переменным лучше придумать разные имена. - - На самом деле мы можем обратиться к глобальной переменной используя оператор `::`. Мы уже работали с ним, обращаясь к пространству имён `std::cout`. - - ```cpp - int user_age { 20 }; - - int main() { - int user_age { 18 }; - std::cout << ::user_age; - } - ``` - -
- 20 -- - Если перед оператором `::`, это значит мы обращаемся к глобальному пространству имен. С пространством имен мы познакомимся дальше в курсе. - -instructions: | - Найдите в программе необъявленную переменную и объявите ее, присвоив ей значение `27`; diff --git a/modules/30-variables/40-errors/ru/data.yml b/modules/30-variables/40-errors/ru/data.yml index 52155dd..5ef08c7 100644 --- a/modules/30-variables/40-errors/ru/data.yml +++ b/modules/30-variables/40-errors/ru/data.yml @@ -1 +1,2 @@ +--- name: Ошибки при работе с переменными diff --git a/modules/30-variables/50-variables-expressions/description.ru.yml b/modules/30-variables/50-variables-expressions/description.ru.yml deleted file mode 100644 index 0d28c0c..0000000 --- a/modules/30-variables/50-variables-expressions/description.ru.yml +++ /dev/null @@ -1,75 +0,0 @@ ---- -name: Выражения в определениях -theory: | - Переменные полезны не только для хранения и переиспользования информации, но и для упрощения сложных вычислений. Давайте рассмотрим пример: нужно перевести евро в рубли через доллары. Подобные конвертации через промежуточную валюту часто делают банки при покупках за рубежом. - - Для начала переведем 50 евро в доллары. Допустим, что один евро — 1.25 доллара: - - ```cpp - auto dollars_count { 50 * 1.25 }; - std::cout << dollars_count; - ``` - - В предыдущем уроке мы записывали в переменную конкретное значение. А здесь `auto dollars_count { 50 * 1.25 };` справа от знака равно находится **выражение**. Программа вычислит результат — *62.5* — и запишет его в переменную. С точки зрения программы не важно, что написано: *62.5* или *50 * 1.25*, эти оба варианта — выражения, которые надо вычислить. И они вычисляются в одно и тоже значение — *62.5*. - - Вы могли заметить, что появилось новое ключевое слово `auto`. Поскольку мы еще не изучали типы данных в С++, мы указали компилятору, что бы он сам определил тип данных у переменной `dollars_count`. - - Любая строка — выражение. Когда программа видит выражение, она вычисляет его и **возвращает** результат. Вот несколько примеров выражений, а в комментариях справа от каждого выражения — итоговое значение: - - ```cpp - 62.5 // 62.5 - 50 * 1.25 // 62.5 - 120 / 10 * 2 // 24 - - "Hexlet" // "Hexlet" - ``` - - Правила построения кода таковы, что в тех местах, где ожидается выражение, можно поставить любое вычисление (не только математическое, но и, например, строковое — как конкатенация), и программа останется работоспособной. По этой причине невозможно описать и показать все случаи использования всех операций. Программы состоят из множества комбинаций выражений, и понимание этой концепции — один из ключевых шагов на вашем пути. - - Вернемся к нашей валютной программе. Запишем стоимость доллара в рублях, как отдельную переменную. Вычислим цену 50 евро в долларах, умножив их на 1.25. Допустим, что 1 доллар — 60 рублей: - - ```cpp - int main() { - auto rubles_per_dollar { 60 }; - auto dollars_count { 50 * 1.25 }; // 62.5 - auto rubles_count { dollars_count * rubles_per_dollar }; // 3750 - - std::cout << rubles_count; // => 3750 - return 0; - } - ``` - - А теперь давайте добавим к выводу текст: - - ```cpp - int main() { - auto rubles_per_dollar { 60 }; - auto dollars_count { 50 * 1.25 }; // 62.5 - auto rubles_count { dollars_count * rubles_per_dollar }; // 3750 - - std::cout << "The price is " << rubles_count << " rubles" << std::endl; // => The price is 3750 rubles - } - ``` - - Любая переменная может быть частью любого выражения. В момент вычисления вместо имени переменной подставляется её значение. - - Значение `dollars_count` вычисляется до того, как она начнет использоваться в других выражениях. Когда подходит момент использования переменной, C++ «знает» значение, потому что уже вычислил его. - -instructions: | - Напишите программу, которая берет исходное количество евро, записанное в переменную `euros_count`, переводит евро в доллары и выводит на экран. Затем полученное значение переводит в рубли и выводит на новой строчке. Не забудьте в конце вывода добавить перевод строки. - - Пример вывода для 100 евро: - -
- 125.0 - 7500.0 -- - Считаем, что: - - - 1 евро = 1.25 долларов - - 1 доллар = 60 рублей - -tips: - - | - Для перевода строчки можно использовать `\n` или `std::endl` между выводом долларов и рублей. diff --git a/modules/30-variables/50-variables-expressions/ru/data.yml b/modules/30-variables/50-variables-expressions/ru/data.yml index 1206dbd..9aad4f5 100644 --- a/modules/30-variables/50-variables-expressions/ru/data.yml +++ b/modules/30-variables/50-variables-expressions/ru/data.yml @@ -1,3 +1,4 @@ +--- name: Выражения в определениях tips: - > diff --git a/modules/30-variables/60-naming-style/description.ru.yml b/modules/30-variables/60-naming-style/description.ru.yml deleted file mode 100644 index 1a1d500..0000000 --- a/modules/30-variables/60-naming-style/description.ru.yml +++ /dev/null @@ -1,36 +0,0 @@ ---- -name: Именование переменных -theory: | - `age` или `number` — пример простого имени, но не все имена так просты. Довольно часто они составные, то есть включают в себя несколько слов. Например, «имя пользователя». В разных языках применяются разные стили кодирования, и имя переменной будет отличаться. - - В именовании переменных можно выделить три основных подхода, которые иногда комбинируют друг с другом. Все эти подходы проявляют себя, когда имя переменной состоит из нескольких слов: - - * kebab-case — составные части переменной разделяются дефисом. Например: `my-super-var`. - * snake_case — для разделения используется подчеркивание. Например: `my_super_var`. - * CamelCase — каждое слово в переменной пишется с заглавной буквы. Например: `MySuperVar`. - * lowerCamelCase — каждое слово в переменной пишется с заглавной буквы, кроме первого. Например: `mySuperVar`. - - В C++ используется смешанный стиль именования: - - * переменные — пишем в стиле snake_case. Например: `current_user`. - * константы — пишем в стиле CamelCase добавляя префикс `k`. Например: `const int kDaysInAWeek = 2`. - * классы — пишем в стиле CamelCase. Например: `MySuperClass`. - * функции — пишем также как и классы в стиле CamelCase. Например: `OpenFile()`. - * файлы - именуются строчными буквами, для разделения можно использовать подчеркивание или дефис. Основные файлы должны иметь расширение .сс, заголовочные .h - - -instructions: | - - Создайте две переменные с именами «первое число» и «второе число» на английском языке, используя snake_case. Запишите в первую переменную число `11`, во вторую — `-100`. Выведите на экран произведение чисел, записанных в получившихся переменных. - - Код будет работать с любыми названиями, а наша система всегда проверяет только результат на экране, поэтому выполнение этого задания — под вашу ответственность. - -tips: - - | - [Google C++ style guide](https://google.github.io/styleguide/cppguide.html) - - | - [Заголовочные файлы](https://learn.microsoft.com/ru-ru/cpp/cpp/header-files-cpp?view=msvc-170) - -definitions: - - name: Стандарт кодирования - description: Набор синтаксических и стилистических правил написания кода. diff --git a/modules/30-variables/60-naming-style/ru/data.yml b/modules/30-variables/60-naming-style/ru/data.yml index eed7998..872dc08 100644 --- a/modules/30-variables/60-naming-style/ru/data.yml +++ b/modules/30-variables/60-naming-style/ru/data.yml @@ -1,3 +1,4 @@ +--- name: Именование переменных tips: - | diff --git a/modules/30-variables/70-magic-numbers/description.ru.yml b/modules/30-variables/70-magic-numbers/description.ru.yml deleted file mode 100644 index f6fb524..0000000 --- a/modules/30-variables/70-magic-numbers/description.ru.yml +++ /dev/null @@ -1,66 +0,0 @@ ---- -name: Магические числа -theory: | - Вспомним один из прошлых уроков: - - ```cpp - // Перевод евро в рубли через доллары - int main() { - int euros { 1000 }; - auto dollars { euros * 1.25 }; // 1250 - auto rubles { dollars * 60 }; // 75000 - - std::cout << rubles << std::endl; // => 75000 - } - ``` - - С точки зрения профессиональной разработки, такой код «пахнет». Так описывают код, который не соответствует так называемым лучшим практикам (best practices). И причина здесь вот в чем: уже сейчас, глядя на числа *60* и *1.25*, вы скорее всего задаетесь вопросом: «что это за числа?». А представьте, что будет через месяц! А как его поймет новый программист, не видевший код ранее? В нашем примере контекст восстанавливается благодаря грамотному именованию, но в реальной жизни код значительно сложнее, и поэтому догадаться до смысла чисел зачастую невозможно. - - Этот «запах» называют магические числа (magic numbers). Числа, происхождение которых невозможно понять без глубокого знания происходящего внутри данного участка кода. - - Выход из ситуации прост: достаточно создать переменные с правильными именами, как все встанет на свои места. - - ```cpp - int main() { - auto dollars_in_euro { 1.25 }; - int rubles_in_dollar { 60 }; - - int euros { 1000 }; - auto dollars { euros * dollars_in_euro }; // 1250 - auto rubles { dollars * rubles_in_dollar }; // 75000 - - std::cout << rubles << std::endl; // => 75000 - } - ``` - - Обратите внимание на следующие детали: - - * Именование *snake_case* - * Две новые переменные отделены от последующих вычислений пустой строчкой. Эти переменные имеют смысл и без вычислений, поэтому такое отделение уместно, оно повышает читаемость. - * Получился хорошо именованный и структурированный код, но он длиннее прошлой версии. Так часто бывает, и это нормально, потому что код должен быть читабельным. - -instructions: | - Вы столкнулись с таким кодом, который выводит на экран среднесуточную температуру в Фаренгейтах: - - ```cpp - std::cout << "Average daily temperature: " << 588 / 24; - ``` - - Как видите, это магические числа: непонятно, что такое _588_ и что такое _24_. - - Избавьтесь от магических чисел, создав новые переменные, а затем выведите текст на экран. - - Получится так: -
- Average daily temperature: 24 -- - Названия переменных должны передавать смысл чисел, но должны при этом оставаться достаточно короткими и ёмкими для комфортного чтения. - - Помните: код будет работать с любыми названиями, а наша система всегда проверяет только результат на экране, поэтому выполнение этого задания — под вашу ответственность. - - В С++ при делении если оба оператора являются целыми числами, то результат будет равен целой доли частного. Незабудьте в конце вывода добавить перевод строки. - -tips: - - | - [Магические числа](https://ru.hexlet.io/blog/posts/magic-numbers) diff --git a/modules/30-variables/70-magic-numbers/ru/data.yml b/modules/30-variables/70-magic-numbers/ru/data.yml index 1d2ce32..f04b759 100644 --- a/modules/30-variables/70-magic-numbers/ru/data.yml +++ b/modules/30-variables/70-magic-numbers/ru/data.yml @@ -1,3 +1,4 @@ +--- name: Магические числа tips: - | diff --git a/modules/40-data-types/10-integer-types/description.ru.yml b/modules/40-data-types/10-integer-types/description.ru.yml deleted file mode 100644 index 22dc210..0000000 --- a/modules/40-data-types/10-integer-types/description.ru.yml +++ /dev/null @@ -1,99 +0,0 @@ ---- -name: Целочисленные типы -theory: | - Целые числа представляют собой бесконечное множество, и в конечном объеме памяти компьютера нельзя представить их все. Таким образом язык может представить только подмножество целых чисел. Отсюда следует определение типа, тип данных - это возможное множество значений. - - Разнообразные целочисленные типы в С++ отличаются друг от друга объемом памяти, выделяемым для хранения целого значения. Чем больше объем памяти, тем шире диапазон предоставляемых целочисленных значений. - - В С++ базовыми целочисленными типами, в порядке увелечения ширины, являются: `char`, `short`, `int`, `long` и `long long`. Каждый из этих типов имеет версии со знаком и без знака. Таким образом, на выбор предоставляется десять целочисленных типов. Давайте познакомимся с ними поближе: - - * `short` - имеет ширину не менее 16 бит и хранит диапазон чисел от - 32768 до 32767 - * `int` - имеет гарантированную ширину, как минимум такую же как `short` и диапазон чисел от -2 147 483 648 до 2 147 483 647 - * `long` - имеет ширину не менее 32 бит и хранит диапазон чисел от -2 147 483 648 до 2 147 483 647 - * `long long` - имеет ширину не менее 64 бит и хранит диапазон чисел от -9 223 372 036 854 775 808 до 9 223 372 036 854 775 807 - - Посмотреть ширину типа в вашей системе можно с помощью оператора `sizeof()`, а диапазоны с помощью символических констант, подключив заголовочный файл стандартной библиотеки `climits`. - - ```cpp - #include
- Integer range: from-2147483648 to 2147483647 - Int type: 4 byte -- - - Стоит сказать, что минимальную ширину типа гарантирует язык, а максимальная зависит от архитектуры процессора. - - Каждый из вышеприведённых типов может быть беззнаковым, то есть хранить только неотрицательные числа. За счет этого можно увеличить значение, которое может хранить переменная. Например, если `short` представляет диапазон от -32 768 до 32 767, то `unsigned short` от 0 до 65 532: - - ```cpp - #include
- Int max: 4294967295 - Int type: 4 byte -- - Видно, что верхняя граница увеличилась в два раза, нижняя будет смещена к нулю, но на количестве выделенной памяти это не сказалось. Такое поведение связано с тем, что для хранения отрицательных чисел используется дополнительный бит. - - Что будет если в беззнаковую переменную сохранить отрицательное число или выйти за пределы диапазона знаковой переменной: - - ```cpp - #include
- new value int_number: -2147483648 - new value u_int_number: 4294967295 -- - Произошла потеря значимости, то есть в обоих случаях мы получили числа другой границы диапазона и компилятор на это не указал. За этим надо следить особенно если вы работаете с большими числами. - - Из всего вышесказанного возникает вопрос: Зачем такой богатый набор типов данных? Допустим, что вы пишете программу для контроллера микроволновой печи и вам известно, что данные которыми будет оперировать программа - это положительные числа, значение которых не превышает 32 000. Естественно ваш выбор падет на `unsigned short` и если у вас большой массив данных, можно существенно сэкономить память. - - В остальных случаях, тип `int` имеет наиболее естественный размер целого числа для целевого компьютера. Под естественным размером подразумевается целочисленная форма, которую компьютер может обработать наиболее эффективным образом. - -instructions: | - Допишите консольную утилиту, которая принимает в качестве аргумента количество секунд и преобразует их в количество дней. Выведите результат на экран. Этап получения аргумента и преобразование его в число уже написан, вам надо дописать логику преобразования секунд в дни. - - Пример вывода: - -
- 86400 seconds = 1 days -- -tips: - - | - [Дополнительный код](https://ru.wikipedia.org/wiki/%D0%94%D0%BE%D0%BF%D0%BE%D0%BB%D0%BD%D0%B8%D1%82%D0%B5%D0%BB%D1%8C%D0%BD%D1%8B%D0%B9_%D0%BA%D0%BE%D0%B4) - - | - [
- 3.33333 - 3.33333 -- - Обратите внимание, что каждое из напечатанных значений имеет только 6 значащих цифр. Это стандартное поведение объекта `cout`, но мы можем переопределить это поведение и повысить точность: - - ```cpp - #include
- 3.333333253860474 - 3.333333333333333 -- - Видно, что число `float` определено не точно и содержит ошибки, поскольку тип `float` гарантирует нам только шесть значащих цифр. - - Существует две особые категории чисел с плавающей точкой. Первая – `Inf`, которая представляет бесконечность. `Inf` может быть положительной или отрицательной. Вторая – `NaN`, что означает Not a Number - не число. Существует несколько различных типов `NaN`. `NaN` и `Inf` доступны только в том случае, если компилятор для чисел с плавающей точкой использует определенный формат. Если используется другой формат, следующий код приводит к неопределенному поведению. - - ```cpp - #include
- inf - -inf - nan -- -instructions: | - Напишите программу, которая рассчитывает сколько километров проедет автомобиль на полном баке. Пусть объем бака будет 43 литра, а расход 8.8 л на 10 км. Результат выведите на экран. - -tips: - - | - [NaN](https://learn.microsoft.com/ru-ru/cpp/c-runtime-library/reference/nan-nanf-nanl?view=msvc-170) diff --git a/modules/40-data-types/20-floating-type/ru/data.yml b/modules/40-data-types/20-floating-type/ru/data.yml index 658b022..ceff378 100644 --- a/modules/40-data-types/20-floating-type/ru/data.yml +++ b/modules/40-data-types/20-floating-type/ru/data.yml @@ -1,3 +1,4 @@ +--- name: Типы данных с плавающей точкой tips: - > diff --git a/modules/40-data-types/30-logic-type/description.ru.yml b/modules/40-data-types/30-logic-type/description.ru.yml deleted file mode 100644 index bfde2d1..0000000 --- a/modules/40-data-types/30-logic-type/description.ru.yml +++ /dev/null @@ -1,37 +0,0 @@ ---- -name: Логический тип -theory: | - Начиная с 11 стандарта в С++ был добавлен новый для этого языка тип по имени `bool`. Он назван в честь английского математика Джорджа Буля, разработавшего математическое представление законов логики. - - Булевская переменная может принимать два значения: `true` (истина) или `false` (ложь). Для хранения булевской переменной выделяется один байт: - - ```cpp - bool is_even; - std::cout sizeof(is_even); // => 1 - ``` - - В ранних версиях языка С++ булевский тип отсутствовал. В место этого С++ интерпретировал ненулевые значения как `true`, а нулевые как `false`. - - Литералы `true` и `false` могут быть преобразованы в тип `int`, причем истина будет преобразована в единицу, а ложь - в ноль: - - ```cpp - int is_even { true }; - int promise { false }; - std::cout << is_even; // => 1 - std::cout << promise; // => 0 - ``` - - Кроме того, любое числовое значение может быть преобразовано неявно в значение `bool`: - - ```cpp - bool is_even { 0 }; // false - bool promise { 1 }; // true - false == 0; // true - ``` - -instructions: | - Допишите программу, которая принимает в качестве аргумента командной строки число и определяет его четность. Результат сохраните в переменной типа `bool` и выведите на экран. Не забудьте сделать перевод строки. - -tips: - - | - [bool](https://learn.microsoft.com/ru-ru/cpp/cpp/bool-cpp?view=msvc-170) diff --git a/modules/40-data-types/30-logic-type/ru/data.yml b/modules/40-data-types/30-logic-type/ru/data.yml index d418c89..6467a1d 100644 --- a/modules/40-data-types/30-logic-type/ru/data.yml +++ b/modules/40-data-types/30-logic-type/ru/data.yml @@ -1,3 +1,4 @@ +--- name: Логический тип tips: - | diff --git a/modules/40-data-types/40-char-type/description.ru.yml b/modules/40-data-types/40-char-type/description.ru.yml deleted file mode 100644 index 90f6efc..0000000 --- a/modules/40-data-types/40-char-type/description.ru.yml +++ /dev/null @@ -1,65 +0,0 @@ ---- -name: Тип сhar -theory: | - В этом уроке мы рассмотрим последний целочисленный тип: `char`. - - `сhar` предназначен для хранения символов, таких как буквы и цифры. Но почему же `char` это целочисленный тип? Все дело в том, что хранение чисел в памяти компьютера не представляет сложности, тогда как хранение букв связанно с рядом проблем. Поэтому в языках программирования принят простой подход: хранить символы в памяти компьютеров в виде числовых кодов. - - Таким образом тип `char` является еще одним целочисленным типом. - - ```cpp - int main() { - char symbol { 'M' }; - int number { symbol }; - std::cout << symbol << std::endl; // => M - std::cout << number << std::endl; // => 77 - return 0; - } - ``` - - Интересный момент состоит в том, что на самом деле и в переменной `number` и в переменной `symbol` хранится значение _77_, а когда дело доходит до вывода объект `cout` по-разному интерпретирует эти переменные. - - `char` по умолчанию может быть как беззнаковым так и знаковым типом. Тут все зависит от компилятора. Если для нас крайне важно, что бы тип `char` обладал определенным поведением, надо указать это явно: - - ```cpp - unsigned char symbol // беззнаковый, диапазон от 0 до 255 - signed char symbol // знаковый, диапазон от -128 до 127 - ``` - - Под переменную типа `char` выделяется один байт, для работы с символами в кодировке ASCII этого вполне достаточно, но для работы с Unicode нет. Если мы попытаемся определить переменную типа `char`, получим ошибку переполнения: - - ```cpp - int main() { - char symbol = 'Ф'; - return 0; - } - ``` - -
- main.cpp:2:21: error: narrowing conversion of '53412' from 'int' to 'char' [-Wnarrowing] - 2 | char symbol = 'Ф'; - | ^~~ -- - Для работы с символами которые превышают один байт, есть расширенный тип `wchar_t` - под котрый выделяется два байта памяти, а начиная со стандарта С++ 11 `char16_t` и `char32_t`: - - ```cpp - int main() { - wchar_t symbol = L'Ф'; - char16_t symbol_16 = u'Ю'; - char32_t symbol_32 = U'Д'; - return 0; - } - ``` - Что бы указать принадлежность к тому или иному символьному типу, перед символом ставится префикс. Например, префикс `L` обозначает расширенный строковый литерал. - -instructions: | - Определите внутри функции `main()` переменную типа `char` и сохраните в нее символ `U`. Выведите значение переменной в консоль. - -tips: - - | - [setlocale](https://learn.microsoft.com/ru-ru/cpp/c-runtime-library/reference/setlocale-wsetlocale?view=msvc-170) - - | - [Стандартная библиотека](https://learn.microsoft.com/ru-ru/cpp/build/reference/std-specify-language-standard-version?view=msvc-170) - - | - [ASCII](https://ru.wikipedia.org/wiki/ASCII) diff --git a/modules/40-data-types/40-char-type/ru/data.yml b/modules/40-data-types/40-char-type/ru/data.yml index c26a3b2..56d16e6 100644 --- a/modules/40-data-types/40-char-type/ru/data.yml +++ b/modules/40-data-types/40-char-type/ru/data.yml @@ -1,3 +1,4 @@ +--- name: Тип сhar tips: - > diff --git a/modules/40-data-types/50-type-casting/description.ru.yml b/modules/40-data-types/50-type-casting/description.ru.yml deleted file mode 100644 index 0811a2c..0000000 --- a/modules/40-data-types/50-type-casting/description.ru.yml +++ /dev/null @@ -1,101 +0,0 @@ ---- -name: Преобразование типов -theory: | - Как мы уже поняли разные типы данных по-разному хранятся в памяти компьютера. Например, целочисленное значение 3 может быть сохранено как двоичное 0000 0000 0000 0000 0000 0000 0000 0011, тогда как значение с плавающей запятой 3.0 может быть сохранено как двоичное 0100 0000 0100 0000 0000 0000 0000 0000. - - Так что же происходит, когда мы инициализируем переменную типа `float` целым числом? - - ```cpp - float f_age { 18 }; - ``` - - А произойдет следующее: поскольку компилятор не может просто сохранить целое число в переменную с плавающей точкой, он преобразует это число в эквивалентное, но типа `float`. - - Процесс преобразования значения из одного типа данных в другой тип данных называется преобразованием типа. - - Преобразования типов могут быть неявные - по решению компилятора и явные по решению программиста. - - Неявное преобразование типа выполняется компилятором автоматически, когда требуется один тип данных, но предоставляется другой тип. Подавляющее большинство преобразований типов в C++ являются неявными преобразованиями типов. - - Неявное преобразование может происходить в следующих случаях: - - * если мы делаем арифметическую операцию двух разных типoв. Преобразование идет в тот тип, который шире. - - ```cpp - double d_result { 4.2 / 3 } // 3 будет преобразована в double - ``` - - * если происходит инициализация переменной другого типа. Например, мы сохраним число типа `long` в переменную типа `int`. И тут преобразование уже будет зависеть от способа инициализации переменной. - - ```cpp - double d_num { 3 }; // 3 будет преобразована в double - int num = 3.14; // будет потеряна вещественная часть - int num = { 3.14 }; // ошибка компиляции - ``` - - * Забегая вперед. При использовании небулевого значения в инструкции if - - ```cpp - if (5) { } // 5 будет преобразованно в true - ``` - - Но явное всегда лучше, чем неявное. В C++ существует 5 различных видов приведений типа: приведения в стиле C, статические приведения, константные приведения, динамические приведения и реинтерпретирующие приведения. Последние четыре иногда называют именованными приведениями. - - Здесь мы рассмотрим приведение в стиле С и статическое приведение. - - В стандартном программировании на C приведение типов выполняется с помощью оператора (), при этом имя типа, в который необходимо преобразовать значение, помещается в круглые скобки. Вы всё еще можете увидеть, что они используются в коде, преобразованном из C. - - ```cpp - int main() { - int x { 5 }; - int y { 4 }; - double d_result { (double)x / y }; // x преобразуется в тип double - } - ``` - - В приведенном выше коде мы явно указали компилятору, чтобы он преобразовал `int x` в `double`. - - C++ также позволяет вам использовать приведение в стиле C с синтаксисом, более похожим на вызов функций: - - ```cpp - double d_result { double(x) / y }; - ``` - Это работает идентично предыдущему примеру, но тут преимущество в том, что преобразуемое значение заключено в скобки - это упрощает определение того, что конвертируется. - - Приведение в стиле С не рекомендуется использовать, поскольку под капотом он может выполнять множество различных преобразований в зависимости от контекста и даже убирать константность. - - В C++ появился оператор приведения типов `static_cast`, который можно использовать для преобразования значения одного типа в значение другого типа. Этот способ наиболее безопасен и рекомендован в С++. - - ```cpp - int main() { - int x { 5 }; - int y { 4 }; - double d_result { static_cast
- 125.0 - 7500.0 -- - Считаем, что: - - - 1 евро = 1.65 долларов - - 1 доллар = 60 рублей - -tips: - - | - [static_cast](https://learn.microsoft.com/ru-ru/cpp/cpp/static-cast-operator?view=msvc-170) - - | - [Преобразования типов и безопасность типов](https://learn.microsoft.com/ru-ru/cpp/cpp/type-conversions-and-type-safety-modern-cpp?view=msvc-170) diff --git a/modules/40-data-types/50-type-casting/ru/data.yml b/modules/40-data-types/50-type-casting/ru/data.yml index ca62599..8499e23 100644 --- a/modules/40-data-types/50-type-casting/ru/data.yml +++ b/modules/40-data-types/50-type-casting/ru/data.yml @@ -1,3 +1,4 @@ +--- name: Преобразование типов tips: - > diff --git a/modules/40-data-types/60-type-alias/description.ru.yml b/modules/40-data-types/60-type-alias/description.ru.yml deleted file mode 100644 index f25d1ea..0000000 --- a/modules/40-data-types/60-type-alias/description.ru.yml +++ /dev/null @@ -1,65 +0,0 @@ ---- -name: Алиасы типов typedef и using -theory: | - В этом уроке мы научимся как определять псевдонимы типов в С++ и узнаем для чего это может быть полезным. - - Представьте что мы пишем программу, которая рассчитывает расстояние пройденное автомобилем за какое-то время и мы бы могли написать ее так: - - ```cpp - int main() { - int speed_in_kilometers { 60 }; - int travel_time { 2 }; - int distance { speed_in_kilometers * travel_time }; - } - ``` - - Но возможность определения псевдонимов может сделать наш код более выразительным, синтаксис определения псевдонимов таков `typedef <тип> <псевдоним>`: - - ```cpp - int main() { - typedef int kilometers_per_hour_t; // определяем kilometers_per_hour_t как псевдоним типа int - typedef int hour_t; - kilometers_per_hour_t speed { 60 }; - hour_t travel_time { 2 }; - } - ``` - - Названия типов стали длиннее, но более понятыми. Мы видим, что скорость измеряется в километрах в час, а время в часах. Обратите внимание, что хорошей практикой является у псевдонима определять суффикс `_t`. Это помогает указать, что идентификатор представляет собой тип, а не переменную или функцию, а также помогает предотвратить конфликты имен с другими типами идентификаторов. - - Когда еще нам могут понадобиться псевдонимы? Например, для определения псевдонимов сложных типов. Далее в курсе мы будем иметь дело с такими типами данных, как вектор и пара. Их определение довольно громоздко: - - ```cpp - #include
- Length domain name: 11 -- - - У функции `length()` есть алиас `size()`, который работает идентично: - - ```cpp - int main() { - std::string domain { "code-basics" }; - std::cout << "Length domain name:" << " " << domain.size() << std::endl; - return 0; - } - ``` - -
- Length domain name: 11 -- - `std::string` сложен и использует многие языковые функции, которые мы еще не рассмотрели. Но нам не нужно разбираться в этих сложностях, чтобы использовать std::string для простых задач, таких как простейший ввод и вывод строк. - - Мы рекомендуем начать экспериментировать со строками уже сейчас, а дополнительные возможности строк мы рассмотрим позже. - -instructions: | - Допишите программу, которая принимает в качестве аргумента командной строки доменное имя, сохраните его в переменную типа `std::string` и выведите на экран количество символов этого имени. Получить аргумент командной строки можно таким образом: `argv[1]` -tips: - - | - [std::string](https://en.cppreference.com/w/cpp/string/basic_string) diff --git a/modules/40-data-types/80-string-type/ru/data.yml b/modules/40-data-types/80-string-type/ru/data.yml index 8ec8d83..32921eb 100644 --- a/modules/40-data-types/80-string-type/ru/data.yml +++ b/modules/40-data-types/80-string-type/ru/data.yml @@ -1,3 +1,4 @@ +--- name: Строки и std::string tips: - | diff --git a/modules/45-ref-and-pointers/10-introduction-pointers/description.ru.yml b/modules/45-ref-and-pointers/10-introduction-pointers/description.ru.yml deleted file mode 100644 index c9609d9..0000000 --- a/modules/45-ref-and-pointers/10-introduction-pointers/description.ru.yml +++ /dev/null @@ -1,83 +0,0 @@ ---- -name: Введение в указатели -theory: | - **Переменная** — это имя части памяти, которая содержит значение. Когда наша программа создает экземпляр переменной, ей автоматически присваивается адрес свободной памяти. Любое значение, которое мы присваиваем переменной, сохраняется в памяти с этим адресом. - - Например: - - ```cpp - int num {}; - ``` - - Когда эта инструкция выполняется процессором, будет выделена часть памяти из ОЗУ. - - В качестве примера предположим, что переменной `num` присвоена ячейка памяти 140. Когда программа видит переменную `num` в выражении или инструкции, она знает, что она должна искать значение в ячейке памяти 140. - - Нам не нужно беспокоиться о том, какой адрес памяти назначен. Мы просто ссылаемся на переменную по ее заданному идентификатору, и компилятор переводит это имя в соответствующий адрес памяти. - - Однако у этого подхода есть некоторые ограничения, которые мы обсудим в этом и будущих уроках. - - ## Оператор адреса `&` - - Оператор адреса `&` позволяет нам увидеть, какой адрес памяти назначен переменной. Это довольно просто: - - ```cpp - #include
- 5 - 0x7ffff5af31bc -- - На вашей машине адрес может быть другой, так как адресное пространство памяти, выделенное программе, может отличаться. - - Хотя оператор адреса выглядит как оператор побитового И, их можно различить, поскольку оператор адреса является унарным, тогда как оператор побитового И является бинарным. - - Теперь имея адрес переменной мы можем с помощью оператора косвенного обращения, получить значение переменной. - - ## Оператор косвенного обращения `*` - - **Оператор косвенного обращения** `*` также называют оператором разыменования. Он позволяет получить доступ к значению по определенному адресу: - - ```cpp - #include
- 5 - 0x7fff9cea58c0 - 5 -- - Теперь у нас есть важные инструменты для работы с указателями, к изучению которых мы приступим в следующем уроке. - -instructions: | - Допишите программу, которая берет значение по адресу переменной `domain_name` и выводит на экран. -tips: - - | - [Оператор косвенного обращения](https://learn.microsoft.com/ru-ru/cpp/cpp/indirection-operator-star?view=msvc-170) - - | - [Оператор address-of: &](https://learn.microsoft.com/ru-ru/cpp/cpp/address-of-operator-amp?view=msvc-170) diff --git a/modules/45-ref-and-pointers/10-introduction-pointers/ru/data.yml b/modules/45-ref-and-pointers/10-introduction-pointers/ru/data.yml index 090e2eb..1951d65 100644 --- a/modules/45-ref-and-pointers/10-introduction-pointers/ru/data.yml +++ b/modules/45-ref-and-pointers/10-introduction-pointers/ru/data.yml @@ -1,3 +1,4 @@ +--- name: Введение в указатели tips: - > diff --git a/modules/45-ref-and-pointers/20-pointers/description.ru.yml b/modules/45-ref-and-pointers/20-pointers/description.ru.yml deleted file mode 100644 index 77d835e..0000000 --- a/modules/45-ref-and-pointers/20-pointers/description.ru.yml +++ /dev/null @@ -1,240 +0,0 @@ ---- -name: Указатели -theory: | - Когда в инструменты добавлены оператор адреса и оператор косвенного обращения, можно поговорить об указателях. В этом уроке мы узнаем что такое указатели, как их объявлять и какие существуют базовые операции. - - ## Объявление указателя - - **Указатель** — это переменная, которая в качестве значения хранит адрес памяти. - - Переменные-указатели объявляются так же, как обычные переменные. Только в этом случае ставится звездочка между типом данных и именем переменной. Эта звездочка не является косвенным обращением. Это часть синтаксиса объявления указателя: - - ```cpp - int *i_ptr {}; // указатель на значение типа int - double *d_ptr {}; // указатель на значение типа double - - int* i_ptr2 {}; // тоже допустимый синтаксис - int * iPtr3{}; // тоже допустимый синтаксис (но не делайте так, это похоже на умножение) - ``` - - Синтаксически C++ принимает звездочку рядом с типом данных, рядом с именем переменной или даже в середине. - - При объявлении переменной-указателя нужно ставить звездочку рядом с типом, чтобы его было легче отличить от косвенного обращения. - - Как и обычные переменные, указатели не инициализируются при объявлении. Если они не инициализированы значением, они будут содержать мусор. - - Указатель X (где X – какой-либо тип) — это обычно используемое сокращение для «указателя на X». Поэтому, когда мы говорим «указатель int», мы на самом деле имеем в виду «указатель на значение типа int». - - Хорошей практикой считается инициализировать указатель значением. - - ## Присвоение значения указателю - - Поскольку указатели содержат только адреса, когда мы присваиваем значение указателю, это значение должно быть адресом. Одна из самых распространенных вещей, которые делают с указателями, – это хранение в них адреса другой переменной. - - Чтобы получить адрес переменной, мы используем оператор адреса: - - ```cpp - #include
- 0x7ffc5d336fc8 - 0x7ffc5d336fc8 -- - `ptr` содержит адрес значения переменной, поэтому мы говорим, что `ptr` «указывает на» `num`. - - Тип указателя должен соответствовать типу переменной, на которую он указывает: - - ```cpp - int i_value { 5 }; - double d_value { 7.0 }; - - int* i_ptr { &iValue }; // ok - double* d_ptr { &dValue }; // ok - i_ptr = &d_value; // ошибка - ``` - - Тип `double` не может указывать на адрес переменной типа `int`. Следующее также некорректно: - - ```cpp - int* ptr { 5 }; - ``` - Это связано с тем, что указатели могут содержать только адреса, а целочисленный литерал 5 не имеет адреса памяти. Если попробовать сделать это, компилятор сообщит, что он не может преобразовать `int` в указатель `int`. - - Вопрос на засыпку: Можно ли инициализировать указатель, явно указав адрес ячейки памяти? - - ```cpp - double* d_ptr{ 0x0012FF7C }; - ``` - - Ответ - нет, компиляция этого кода завершится с ошибкой! Хотя казалось бы, почему, ведь оператор адреса `&`, так же возвращает адрес? Тут есть отличие - оператор `&` возвращает тоже указатель. - - ## Возвращение указателя оператором адреса - - Оператор адреса (&) не возвращает адрес своего операнда в виде литерала. Вместо этого он возвращает указатель, содержащий адрес операнда, тип которого является производным от аргумента. Например, взятие адреса значения `int` вернет адрес в указателе `int`. - - Мы можем увидеть это в следующем примере: - - ```cpp - #include
- int * -- - При компиляции gcc вместо этого выводит "pi" («pointer to int», указатель на int). - - Одной из основных операций является получение значения переменной через указатель - косвенное обращение. - - ## Косвенное обращение через указатели - - У нас есть переменная-указатель, которая указывает на что-то. Значит, другая распространенная вещь, которую мы делаем с ней, — это косвенное обращение через указатель. Это нужно, чтобы получить значение того, на что он указывает. - - Косвенное обращение через указатель вычисляет содержимое адреса, на который он указывает: - - ```cpp - int value { 5 }; - std::cout << &value << std::endl; // выводит адрес value - std::cout << value << std::endl; // выводит содержимое value - - int* ptr { &value }; // ptr указывает на value - std::cout << ptr << std::endl; // выводит адрес, содержащийся в ptr, который равен &value - std::cout << *ptr << std::endl; // косвенное обращение через ptr — получаем значение, на которое указывает ptr - ``` - - Эта программа создает следующий вывод: - -
- 0x7ffcc0b6824c - 5 - 0x7ffcc0b6824c - 5 -- - Без типа при косвенном обращении через указатель он не знал бы, как интерпретировать содержимое, на которое он указывает. По этой же причине тип указателя и тип переменной, адрес которой ему присваивается, должны совпадать. Если бы это было не так, косвенное обращение через указатель неверно интерпретировало бы биты как другой тип. - - После присваивания значению указателя можно присвоить другое значение: - - ```cpp - int value1{ 5 }; - int value2{ 7 }; - - int* ptr{}; - - ptr = &value1; // ptr указывает на value1 - std::cout << *ptr; // выводит 5 - - ptr = &value2; // ptr теперь указывает на value2 - std::cout << *ptr; // выводит 7 - ``` - - Когда адрес переменной `value` присваивается указателю `ptr`, верно следующее: - - - `ptr` равен `&value` - - `*ptr` обрабатывается так же, как value - - Поскольку `*ptr` обрабатывается так же, как `value`, можно присваивать ему значения, как если бы это была переменная `value`. - - Следующая программа напечатает 7: - - ```cpp - int value { 5 }; - int* ptr { &value }; // ptr указывает на value - - *ptr = 7; // *ptr - это то же, что и value, которому присвоено 7 - std::cout << value << std::endl; // выводит 7 - ``` - - Обратите внимание, через указатель мы можем работать с переменной `value` - получить значение, и даже изменить его. - - Такой мощный механизм имеет свои минусы. - - ## Предупреждение о косвенном обращении через недействительные указатели - - Указатели в C++ по своей сути небезопасны. Неправильное использование указателей — один из лучших способов вывести приложение из строя. - - Во время косвенного обращения через указатель приложение пытается перейти в ячейку памяти, которая хранится в указателе, и получить содержимое памяти. В целях безопасности современные операционные системы используют приложения-песочницы. Они предотвращают неправильное взаимодействие операционной системы с другими приложениями и защитить стабильность самой операционной системы. - - Если приложение пытается получить доступ к области памяти, не выделенной ему операционной системой, операционная система может завершить работу приложения. - - Следующая программа иллюстрирует это и вероятнее всего упадет с ошибкой: - - ```cpp - #include
- Starting main() - Hello Code Basics! - Ending main() -- - Программа начинает выполняться с начала функции `main`. Первая строка, которая будет выполняться, выводит текст `Starting main()`. Вторая строка в `main` — это вызов функции `Greating()`. Мы вызываем функцию `Greating()`, добавляя скобки к имени функции, например: `Greating()`. Если забыть про скобки, программа может не компилироваться. - - Так как мы вызвали функцию, выполнение инструкций в `main` приостанавливается. В итоге выполнение переходит к началу вызываемой функции `Greating()`. - - Первая и единственная строка в `Greating()` печатает текст `Hello Code Basics!`. Когда `Greating()` завершается, выполнение возвращается к вызывающей функции — здесь: функция `main`. Далее оно возобновляется с того места, где остановилось. Получается, следующая инструкция, которая выполняется в `main`, выводит на печать `Ending main()`. - - ### Особенность функций - - Полезная особенность функций заключается в том, что их можно вызывать более одного раза в разных частях программы. Вот программа, которая демонстрирует это: - - ```cpp - void Greating() { - std::cout << "Hello Code Basics!" << std::endl; - } - - int main() { - Greating(); - Greating(); - return 0; - } - ``` - - Обратите внимание мы вызываем функцию `Greating()` два раза. - - Эта программа создает следующий вывод: - -
- Hello Code Basics! - Hello Code Basics! -- - В отличие от некоторых других языков программирования в C++ функции не могут быть определены внутри других функций. Следующая программа не является корректной: - - ```cpp - int main() { - void Greating() { - std::cout << "Hello Code Basics!"; - } - Greating(); - return 0; - } - ``` - В коде выше мы попытались определить внутри функции `main()`, функцию `Greating()` и получили ошибку компиляции: - -
- main.cpp:12:3: error: ‘PrintMoto’ was not declared in this scope - 12 | PrintMoto(); - | ^~~~~~~~~ -- - Это прежде всего связанно с особенностью языка и выделением памяти. - - В этом уроке мы рассмотрели структуру и создание функций. Понятие «создать функцию» имеет много синонимов: «реализовать», «определить» и даже «заимплементить» (от слова implement). Все они встречаются в повседневной практике на работе. - -instructions: | - Напишите функцию `PrintMoto()`, которая выводит на экран фразу `Spring is coming`, и вызовите ее внутри функции `main`. - -tips: - - | - [Функции в С++](https://learn.microsoft.com/ru-ru/cpp/cpp/functions-cpp?view=msvc-170) diff --git a/modules/50-functions/10-define-function/ru/data.yml b/modules/50-functions/10-define-function/ru/data.yml index 59ad414..dd39017 100644 --- a/modules/50-functions/10-define-function/ru/data.yml +++ b/modules/50-functions/10-define-function/ru/data.yml @@ -1,3 +1,4 @@ +--- name: Определение функций tips: - > diff --git a/modules/50-functions/20-function-return/description.ru.yml b/modules/50-functions/20-function-return/description.ru.yml deleted file mode 100644 index 1eaee6a..0000000 --- a/modules/50-functions/20-function-return/description.ru.yml +++ /dev/null @@ -1,119 +0,0 @@ ---- -name: Возврат значений -theory: | - Функции, которые мы определяли в предыдущих уроках, заканчивали свою работу тем, что печатали на экран какие-то данные: - - ```cpp - void Greating() { - std::cout << "Hello Code Basics!" << std::endl; - } - - - int main() { - Greating(); - return 0; - } - ``` - - Пользы от таких функций не очень много, так как результатом их работы невозможно воспользоваться внутри программы. - - В этом уроки мы рассмотрим, как сделать наши функции полезными. - - ## Возвращаемые значения - - Когда мы пишем пользовательскую функцию, мы можем определить, будет ли она возвращать значение вызывающей стороне или нет. - - Чтобы вернуть значение вызывающей стороне, необходимы две вещи: - - * Функция должна указать, значение какого типа будет возвращено. Это делается путем установки типа возвращаемого значения функции, который является типом, определенным перед именем функции - * Внутри функции, которая будет возвращать значение, используем инструкцию `return`. Это нужно, чтобы указать конкретное значение, возвращаемое вызывающей стороне. Конкретное значение, возвращаемое функцией, называется возвращаемым значением. Когда инструкция `return` выполняется, возвращаемое значение копируется из функции обратно в вызывающую функцию. Этот процесс называется возвратом по значению - - Общий вид функции: - - ```cpp - <тип возвращаемого значения> имя функции(аргументы) { - return возвращаемое значение - } - ``` - - Рассмотрим простую функцию, которая возвращает строку. Также пример программы, которая ее вызывает: - - ```cpp - #include
- number = 1 - result = 2 -- - Обратим внимание на следующее: - - - Мы передаем переменную в вызов функции `AddOne`. При компиляции программы вместо переменной подставляется ее значение - - Значение переменной `number` не изменяется. Это происходит из-за того, что параметр передается по значению. Также при вызове функции внутри нее создается локальная переменная `num`, в которую копируется значение переменной `number`. Так функция работает с копией данных, а не с их оригиналом - - Этот процесс и называется передачей параметров в функцию по значению. - - Итак, в этом уроке мы изучили понятия параметров и аргументов функции, а также то, как передавать параметры в функцию по значению. В C++ еще существует способ передачи параметров по ссылке, о котором мы поговорим в следующих уроках. - -instructions: | - Реализуйте функцию `Remainder()`, которая принимает число типа `int` и возвращает остаток от деления на два. Остаток от деления можно взять с помощью оператора `%`. -tips: - - | - [Функции С++](https://learn.microsoft.com/ru-ru/cpp/cpp/functions-cpp?view=msvc-170) - diff --git a/modules/50-functions/30-functions-parameters/ru/data.yml b/modules/50-functions/30-functions-parameters/ru/data.yml index 947425f..e5eea7b 100644 --- a/modules/50-functions/30-functions-parameters/ru/data.yml +++ b/modules/50-functions/30-functions-parameters/ru/data.yml @@ -1,3 +1,4 @@ +--- name: Параметры и аргументы функции tips: - > diff --git a/modules/50-functions/40-function-prototypes/description.ru.yml b/modules/50-functions/40-function-prototypes/description.ru.yml deleted file mode 100644 index 4b92602..0000000 --- a/modules/50-functions/40-function-prototypes/description.ru.yml +++ /dev/null @@ -1,68 +0,0 @@ ---- -name: Прототипы функций -theory: | - В этом уроке мы познакомимся с прототипами функций. Также узнаем, как их определять и для чего они нужны. - - ## Определение прототипа. - - Посмотрим на код: - - ```cpp - #include
- 3 - 4.6 -- - Когда мы передаем целочисленные аргументы в вызове Add(1, 2), компилятор определяет, что мы пытаемся вызвать функцию Add(int, int). А когда мы передаем аргументы с плавающей запятой в вызове Add(1.2, 3.4), компилятор определяет, что мы пытаемся вызвать функцию Add(double, double). - - Перегрузка функций предоставляет отличный способ снизить сложность программы за счет уменьшения количества имен функций, которые нужно запомнить. Ее можно и нужно использовать. - - -instructions: | - Напишите две перегруженные функции Cube, которые принимают числа и возводят их в третью степень. Одна функция должна работать с целочисленными значениями (int), а другая — с числами с плавающей точкой (float). - -tips: - - | - [Перегрузка функций](https://learn.microsoft.com/ru-ru/cpp/cpp/function-overloading?view=msvc-170) diff --git a/modules/50-functions/50-reloading-function/ru/data.yml b/modules/50-functions/50-reloading-function/ru/data.yml index 9835a47..dca6b5f 100644 --- a/modules/50-functions/50-reloading-function/ru/data.yml +++ b/modules/50-functions/50-reloading-function/ru/data.yml @@ -1,3 +1,4 @@ +--- name: Перегрузка функций tips: - > diff --git a/modules/50-functions/60-types-overloads-function/description.ru.yml b/modules/50-functions/60-types-overloads-function/description.ru.yml deleted file mode 100644 index c30328d..0000000 --- a/modules/50-functions/60-types-overloads-function/description.ru.yml +++ /dev/null @@ -1,98 +0,0 @@ ---- -name: Виды перегрузок функций -theory: | - В предыдущем уроке мы познакомились с перегрузкой функций, которая позволяет нам создавать несколько функций с одинаковыми именами, если все эти функции имеют разные параметры. - - В этом уроке мы поговорим о том, как различаются перегруженные функции. Перегруженные функции, которые не различаются должным образом, приведут к тому, что компилятор выдаст ошибку компиляции. - - Самый простой способ дифференцировать перегрузку функций – убедиться, что каждая перегруженная функция имеет отличающийся набор параметров. - - ## Перегрузка по количеству параметров - - Перегруженные функции различаются до тех пор, пока каждая перегруженная функция имеет разное количество параметров. Например: - - ```cpp - #include
- User full name: John Doe - User email: john@gmail.com - User full name: john - User email: john@gmail.com -- - Компилятор может легко сказать, что вызов функции с одним параметром `std::string` должен идти на `GetUserData(std::string)`, а вызов функции с тремя параметрами `std::string` должен идти на `GetUserData(std::string, std::string, std::string)`. - - Обратите внимание от количества параметров может меняться и поведение функции, из примера видно, что если пользователь не ввел имя и фамилию, мы попытаем получить его полное имя из почтового адреса. - - ## Перегрузка по типу параметров - - Функции также можно различать, если различаются наборы типов параметров каждой перегруженной функции. Например, различаются все следующие перегрузки: - - ```cpp - int Add(int x, int y); // целочисленная версия - double Add(double x, double y); // версия с плавающей запятой - double Add(int x, double y); // смешанная версия - double Add(double x, int y); // смешанная версия - ``` - - Поскольку псевдонимы типов (или определения `typedef`) не являются отдельными типами, перегруженные функции, использующие псевдонимы типов, не отличаются от перегрузок, использующих исходные типы. Например, все следующие перегрузки не различаются (и приведут к ошибке компиляции): - - ```cpp - typedef int height_t; // typedef - using age_t = int; // псевдоним типа - - void Print(int value); - void Print(age_t value); // не отличается от print(int) - void Print(height_t value); // не отличается от print(int) - ``` - - Для параметров, передаваемых по значению, квалификатор `const` также не учитывается. Поэтому следующие функции не считаются разными: - - ```cpp - void Print(int); - void Print(const int); // не отличается от print(int) - ``` - - Это код также вызовет ошибку компиляции. - - Тип возвращаемого значения функции не учитывается при различении перегруженных функций. - - Рассмотрим случай, когда вы хотите написать функцию, возвращающую случайное число, но вам нужна одна версия, которая вернет `int`, и другая версия, которая вернет `double`. У вас может возникнуть соблазн сделать так: - - ```cpp - int GetRandomValue(); - double GetRandomValue(); - ``` - - При компиляции такой программы возникнет ошибка и это логично, компилятор видит вызов функции `GetRandomValue()`, как понять какую функцию вызывать? - - Выход из этой ситуации дать функциям разные имена. - - В этом уроке мы узнали, что функции можно перегружать по типу передаваемых параметров и по количеству передаваемых параметров. - -instructions: | - Напишите две перегруженные функции `std::string GetFullName()`. Эти функции должны работать так: если в функцию передали имя и фамилию, то она возвращает полное имя, если в функцию не передали аргументы, она вернет строку "Anonymous". - -tips: - - | - [Перегрузка функций](https://learn.microsoft.com/ru-ru/cpp/cpp/function-overloading?view=msvc-170) diff --git a/modules/50-functions/60-types-overloads-function/ru/data.yml b/modules/50-functions/60-types-overloads-function/ru/data.yml index 250e2da..f0feba5 100644 --- a/modules/50-functions/60-types-overloads-function/ru/data.yml +++ b/modules/50-functions/60-types-overloads-function/ru/data.yml @@ -1,3 +1,4 @@ +--- name: Виды перегрузок функций tips: - > diff --git a/modules/50-functions/70-default-arguments/description.ru.yml b/modules/50-functions/70-default-arguments/description.ru.yml deleted file mode 100644 index 55f8441..0000000 --- a/modules/50-functions/70-default-arguments/description.ru.yml +++ /dev/null @@ -1,121 +0,0 @@ ---- -name: Аргументы по умолчанию -theory: | - **Аргумент по умолчанию** – это значение по умолчанию, предоставленное для параметра функции. Если пользователь не предоставляет явный аргумент для параметра с аргументом по умолчанию, то будет использоваться значение по умолчанию. Если пользователь предоставляет аргумент для этого параметра, то используется аргумент, предоставленный пользователем. - - Поскольку пользователь может выбрать, следует ли предоставить конкретное значение аргумента или использовать значение по умолчанию, параметр с предоставленным значением по умолчанию часто называется необязательным параметром. - - Посмотрим на следующий код: - - ```cpp - #include