Computer Science
Відкрити в Telegram
По всем вопросам: @altmainf Уважаемый менеджер: @altaiface
Показати більше7 922
Підписники
-124 години
-57 днів
-2930 день
Архів дописів
7 922
Распространение констант
Распространение констант — оптимизация, заменяющее выражение, которое при выполнении всегда возвращает одну и ту же константу, самой этой константой. Это может быть константа, определённая ранее, или встроенная функция, применённая к константам.
Рассмотрим следующий пример:
int x = 14;
int y = 7 - x / 2;
return y * (28 / x + 2);
Распространение x возвращает:
int x = 14;
int y = 7 - 14 / 2;
return y * (28 / 14 + 2);
Далее, свёртка констант и распространение y возвращают следующее:
int x = 14;
int y = 0;
return 0;7 922
Свёртка констант
Свёртка констант — оптимизация, вычисляющая константные выражения на этапе компиляции. Прежде всего, упрощаются константные выражения, содержащие числовые литералы(фиксированное значение). Также могут быть упрощены выражения, содержащие никогда не изменяемые переменные или переменные, объявленные как константы.
Рассмотрим пример:
i = 320 * 200 * 32;Компилятор, поддерживающий свёртку констант, не будет генерировать две инструкции умножения и запись полученного результата. Вместо этого он распознает эту конструкцию как константное выражение и заменит её на вычисленное значение (в данном случае
2 048 000).7 922
Принцип оптимизации удаления общих подвыражений
Применение оптимизации основано на анализе доступных выражений.
Выражение x + y является доступным в некоторой точке p программы, если:
- вдоль любого пути от начального узла к p выражение x + y вычисляется до достижения точки p;
- между вычислениями выражений и достижением точки p нет изменения значений переменных x и y.
Эффективность преобразования главным образом определяется тем, что суммарное время записи и нескольких чтений новой переменной оказывается меньше, чем суммарное время, затрачиваемое на вычисление старого выражения каждый раз, когда оно встречается в коде.
На практике на итоговую эффективность влияет также ряд других факторов, в частности распределение переменных по регистрам.
7 922
Удаление общих подвыражений
Удаление общих подвыражений — это оптимизация компилятора (преобразование), которое уничтожает повторные вычисления общих подвыражений и заменяет их на использование сохраненного после первого вычисления значения.
Рассмотрим следующий фрагмент кода:
a = b * c + g; d = b * c * d;К нему применимо следующее преобразование:
tmp = b * c; a = tmp + g; d = tmp * d;которое окажется эффективным, если суммарное время записи и нескольких чтений новой переменной "tmp" окажется меньше, чем суммарное время, затрачиваемое на вычисление выражения "
b * c" каждый раз, когда оно встречается в коде.7 922
Слияние циклов
Слияние циклов — оптимизация компилятора, выполняющая объединение нескольких циклов, смежных в дереве циклов, в один. Преобразование возможно, если циклы имеют одинаковое количество итераций и не зависят друг от друга по данным. Слияние циклов может повысить локальность данных, что повышает эффективность работы кэша.
Например, код:
int i, a[100], b[100];
for (i = 0; i < 100; i++) {
a[i] = 1;
}
for (i = 0; i < 100; i++) {
b[i] = 2;
}
эквивалентен:
int i, a[100], b[100];
for (i = 0; i < 100; i++) {
a[i] = 1;
b[i] = 2;
}7 922
Расщепление тела цикла
Расщепление тела цикла или Loop fission — оптимизация компилятора, которая разбивает цикл в программе на несколько циклов, каждый из которых имеет те же индексные границы, однако содержит только часть тела исходного цикла.
Например, следующий код:
int i, a[100], b[100];
for (i = 0; i < 100; i++) {
a[i] = 1;
b[i] = 2;
}
в результате применения оптимизации преобразовывается в:
int i, a[100], b[100];
for (i = 0; i < 100; i++) {
a[i] = 1;
}
for (i = 0; i < 100; i++) {
b[i] = 2;
}
На некоторых архитектурах может оказаться более выгодным исполнить два цикла вместо одного объединённого, так как, например, локальность данных в таком случае может оказаться выше.7 922
Оптимизация циклов
Оптимизация циклов - это один из методов внутрипроцедурной оптимизации.
Основные методы оптимизации циклов:
- Раскрытие цикла - вместо выполнения цикла используются отдельные инструкции, которые применяются к нескольким элементам данных сразу. Это уменьшает количество итераций цикла и увеличивает производительность выполнения.
- Использование константной оптимизации - константные значения выносятся за пределы цикла, чтобы они не вычислялись каждый раз во время выполнения цикла.
- Обратный цикл - цикл выполняется в обратном порядке. Это может уменьшить количество пропусков кэша и увеличить локальность ссылок на данные.
- Предварительный выход из цикла - проверка условия выхода из цикла применяется на каждой итерации цикла в начале итерации, вместо того чтобы проверять его в конце итерации.
7 922
Принципы оптимизации, применяемые в различных методах оптимизации в компиляторах
1. Уменьшение избыточности- повторное использование результатов вычислений, сокращение числа перевычислений;
2. Компактификация кода - удаление ненужных вычислений и промежуточных значений;
3. Сокращение числа переходов в коде. Например, использование встраивания функций или размотки цикла позволяет во многих случаях ускорить выполнение программы ценой увеличения размера кода;
4. Локальность - код и данные, доступ к которым необходим в ближайшее время, должны быть размещены рядом друг с другом в памяти, чтобы следовать принципу локальности ссылок;
5. Использование иерархии памяти - размещать наиболее часто используемые данные в регистрах общего назначения, менее используемые — в кэш, ещё менее используемые — в оперативную память, наименее используемые — размещать на диске.
6. Распараллеливание - изменение порядка операций может позволить выполнить несколько вычислений параллельно, что ускоряет исполнение программы.
7 922
Локальная оптимизация
Локальная оптимизация - это метод оптимизации, который применяется к небольшим фрагментам кода, называемым базовыми блоками, для улучшения его производительности. Она применяется в компиляторах на этапе генерации машинного кода для оптимизации программы.
Примеры локальных оптимизаций включают в себя следующие:
- Устранение идентичных операций;
- Устранение избыточных вычислений;
- Замена сложных операций более простыми (например, замена умножения на степень двойки на сдвиг);
- Замена медленных операций более быстрыми;
- Ускорение циклов;
- Встраивание функций;
- Устранение ненужных промежуточных вычислений и т.д.
Такая оптимизация позволяет сократить количество инструкций, необходимых для выполнения программы, что уменьшает время ее работы и повышает производительность.
7 922
Peephole-оптимизация
Метод оптимизации программного кода, который заключается в анализе нескольких последовательных инструкций в программе (называемых "peephole"), чтобы определить, можно ли их заменить более эффективной комбинацией или порядком инструкций.
Для проведения peephole-оптимизации на уровне ассемблера, компилятор анализирует последовательность операций и заменяет их на более оптимизированные.
Например, удвоение числа может быть более эффективно выполнено с использованием левого сдвига или путём сложения числа с таким же.
7 922
Что такое peephole?
Peephole - это небольшой набор последовательных машинных инструкций, который обычно составляется для исправления архитектурных особенностей процессора или определенного набора инструкций.
Peephole может использоваться для оптимизации кода, например, замена нескольких инструкций на эквивалентную одну для сокращения времени выполнения программы.
Также, peephole может использоваться для обработки ошибок в коде, например, если происходит переполнение стека, то в peephole можно добавить инструкцию для очистки стека или добавление проверки на условия, при которых может произойти переполнение.
7 922
Первый в России ускоренный онлайн-бакалавриат «Фронтенд и мобильная разработка» — от Яндекс Практикума и университета ИТМО.
— Учёба 2,5 года вместо четырёх лет
— Диплом государственного образца по направлению «Прикладная информатика»
— Можно выбрать специализацию: фронтенд, iOS- или Android-разработка
— Удобно совмещать с работой: обучение онлайн, 20-25 часов в неделю
— Преподаватели — действующие разработчики
— Стажировка в крупных IT-компаниях
— Системное развитие хард-и софтскилов
Программы рассчитаны на людей со средним профессиональным или высшим образованием.
Обучение почти в два раза короче, потому что формат позволяет перезачесть часть общих предметов и оставить только профильные.
Поступайте, чтобы стать востребованным разработчиком и подготовиться к роли тимлида.
→ Оставить заявку
Реклама АНО ДПО "Образовательные технологии Яндекса", ИНН:7704282033, erid: LjN8KCpNz
7 922
Для чего нужен оптимизирующий компилятор?
Оптимизирующий компилятор - это компилятор, который способен преобразовывать входной код программы с целью повышения её эффективности и оптимизации. Это достигается за счет применения различных алгоритмов оптимизации, которые преобразуют и оптимизируют код для улучшения его производительности.
Могут выполнять различные виды оптимизаций:
1. Устранение неиспользуемого кода.
2. Избавление от повторяющегося кода.
3. Векторизация - преобразование циклов с пересылкой данных между массивами в код, работающий с векторами.
4. Улучшение работы с памятью.
5. Избавление от избыточных вычислений.
6. Использование регистров и кэша процессора.
7 922
Межпроцедурная оптимизация
Межпроцедурная оптимизация - это вид оптимизации в программировании, который позволяет улучшить производительность программы, включая несколько функций и подпрограмм. Она основана на анализе потоков данных и контроля за использованием памяти, и как следствие, на основании этого, устранении избыточных действий.
Применение межпроцедурной оптимизации может ускорить выполнение программы, снизить потребление памяти, а также сократить объем программы за счет удаления неиспользуемого кода или упрощения сложных алгоритмов.
Кроме того, межпроцедурная оптимизация может применяться для устранения повторяющихся действий, таких как перезапись переменных, вызов одной и той же функции с разными аргументами и др.
7 922
Индуктивная переменная
Индуктивная переменная — переменная в циклах, последовательные значения которой образуют арифметическую прогрессию. Наиболее распространенный пример — счётчик цикла. Индуктивные переменные часто используются в индексных выражениях массивов.
Например, в следующем примере, i и j являются индуктивными переменными:
for ( i = 0; i < 10; i++ ) {
j = 17 * i;
}
Традиционные методы оптимизации предусматривают распознавание индуктивных переменных и удаление их из цикла.7 922
Инверсия цикла
Инверсия цикла (англ. Loop inversion) — оптимизация компилятора и трансформация цикла, в ходе которой while-цикл заменяется на оператор ветвления, содержащий do-while-цикл. При правильном использовании данная оптимизация повышает производительность за счет конвейеризации.
Пример на С, код:
int i, a[100];
i = 0;
while (i < 100) {
a[i] = 0;
i++;
}
в результате применения оптимизации преобразовывается в:
int i, a[100];
i = 0;
if (i < 100) {
do {
a[i] = 0;
i++;
} while (i < 100);
}7 922
Вызов по необходимости
Вызов по необходимости - это стратегия передачи аргументов при вызове функции, при которой аргумент вычисляется только тогда, когда он действительно нужен в вычислениях.
В отличие от вызова по значению, где аргумент вычисляется до вызова функции (независимо от того, используется ли он или нет), вызов по необходимости определяет, когда и каким образом вычислять аргументы, анализируя инструкции в теле функции, которые используют значения этих аргументов.
Вызов по необходимости может быть полезен для экономии памяти и ускорения выполнения программы, когда часть аргументов не используется в вычислениях. Однако, вызов по необходимости также может внести определенные задержки в работе программы, связанные с вычислением аргументов во время выполнения.
7 922
Вызов по имени
Вызов по имени - это стратегия передачи аргументов при вызове функции. При вызове по имени аргумент не вычисляется заранее, только когда он необходим внутри функции. Вместо этого, выражение-аргумент передаются как формальный параметр, который заменяется на соответствующее выражение-аргумент в теле функции, только когда значения аргумента действительно требуется.
Вызов по имени может быть полезным, когда некоторые аргументы могут не быть использованными в функции. В этом случае вычисление аргумента не будет совершаться и не будет занимать память. Однако, это также может приводить к существенно большему времени выполнения функции, если какие-то аргументы используются многократно внутри функции.
Вызов по имени часто используется в функциональных языках программирования, таких как Haskell, а также в лямбда-исчислении и функциональных парадигмах программирования в целом.
7 922
Нормальный порядок вычислений
Нормальный порядок вычислений или "ленивые вычисления" — это парадигма вычислений, в которой все аргументы функции вычисляются только при необходимости их использования в вычислении значения функции.
Другими словами, нормальный порядок вычислений означает вычисление выражения от самого "верха" до "низа" в том порядке, в котором выражение записано, и только тогда, когда оно необходимо для дальнейшего вычисления.
Преимущество нормального порядка заключается в возможности сохранять ресурсы, поскольку вычисление аргументов функции, которые не используются в конечном результате, откладывается до последнего момента.
Однако это может приводить к потере производительности в том случае, если одни и те же выражения вычисляются несколько раз.
7 922
Вызов по ссылке
Вызов функции по ссылке - это метод передачи аргументов в функцию, при котором передаются ссылки на переменные, а не их значения. В этом случае функция работает с оригинальными переменными, а не их копиями.
При вызове функции по ссылке вместо передачи значения переменной в качестве аргумента, передается ссылка на эту переменную. Таким образом, вызов функции может изменить значение переменной, на которую ссылается аргумент, и эти изменения будут видны за пределами самой функции.
Таким образом, вызов функции по ссылке означает, что передаются ссылки на переменные, а не их значения. Это позволяет функции изменять значения переменных, на которые ссылаются аргументы.
Вже доступно! Дослідження Telegram за 2025 — головні інсайти року 
