7 922
订阅者
-124 小时
-57 天
-2930 天
帖子存档
7 922
Инкапсуляция в ООП
Инкапсуляция является одним из основных принципов ООп, который обеспечивает сокрытие деталей реализации объекта и предоставление интерфейса для взаимодействия с ним.
Этот подход упрощает использование объектов и является одним из способов защиты данных от различных изменений из-за ошибок в других частях программы.
Определение свойств и методов с модификаторами доступа private, protected или public – один из способов обеспечения инкапсуляции. Это позволяет установить уровни доступа к объектам и их методам, определяя, кто и как может обращаться к свойствам классов.
Инкапсуляция имеет множество преимуществ, таких как уменьшение зависимостей между объектами, увеличение безопасности кода, возможность изменения внутренней реализации класса без влияния на другие части программы, улучшение гибкости и масштабируемости кода.
7 922
Причины возникновения недостижимого кода
Существование недостижимого кода может быть обусловлено разными факторами, например:
- Программные ошибки в сложных условных переходах;
- Вследствие внутренних преобразований, выполняемых оптимизирующим компилятором;
- Неполное тестирование новой или модифицированной программы, которому не удалось обнаружить недостижимый код;
- При исправлении одной ошибки, программист создал другую ошибку, которая обходит недостижимый код и не была обнаружена во время тестирования;
- Устаревший код, который не был полностью удалён программистом, так как он был смешан с действующим кодом;
- Устаревший код, который программист забыл удалить;
- Ранее полезный код, который никогда не будет исполнен, так как, в дальнейшем, ввод данных никогда не приведёт к исполнению этого кода;
- Устаревший код, который был намеренно сохранён, но сделан недостижимым, для того чтобы его можно было при необходимости снова включить в программу;
- Отладочные конструкции и остаточные части кода, которые ещё должны быть удалены из программы.
7 922
Яндекс Практикум и Томский государственный университет запускают совместную программу магистратуры «Дата-аналитика для бизнеса».
Поможем закрепить теорию бакалавриата реальными проектами и получить степень магистра ТГУ за 2 года.
— Обучение онлайн 20-25 часов в неделю
— Диплом государственного образца по направлению «Прикладная информатика»
— Преподаватели – действующие аналитики данных
— 6 месяцев практики, 30 проектов в резюме
— Стажировка в Яндексе, Сбере, Ozon или других ведущих IT-компаниях
Подавайте заявку, чтобы выпуститься уверенным middle-специалистом.
7 922
Что такое недостижимый код?
Недостижимый код - это код в программе, который никогда не будет выполнен в процессе ее работы.
Пример:
int foo(int a) {
int b;
b = a << 2;
return b;
b = 47; // Недостижимый код
return 0; // Недостижимый код
}7 922
Применение хвостовой рекурсии
Хвостовая рекурсия часто применяется в программах на функциональных языках программирования. Многие вычисления на таких языках естественно выражать в виде рекурсивных функций, а возможность автоматической замены транслятором хвостовой рекурсии на итерацию означает, что по вычислительной эффективности она равна эквивалентному коду, записанному в итеративном виде.
Создатели функционального языка Scheme, одного из диалектов Lisp, оценили важность хвостовой рекурсии настолько, что в спецификации языка предписали каждому транслятору этого языка в обязательном порядке реализовывать оптимизацию хвостовой рекурсии и описали точный набор условий, которым должна отвечать рекурсивная функция, чтобы рекурсия в ней была оптимизирована.
7 922
Что такое хвостовая рекурсия?
Хвостовая рекурсия - это способ рекурсивной функции, в котором вызов функции является последней операцией в теле функции.
Имеет одно важное преимущество - она может быть оптимизирована компилятором или интерпретатором, чтобы избежать проблемы переполнения стека при вызове рекурсивной функции многократно. Вместо того чтобы сохранять каждое состояние рекурсивного вызова на стеке, компилятор может использовать только один кадр стека для всех вызовов хвостовой рекурсии.
Пример хвостовой рекурсии в языках программирования, поддерживающих оптимизацию хвостовой рекурсии, таких как Scheme или Clojure, может быть таким:
(define (factorial-iter n acc)
(if (<= n 1)
acc
(factorial-iter (- n 1) (* acc n))))
(define (factorial n)
(factorial-iter n 1))7 922
Удаление мёртвого кода
Удаление мертвого кода важно, чтобы поддерживать код базы проекта в чистоте и упростить его чтение и понимание.
Процесс включает в себя несколько шагов:
1. Поиск кода, не выполняющего никакой полезной работы. Это может быть код, который не используется в любом месте проекта или код, который не имеет никакого влияния на результаты работы проекта.
2. Рефакторинг. После того, как мы выявили ненужный код, мы можем начать процесс рефакторинга. Это может включать в себя удаление этого кода из базы проекта или переписывание его, если он все же нужен для функционала проекта.
3. Тестирование. После удаления мертвого кода мы должны тестировать код, чтобы убедиться, что его удаление не имело никакого влияния на функционал проекта. Об этом не следует забывать, поскольку удаление кода может ухудшить состояние проекта.
7 922
Что такое мёртвый код?
Мертвый код - это кусок кода, который больше не используется в программе и не выполняет никакой полезной работы, но все еще присутствует в исходном коде. Это может быть следствием изменений в требованиях программы или просто из-за того, что кто-то забыл удалить его.
Мертвый код может иметь влияние на производительность программы, поскольку он увеличивает время компиляции и усложняет чтение исходного кода. Кроме того, мертвый код может занимать дополнительное место в памяти и усложнять обслуживание программы.
Чтобы избежать мертвого кода, следует регулярно проверять исходный код и удалять неиспользуемый код. Это также помогает сохранить исходный код более организованным и читаемым. Кроме того, код можно комментировать, чтобы сохранить его для будущих использований, если это необходимо.
7 922
Методы снижения стоимости операций:
1. Автоматизация - внедрение автоматизированных систем и процессов может значительно сократить время и затраты на выполнение задач.
2. Оптимизация процессов производства - анализ и оптимизация процессов производства позволяют поднять эффективность работы и снизить затраты.
3. Инновации - внедрение новых технологий и методов, которые улучшают процессы, может позволить снизить затраты.
4. Управление запасами - оптимизация управления запасами позволяет снизить затраты на их хранение и обработку.
5. Управление процессом закупок - управление процессом закупок и работа с поставщиками позволяют сэкономить на затратах и получить лучшие условия поставки.
6. Улучшение качества работы персонала - постоянное повышение уровня квалификации и мотивация персонала позволяют повысить эффективность работы и снизить вероятность ошибок и изменений в производственных процессах.
7 922
Распределение регистров
Распределение регистров – это процесс оптимизации использования регистров процессора в программной реализации вычислительных процессов.
Это распределение является одним из методов оптимизации производительности кода. Состоит из нескольких шагов:
1. Анализ исходного кода, позволяющий определить переменные, которые используются наиболее часто.
2. Распределение регистров таким образом, чтобы переменные, которые используются наиболее часто, сохранялись в регистрах.
3. Создание кода с использованием этих регистров.
Преимущества использования распределения регистров включают увеличение скорости выполнения программы, уменьшение количества доступов к памяти и улучшение использования ресурсов процессора.
7 922
Размыкание цикла
Размыкание цикла или loop unswitching состоит в вынесении условия за пределы цикла и дублирования тела цикла с помещением соответствующих вариантов в соответствующие ветви условия. Это позволяет улучшить производительность за счёт того, что современные процессоры могут выполнять векторные операции.
Пример преобразования:
for (i = 0; i < 1000; i++) {
x[i] += y[i];
if (w)
y[i] = 0;
}
Условие внутри тела цикла мешает его распараллеливанию. После размыкания оно принимает следующий вид:
if (w) {
for (i = 0; i < 1000; i++) {
x[i] += y[i];
y[i] = 0;
}
}
else {
for (i = 0; i < 1000; i++)
x[i] += y[i];
}7 922
Размотка цикла
Размотка цикла — техника оптимизации компьютерных программ, состоящая в искусственном увеличении количества инструкций, исполняемых в течение одной итерации цикла. Позволяет во многих случаях увеличить количество параллельно исполняемых блоков инструкций и более интенсивно использовать регистры процессора, кэш данных и исполнительных устройств.
Пример, код:
int i;
for ( i = 1; i < n; i++) {
a[i] = (i % b[i]);
}
преобразуется в:
int i;
for (i = 1; i < n - 3; i += 4) {
a[i] = (i % b[i]);
a[i + 1] = ((i + 1) % b[i + 1]);
a[i + 2] = ((i + 2) % b[i + 2]);
a[i + 3] = ((i + 3) % b[i + 3]);
}7 922
Разбиение цикла на блоки
Loop tiling - оптимизирующее преобразование, призванное сделать исполнение некоторых типов циклов более эффективным.
Данный способ оптимизации состоит в разбиении пространства итерирования исходного цикла на небольшие блоки меньшего размера, что позволяет хранить используемые в этих небольших блоках данные в кэше полностью для их неоднократного использования в процессе выполнения блока.
Пример: умножение матрицы на вектор
for (i = 0; i < N; i++)
for (j = 0; j < N; j++)
c[i] = c[i] + a[i, j] * b[j];
После разбиения цикла на блоки 2 × 2:
for (i = 0; i < N; i += 2)
for (j = 0; j < N; j += 2)
for (ii = i; ii < min(i+2, N); ii++)
for (jj = j; jj < min(j+2, N); jj++)
c[ii] = c[ii] + a[ii, jj] * b[ii];7 922
Пул строк
Пул строк относится к двум видам оптимизации компилятора, связанным со строками:
1) Объединение строк из разных модулейПри обработке исходного кода компилятор должен каждую литеральную строку поместить в метаданные управляемого модуля. Если одна строка встречается в исходном коде много раз, размещение всех таких строк в метаданных приведет к росту результирующего файла. Чтобы не допустить роста объёма кода, многие компиляторы хранят литеральную строку в метаданных модуля только в одном экземпляре.
2) Ленивые присваивания строк
Обычно строка — это объект большого размера, требующий для своей работы выделения большого блока памяти. Данная оптимизация выделяет память под строки только при надобности, позволяя нескольким переменным указывать на одну цепочку символов. Только если одна из переменных меняет своё содержимое, строка копируется.
7 922
Программная конвейеризация
Программная конвейеризация - это методология разработки программного обеспечения, при которой процесс создания программного продукта разбивается на этапы, которые выполняются последовательно и независимо друг от друга. Каждый этап представляет собой отдельное звено конвейера и обрабатывает определенный набор функциональных задач, необходимых для создания конечного продукта.
Процесс выполняется следующим образом:
- на первом этапе код программы проходит через анализатор кода для обнаружения ошибок исходного кода,
- на втором этапе код проверяется на соответствие правилам проекта,
- на третьем этапе производится сборка и создание исполняемого файла,
- на последнем этапе программа тестируется на соответствие требованиям проекта.
Программная конвейеризация позволяет повысить качество создаваемого продукта и скорость его создания, а также повысить эффективность работы команды программистов. Кроме того, это позволяет ускорить процесс поиска и устранения ошибок и облегчить процесс масштабирования проекта.
7 922
Аппаратные решения, упрощающие программную конвейеризацию
- «Вращающиеся регистры» — часть регистрового файла отводится на область вращающихся регистров. Инструкции, использующие некоторый архитектурный регистр из этой области будут обращаться к различным физическим регистрам по мере исполнения итераций. Через какое-то количество итераций, вновь произойдет обращение к исходному физическому регистру. Это позволяет работать с различными итерациями цикла одновременно, и при этом не требуется явных пересылок между регистрами.
- Предикаты и предикатное исполнение инструкций, при котором предикатом работает некоторые специальные предикаты цикла. Эти предикаты позволяют включать и отключать некоторые инструкции цикла в процессе прохождения итераций, тем самым реализуя пролог и эпилог цикла в основном его коде, а также упрощают раскрытие условных операций в теле цикла.
- Аппаратная поддержка циклов, при которой программа дает процессору информацию о размере цикла и о параметрах индексной переменной. Это позволяет сократить накладные расходы на организацию цикла. Также позволяет настроить скорость вращения и размер группы вращающихся регистров.
7 922
Способы реализации предвыборки кода
1. Предварительная загрузка:
Способ заключается в том, чтобы загружать в память блоки данных, которые скорее всего понадобятся в будущем. Загруженные данные могут сохраняться в кеше для более быстрого доступа к ним в дальнейшем.
2. Спекулятивная предвыборка:
Способ заключается в том, чтобы загрузить в память данных, которые могут понадобиться в будущем на основе анализа предыдущих запросов. Например, если вы последовательно обрабатываете элементы списка, то спекулятивная предвыборка может загрузить следующий элемент в память на всякий случай.
3. Предвыборка на основе предсказаний:
Способ заключается в том, чтобы загрузить данные, которые вероятно будут запрошены в будущем, на основе анализа предыдущих запросов и статистических данных.
4. Асинхронная предвыборка:
Способ заключается в том, чтобы асинхронно загружать данные в фоновом режиме.
7 922
Предвыборка кода
В компьютерной архитектуре, предвыборкой кода называют технологию, используемую в микропроцессоре для увеличения скорости исполнения программ, уменьшая время, в течение которого процессор находится в состоянии ожидания из-за отсутствия инструкций для исполнения.
Современные микропроцессоры гораздо быстрее чем память, вследствие чего инструкции исполняемой программы не могут считывыться достаточно быстро, чтобы обеспечить непрерывность работы процессора. Добавление кэша может обеспечить более быстрый доступ к необходимым инструкциям.
Другими словами, предвыборка кода - выдача запросов со стороны процессора в оперативную память для считывания инструкций заблаговременно, до того момента, как эти инструкции потребуется исполнять. В результате этих запросов, инструкции загружаются из памяти в кэш. Когда инструкции потребуется исполнять, доступ к ним будет осуществляться значительно быстрее, так как задержка при обращении в кэш на порядки меньше, чем при обращении в оперативную память.
7 922
Векторизация (параллельные вычисления)
Векторизация — вид распараллеливания программы, при котором однопоточные приложения, выполняющие одну операцию в каждый момент времени, модифицируются для выполнения нескольких однотипных операций одновременно.
Скалярные операции, обрабатывающие по паре операндов заменяются на операции над массивами, обрабатывающие несколько элементов вектора в каждый момент времени.
Пример:
Показан фрагмент программы, который поэлементно перемножает два массива, состоящие из чисел:
for (i = 0; i < 1024; i++)
C[i] = A[i] * B[i];
Данный цикл может быть векторизован так:
for (i = 0; i < 1024; i+=4)
C[i:i+3] = A[i:i+3] * B[i:i+3];
Запись C[i:i+3] означает вектор из 4 элементов — от C[i] до C[i+3] включительно, а под * понимается операция поэлементного умножения векторов.7 922
Перестановка циклов
Перестановка циклов — оптимизация компилятора при которой меняется порядок итерационных переменных, относящихся к группе вложенных циклов. Итерационная переменная, используемая во внутреннем цикле, перемещается во внешний цикл, и наоборот. Это часто делается, чтобы гарантировать, что элементы многомерных массивов доступны в том порядке, в котором они хранятся в памяти, т.е. для улучшения локальности ссылок.
Например, следующий код:
for (int j=0; j<10; j++) {
for (int i=0; i<20; i++)
y[i][j] = i + j;
}
в результате применения оптимизации может быть преобразован в:
for (int i=0; i<20; i++) {
for (int j=0; j<10; j++)
y[i][j] = i + j;
}
В отдельных случаях, такое преобразование может создать контекст для дальнейших оптимизаций.
现已上线!2025 年 Telegram 研究 — 年度关键洞察 
