Индексы в базах данных, таких как SQL Server, MySQL или PostgreSQL, существенно улучшают производительность запросов, особенно для операций поиска, сортировки и фильтрации. Однако у индексов есть и минусы, которые могут негативно сказаться на производительности и других аспектах работы базы данных.
Индексы замедляют операции вставки (
INSERT), обновления (UPDATE) и удаления (DELETE), так как при каждом изменении данных необходимо также обновлять индексы.Вставка: При вставке новой записи нужно обновить все соответствующие индексы.
Обновление: При обновлении записи могут изменяться индексируемые колонки, что требует обновления индексов.
Удаление: При удалении записи нужно удалить соответствующие записи из индексов.
Индексы занимают дополнительное пространство на диске и в оперативной памяти. Чем больше индексов на таблице, тем больше требуется места для их хранения.
Дисковое пространство: Каждому индексу требуется место на диске для хранения его данных.
Память: Индексы занимают память при их использовании, особенно в случае часто запрашиваемых индексов, которые кэшируются в оперативной памяти.
При массовой загрузке данных, например, при использовании операций
LOAD DATA или BULK INSERT, наличие индексов замедляет процесс, так как индексы должны обновляться по мере добавления каждой записи.Индексы могут фрагментироваться, особенно если в таблице часто выполняются операции вставки, обновления и удаления. Фрагментация индексов приводит к ухудшению производительности запросов.
Фрагментация: При частых изменениях данных индексы могут становиться фрагментированными, что увеличивает время доступа к данным.
Реорганизация: Периодически индексы нужно реорганизовывать или перестраивать, что требует дополнительных ресурсов и времени.
Управление индексами требует дополнительного администрирования и мониторинга. Нужно следить за эффективностью индексов, удалять неиспользуемые индексы и создавать новые по мере изменения запросов и структуры данных.
Наличие нескольких индексов на одной таблице может привести к конфликтам при планировании запросов. Оптимизатор запросов может выбирать менее эффективные индексы, что ухудшает производительность.
Плохо спроектированные индексы могут негативно повлиять на производительность запросов. Например, индексы на часто изменяемых колонках или слишком большое количество индексов могут привести к значительным издержкам при обновлении данных.
CREATE TABLE Employees (
EmployeeID INT PRIMARY KEY,
FirstName VARCHAR(50),
LastName VARCHAR(50),
DepartmentID INT,
Salary DECIMAL(10, 2)
);
CREATE INDEX idx_firstname ON Employees(FirstName);
CREATE INDEX idx_lastname ON Employees(LastName);
CREATE INDEX idx_department ON Employees(DepartmentID);
INSERT INTO Employees (EmployeeID, FirstName, LastName, DepartmentID, Salary)
VALUES (1, 'John', 'Doe', 10, 60000.00);
UPDATE Employees
SET Salary = Salary * 1.05
WHERE DepartmentID = 10;
Ставь 👍 и забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
👍2
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
👍2
Это структура данных, используемая в системах управления базами данных (СУБД) для организации и ускорения доступа к данным. B-tree индекс является сбалансированным деревом, обеспечивающим эффективное выполнение операций поиска, вставки, удаления и диапазонного поиска. B-tree индекс используется большинством реляционных СУБД, таких как SQL Server, MySQL, PostgreSQL и Oracle.
B-tree индекс является сбалансированным деревом, где все листья находятся на одном уровне. Это обеспечивает равномерное время доступа к данным.
В узлах B-tree хранятся ключи, которые могут ссылаться на строки в таблице или на другие узлы дерева.
Ключи в каждом узле упорядочены, что позволяет эффективно выполнять бинарный поиск внутри узла.
B-tree индекс эффективно поддерживает диапазонные запросы (например, поиск всех записей с ключами между заданными значениями).
B-tree автоматически сбалансирован, что позволяет эффективно выполнять операции вставки, удаления и обновления.
CREATE TABLE Employees (
EmployeeID INT PRIMARY KEY,
FirstName VARCHAR(50),
LastName VARCHAR(50),
DepartmentID INT,
Salary DECIMAL(10, 2)
);
CREATE INDEX idx_lastname ON Employees(LastName);
Операция поиска в B-tree выполняется за логарифмическое время O(log n), где n — количество узлов.
SELECT * FROM Employees WHERE LastName = 'Smith';
Диапазонные запросы, такие как поиск всех сотрудников с фамилией от 'A' до 'M', выполняются эффективно.
SELECT * FROM Employees WHERE LastName BETWEEN 'A' AND 'M';
При вставке новой записи в таблицу с индексом B-tree, запись добавляется в соответствующее место, поддерживая балансировку дерева.
INSERT INTO Employees (EmployeeID, FirstName, LastName, DepartmentID, Salary)
VALUES (1, 'John', 'Doe', 10, 60000.00);
При удалении записи соответствующий ключ удаляется из B-tree, и дерево автоматически перестраивается, чтобы сохранить балансировку.
DELETE FROM Employees WHERE EmployeeID = 1;
Обеспечивает быстрый доступ к данным благодаря сбалансированной структуре дерева.
Поддерживает операции вставки, удаления и поиска с логарифмической сложностью.
Эффективно обрабатывает диапазонные запросы благодаря упорядоченной структуре.
Динамическая балансировка дерева обеспечивает равномерное время доступа и вставки/удаления.
Требует дополнительного пространства для хранения структуры дерева и ключей.
Вставка и удаление могут требовать перестроения узлов, что влечет за собой дополнительные вычислительные затраты.
При частых операциях вставки и удаления может возникнуть фрагментация, что может потребовать периодического обслуживания (например, реорганизации индекса).
Ставь 👍 и забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
❤1
Anonymous Quiz
15%
Array()
5%
CreateArray()
76%
new Array()
4%
List()
🤯21👍1
В SQL существует несколько типов соединений (joins), которые используются для объединения строк из двух или более таблиц на основе логических отношений между ними.
INNER JOIN возвращает только те строки, которые имеют совпадающие значения в обеих таблицах. Это наиболее часто используемый тип соединения.
SELECT employees.name, departments.name
FROM employees
INNER JOIN departments
ON employees.department_id = departments.id;
LEFT JOIN возвращает все строки из левой таблицы и совпадающие строки из правой таблицы. Если совпадений нет, результат содержит
NULL для столбцов из правой таблицы.SELECT employees.name, departments.name
FROM employees
LEFT JOIN departments
ON employees.department_id = departments.id;
RIGHT JOIN возвращает все строки из правой таблицы и совпадающие строки из левой таблицы. Если совпадений нет, результат содержит
NULL для столбцов из левой таблицы.SELECT employees.name, departments.name
FROM employees
RIGHT JOIN departments
ON employees.department_id = departments.id;
FULL JOIN возвращает все строки, когда есть совпадение в левой или правой таблице. Если совпадений нет, результат содержит
NULL для столбцов из таблицы, где нет совпадений.SELECT employees.name, departments.name
FROM employees
FULL JOIN departments
ON employees.department_id = departments.id;
CROSS JOIN возвращает декартово произведение двух таблиц, то есть каждая строка из первой таблицы соединяется с каждой строкой из второй таблицы.
SELECT employees.name, departments.name
FROM employees
CROSS JOIN departments;
SELF JOIN используется для соединения таблицы с самой собой. Это полезно для работы с иерархическими данными или когда требуется сравнить строки в одной таблице.
SELECT e1.name AS Employee1, e2.name AS Employee2
FROM employees e1
INNER JOIN employees e2
ON e1.manager_id = e2.id;
employees
idnamedepartment_idmanager_iddepartments
idnameВернет только тех сотрудников, которые имеют департамент.
SELECT employees.name, departments.name AS department
FROM employees
INNER JOIN departments
ON employees.department_id = departments.id;
Вернет всех сотрудников, включая тех, у которых нет департамента.
SELECT employees.name, departments.name AS department
FROM employees
LEFT JOIN departments
ON employees.department_id = departments.id;
Вернет все департаменты, включая те, в которых нет сотрудников.
SELECT employees.name, departments.name AS department
FROM employees
RIGHT JOIN departments
ON employees.department_id = departments.id;
Вернет всех сотрудников и все департаменты, включая те, у которых нет совпадений.
SELECT employees.name, departments.name AS department
FROM employees
FULL JOIN departments
ON employees.department_id = departments.id;
Вернет каждую комбинацию сотрудников и департаментов.
SELECT employees.name, departments.name AS department
FROM employees
CROSS JOIN departments;
Вернет всех сотрудников и их менеджеров.
SELECT e1.name AS Employee, e2.name AS Manager
FROM employees e1
INNER JOIN employees e2
ON e1.manager_id = e2.id;
Ставь 👍 и забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
👍7
1. INNER JOIN: возвращает строки, где есть совпадения в обеих таблицах.
2. LEFT JOIN: возвращает все строки из левой таблицы и совпадающие строки из правой, остальные заполняются NULL.
3. RIGHT JOIN: аналогично LEFT JOIN, но с правой таблицей.
4. FULL OUTER JOIN: объединяет все строки из обеих таблиц, заполняя несоответствия NULL.
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
👍1
Соединения (
JOIN) в SQL позволяют объединять строки из двух или более таблиц на основе логических отношений между ними. Существует несколько видов соединений, которые можно использовать для разных типов логических связей между таблицами. Возвращает только те строки, которые имеют совпадающие значения в обеих таблицах. Это наиболее распространенный тип соединения.
SELECT employees.name, departments.name AS department
FROM employees
INNER JOIN departments
ON employees.department_id = departments.id;
LEFT JOIN возвращает все строки из левой таблицы и совпадающие строки из правой таблицы. Если совпадающих строк нет, то правые столбцы будут заполнены
NULL.SELECT employees.name, departments.name AS department
FROM employees
LEFT JOIN departments
ON employees.department_id = departments.id;
RIGHT JOIN возвращает все строки из правой таблицы и совпадающие строки из левой таблицы. Если совпадающих строк нет, то левые столбцы будут заполнены NULL.
SELECT employees.name, departments.name AS department
FROM employees
RIGHT JOIN departments
ON employees.department_id = departments.id;
FULL JOIN возвращает все строки, когда есть совпадение в левой или правой таблице. Если совпадающих строк нет, то в соответствующих столбцах будут NULL.
SELECT employees.name, departments.name AS department
FROM employees
FULL JOIN departments
ON employees.department_id = departments.id;
CROSS JOIN возвращает декартово произведение двух таблиц, то есть каждая строка из первой таблицы соединяется с каждой строкой из второй таблицы.
SELECT employees.name, departments.name AS department
FROM employees
CROSS JOIN departments;
SELF JOIN используется для соединения таблицы с самой собой. Это полезно для работы с иерархическими данными или для сравнения строк в одной таблице.
SELECT e1.name AS Employee, e2.name AS Manager
FROM employees e1
INNER JOIN employees e2
ON e1.manager_id = e2.id;
NATURAL JOIN автоматически соединяет таблицы по всем столбцам с одинаковыми именами и совместимыми типами данных. Обычно используется редко из-за потенциальных проблем с неявным сопоставлением столбцов.
SELECT employees.name, departments.name AS department
FROM employees
NATURAL JOIN departments;
Пример: Один департамент имеет много сотрудников. Соединение:
INNER JOIN, LEFT JOIN, RIGHT JOIN.Пример: Студенты записаны на несколько курсов, а каждый курс может включать многих студентов. Соединение: Потребуется промежуточная таблица для реализации соединения.
Пример: Каждому сотруднику соответствует одна учетная запись в системе. Соединение:
INNER JOIN, LEFT JOIN.Ставь 👍 и забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
👍4
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
👍9
В реляционных базах данных связь "один к одному" (one-to-one) подразумевает, что каждая запись в одной таблице соответствует ровно одной записи в другой таблице. Для реализации связи "один к одному" в SQL можно использовать несколько подходов, в зависимости от требований и архитектуры базы данных.
Используйте внешний ключ в одной таблице, который ссылается на первичный ключ другой таблицы, и сделайте этот внешний ключ уникальным.
Используйте один и тот же первичный ключ в обеих таблицах, где одна таблица содержит внешний ключ, который является также первичным ключом.
Таблицы
Users (пользователи)
UserId (Primary Key)
UserName
Profiles (профили)
ProfileId (Primary Key)
UserId (Foreign Key, Unique)
ProfileData
Создание таблиц
CREATE TABLE Users (
UserId INT PRIMARY KEY,
UserName VARCHAR(100)
);
CREATE TABLE Profiles (
ProfileId INT PRIMARY KEY,
UserId INT UNIQUE,
ProfileData VARCHAR(255),
FOREIGN KEY (UserId) REFERENCES Users(UserId)
);
Вставка данных
INSERT INTO Users (UserId, UserName) VALUES (1, 'John Doe');
INSERT INTO Profiles (ProfileId, UserId, ProfileData) VALUES (1, 1, 'Profile data for John Doe');
Запрос данных
SELECT Users.UserName, Profiles.ProfileData
FROM Users
JOIN Profiles ON Users.UserId = Profiles.UserId;
Таблицы
Users (пользователи)
UserId (Primary Key)
UserName
Profiles (профили)
UserId (Primary Key, Foreign Key)
ProfileData
Создание таблиц
CREATE TABLE Users (
UserId INT PRIMARY KEY,
UserName VARCHAR(100)
);
CREATE TABLE Profiles (
UserId INT PRIMARY KEY,
ProfileData VARCHAR(255),
FOREIGN KEY (UserId) REFERENCES Users(UserId)
);
Вставка данных
INSERT INTO Users (UserId, UserName) VALUES (1, 'John Doe');
INSERT INTO Profiles (UserId, ProfileData) VALUES (1, 'Profile data for John Doe');
Запрос данных
SELECT Users.UserName, Profiles.ProfileData
FROM Users
JOIN Profiles ON Users.UserId = Profiles.UserId;
Ставь 👍 и забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
👍1
out также передаёт аргумент по ссылке, но переменная может быть не инициализирована, так как метод обязан присвоить ей значение.
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
👍7🔥2
Для реализации отношения "один ко многим" (one-to-many) в C# с использованием Entity Framework Core, нужно определить две модели, где одна модель будет содержать коллекцию объектов другой модели.
Предположим, у нас есть две модели:
Author и Book. Один автор может написать много книг, поэтому у нас будет отношение "один ко многим". public class Author
{
public int AuthorId { get; set; }
public string Name { get; set; }
// Связь один ко многим
public ICollection<Book> Books { get; set; }
}
public class Book
{
public int BookId { get; set; }
public string Title { get; set; }
public int AuthorId { get; set; }
// Навигационное свойство
public Author Author { get; set; }
}
Создаем контекст данных, который наследуется от
DbContext. using Microsoft.EntityFrameworkCore;
public class AppDbContext : DbContext
{
public DbSet<Author> Authors { get; set; }
public DbSet<Book> Books { get; set; }
protected override void OnConfiguring(DbContextOptionsBuilder optionsBuilder)
{
optionsBuilder.UseSqlServer("YourConnectionStringHere");
}
protected override void OnModelCreating(ModelBuilder modelBuilder)
{
// Настройка связи один ко многим
modelBuilder.Entity<Author>()
.HasMany(a => a.Books)
.WithOne(b => b.Author)
.HasForeignKey(b => b.AuthorId);
}
}
Для создания и применения миграций выполните следующие команды в консоли диспетчера пакетов или CLI.
Использование Package Manager Console (PMC)
Add-Migration InitialCreate
Update-Database
Использование .NET CLI
dotnet ef migrations add InitialCreate
dotnet ef database update
Теперь можно добавить данные в базу данных и протестировать отношение "один ко многим".
public class Program
{
public static void Main()
{
using (var context = new AppDbContext())
{
var author = new Author
{
Name = "George Orwell",
Books = new List<Book>
{
new Book { Title = "1984" },
new Book { Title = "Animal Farm" }
}
};
context.Authors.Add(author);
context.SaveChanges();
}
}
}
Теперь можно извлечь данные и проверить отношение.
using (var context = new AppDbContext())
{
var authors = context.Authors.Include(a => a.Books).ToList();
foreach (var author in authors)
{
Console.WriteLine($"Author: {author.Name}");
foreach (var book in author.Books)
{
Console.WriteLine($" Book: {book.Title}");
}
}
}
Ставь 👍 и забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
👍4
readonly — это поле, значение которого можно установить только при инициализации или в конструкторе, но оно может быть вычислено во время выполнения.
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
👍11🔥6
Для реализации связи "многие ко многим" (many-to-many) в реляционных базах данных используется промежуточная (связующая) таблица, которая содержит внешние ключи, ссылающиеся на обе связанные таблицы. Эта промежуточная таблица позволяет управлять отношениями между записями двух таблиц.
Предположим, у нас есть две таблицы:
Students (студенты) и Courses (курсы). Один студент может быть записан на несколько курсов, и один курс может включать нескольких студентов.Таблица
Students для хранения данных о студентах. Таблица Courses для хранения данных о курсах.Таблица
StudentCourses, которая содержит внешние ключи, ссылающиеся на Students и Courses.StudentsCREATE TABLE Students (
StudentId INT PRIMARY KEY,
StudentName VARCHAR(100)
);
CoursesCREATE TABLE Courses (
CourseId INT PRIMARY KEY,
CourseName VARCHAR(100)
);
StudentCoursesCREATE TABLE StudentCourses (
StudentId INT,
CourseId INT,
PRIMARY KEY (StudentId, CourseId),
FOREIGN KEY (StudentId) REFERENCES Students(StudentId),
FOREIGN KEY (CourseId) REFERENCES Courses(CourseId)
);
Вставка данных в таблицу
Students: INSERT INTO Students (StudentId, StudentName) VALUES (1, 'John Doe');
INSERT INTO Students (StudentId, StudentName) VALUES (2, 'Jane Smith');
Вставка данных в таблицу
CoursesINSERT INTO Courses (CourseId, CourseName) VALUES (1, 'Mathematics');
INSERT INTO Courses (CourseId, CourseName) VALUES (2, 'Science');
Вставка данных в таблицу
StudentCourses (связывающая таблица): INSERT INTO StudentCourses (StudentId, CourseId) VALUES (1, 1); -- John Doe записан на Mathematics
INSERT INTO StudentCourses (StudentId, CourseId) VALUES (1, 2); -- John Doe записан на Science
INSERT INTO StudentCourses (StudentId, CourseId) VALUES (2, 1); -- Jane Smith записана на Mathematics
Ставь 👍 и забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
🤔1
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
👍3🔥1
Связь "один к одному" (one-to-one) и связь "многие ко многим" (many-to-many) представляют разные типы логических отношений между записями в реляционных базах данных. Основные отличия между этими двумя типами связей заключаются в их природе, реализации и сценариях использования.
Каждая запись в одной таблице соответствует ровно одной записи в другой таблице. Используется для разделения информации на связанные, но отдельные части, например, для повышения безопасности или упрощения структуры данных. Можно использовать уникальный внешний ключ или один и тот же первичный ключ в обеих таблицах.
Пользователи и их профили.
Клиенты и их учетные данные.
CREATE TABLE Users (
UserId INT PRIMARY KEY,
UserName VARCHAR(100)
);
CREATE TABLE Profiles (
ProfileId INT PRIMARY KEY,
UserId INT UNIQUE,
ProfileData VARCHAR(255),
FOREIGN KEY (UserId) REFERENCES Users(UserId)
);
Каждая запись в одной таблице может соответствовать множеству записей в другой таблице, и наоборот. Используется для представления сложных взаимосвязей, таких как студенты и курсы, где каждый студент может записаться на множество курсов, и каждый курс может включать множество студентов. Требуется промежуточная (связующая) таблица, содержащая два внешних ключа, ссылающихся на связанные таблицы. Студенты и курсы. Авторы и книги. Таблицы
Students и Courses с промежуточной таблицей StudentCourses. CREATE TABLE Students (
StudentId INT PRIMARY KEY,
StudentName VARCHAR(100)
);
CREATE TABLE Courses (
CourseId INT PRIMARY KEY,
CourseName VARCHAR(100)
);
CREATE TABLE StudentCourses (
StudentId INT,
CourseId INT,
PRIMARY KEY (StudentId, CourseId),
FOREIGN KEY (StudentId) REFERENCES Students(StudentId),
FOREIGN KEY (CourseId) REFERENCES Courses(CourseId)
);
Каждая запись в одной таблице соответствует ровно одной записи в другой таблице.
Каждая запись в одной таблице может соответствовать множеству записей в другой таблице и наоборот.
Используется уникальный внешний ключ или одинаковый первичный ключ в обеих таблицах.
Требуется промежуточная таблица для хранения связей между записями.
Пользователь и его профиль, клиент и его учетные данные.
Студенты и курсы, авторы и книги.
Получение данных пользователя и его профиля
SELECT Users.UserName, Profiles.ProfileData
FROM Users
JOIN Profiles ON Users.UserId = Profiles.UserId;
Получение всех курсов, на которые записан студент
SELECT Students.StudentName, Courses.CourseName
FROM Students
JOIN StudentCourses ON Students.StudentId = StudentCourses.StudentId
JOIN Courses ON StudentCourses.CourseId = Courses.CourseId;
Получение всех студентов, записанных на определенный курс
SELECT Courses.CourseName, Students.StudentName
FROM Courses
JOIN StudentCourses ON Courses.CourseId = StudentCourses.CourseId
JOIN Students ON StudentCourses.StudentId = Students.StudentId;
Ставь 👍 и забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
👍2
1. Как работает: GC отслеживает объекты, на которые есть активные ссылки, и удаляет те, которые недоступны.
2. Цель: предотвращение утечек памяти и избавление разработчика от необходимости вручную освобождать память.
3. GC работает асинхронно и может вызывать паузы в работе приложения, известные как "stop-the-world".
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
❤2🔥2
Наследование в объектно-ориентированном программировании, является мощным инструментом для создания иерархий классов и повторного использования кода. Однако его использование должно быть осознанным и оправданным.
Наследование используется для создания обобщенных базовых классов и специализированных производных классов.
public class Animal
{
public void Eat() { /*...*/ }
}
public class Dog : Animal
{
public void Bark() { /*...*/ }
}
Наследование позволяет повторно использовать код базового класса в производных классах, избегая дублирования.
public class Vehicle
{
public int Speed { get; set; }
public void Move() { /*...*/ }
}
public class Car : Vehicle
{
public void Honk() { /*...*/ }
}
Наследование позволяет использовать объекты производных классов через ссылки на базовые классы, обеспечивая гибкость и расширяемость кода.
public class Shape
{
public virtual void Draw() { /*...*/ }
}
public class Circle : Shape
{
public override void Draw() { /*...*/ }
}
public void DrawShape(Shape shape)
{
shape.Draw();
}
Слишком глубокие иерархии усложняют понимание и поддержку кода. В таких случаях рассмотрите использование композиции.
Интерфейсы и абстрактные классы могут обеспечить большую гибкость и возможность для реализации различных вариантов поведения.
public interface IFlyable
{
void Fly();
}
public class Bird : IFlyable
{
public void Fly() { /*...*/ }
}
Композиция может быть предпочтительнее наследования, так как позволяет более гибко комбинировать функциональности.
public class Engine
{
public void Start() { /*...*/ }
}
public class Car
{
private Engine _engine = new Engine();
public void Start() { _engine.Start(); }
}
На практике наследование может использоваться реже, чем композиция, особенно в современных подходах к проектированию, таких как микросервисы или системы, ориентированные на интерфейсы. Тем не менее, оно остается важным инструментом, особенно когда необходимо обеспечить полиморфизм или когда существуют естественные иерархии объектов.
Наследование широко используется в библиотеках и фреймворках для предоставления базовых классов, от которых пользователи могут наследовать свои классы.
public abstract class ControllerBase
{
public abstract void Execute();
}
public class HomeController : ControllerBase
{
public override void Execute() { /*...*/ }
}
Наследование полезно при моделировании реальных объектов и их иерархий.
public class Employee
{
public string Name { get; set; }
public decimal Salary { get; set; }
}
public class Manager : Employee
{
public int NumberOfSubordinates { get; set; }
}
Ставь 👍 и забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
❤3👍1👾1
1. ref: позволяет передавать ссылочный тип по ссылке, чтобы изменения в методе влияли на оригинальный объект.
o Это нужно, когда требуется изменить ссылку (например, перенаправить на новый объект) внутри метода.
2. out: используется для передачи данных из метода наружу, не требуя их предварительной инициализации.
o Это упрощает возврат нескольких значений из метода через параметры, особенно для ссылочных типов.
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
👍2🔥1
Это два различных механизма в C# для изменения поведения методов, свойств или событий, унаследованных от базового класса.
Переопределение используется для предоставления новой реализации метода, свойства или события в производном классе, который уже был объявлен в базовом классе как
virtual, abstract или override.Ключевое слово
override.Требование
Метод базового класса должен быть объявлен с ключевым словом
virtual или abstract.public class BaseClass
{
public virtual void Display()
{
Console.WriteLine("BaseClass Display");
}
}
public class DerivedClass : BaseClass
{
public override void Display()
{
Console.WriteLine("DerivedClass Display");
}
}
Использование
BaseClass obj = new DerivedClass();
obj.Display(); // Выведет: "DerivedClass Display"
Сокрытие используется для создания новой реализации метода, свойства или события в производном классе, который уже существует в базовом классе, но без использования
virtual и override. При сокрытии используется ключевое слово new.Ключевое слово
new.Требование
Метод базового класса не обязательно должен быть объявлен с ключевым словом
virtual.public class BaseClass
{
public void Display()
{
Console.WriteLine("BaseClass Display");
}
}
public class DerivedClass : BaseClass
{
public new void Display()
{
Console.WriteLine("DerivedClass Display");
}
}
Использование
BaseClass obj1 = new BaseClass();
obj1.Display(); // Выведет: "BaseClass Display"
DerivedClass obj2 = new DerivedClass();
obj2.Display(); // Выведет: "DerivedClass Display"
BaseClass obj3 = new DerivedClass();
obj3.Display(); // Выведет: "BaseClass Display"
Переопределение: Базовый метод должен быть объявлен как virtual, abstract или override. Сокрытие: Базовый метод не обязательно должен быть виртуальным.
Переопределение: Используется ключевое слово override. Сокрытие: Используется ключевое слово new.
Переопределение: Изменяет указатель в виртуальной таблице базового класса, что позволяет полиморфизм. Сокрытие: Создает новый метод, который скрывает метод базового класса, но не изменяет виртуальную таблицу.
Переопределение: Поддерживает полиморфизм. Вызовы методов производятся на основе реального типа объекта. Сокрытие: Не поддерживает полиморфизм. Вызовы методов зависят от типа ссылки, а не от реального типа объекта.
Ставь 👍 и забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
👍8
2. Ссылочные: классы, интерфейсы, массивы, строки.
3. Обобщённые (Generic): позволяют работать с типами данных, указанными при создании экземпляра (например, List<T>).
4. Пользовательские: структуры, перечисления (enum), типы, созданные программистом.
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
👍6👾1
Блок
using используется для управления временем жизни объектов, которые потребляют неуправляемые ресурсы, такие как файлы, соединения с базой данных или сетевые соединения. Основное преимущество использования блока using заключается в автоматическом освобождении этих ресурсов, что помогает предотвратить утечки ресурсов и улучшить управление памятью.Блок
using гарантирует, что метод Dispose() будет вызван автоматически, когда выполнение кода выйдет из блока using, даже если возникнет исключение. Это освобождает программиста от необходимости вручную вызывать Dispose() и уменьшает вероятность ошибок. using (StreamReader reader = new StreamReader("file.txt"))
{
string content = reader.ReadToEnd();
Console.WriteLine(content);
}
// StreamReader автоматически закрывается и освобождает ресурсы после выхода из блока using. Использование блока
using уменьшает количество необходимого кода для обеспечения правильного освобождения ресурсов. using (SqlConnection connection = new SqlConnection(connectionString))
{
connection.Open();
// Работа с базой данных
}
// SqlConnection автоматически закрывается и освобождает ресурсы после выхода из блока using.
Блок
using повышает надежность кода, так как гарантирует, что ресурсы будут освобождены даже в случае возникновения исключений. try
{
using (StreamWriter writer = new StreamWriter("file.txt"))
{
writer.WriteLine("Hello, World!");
}
}
catch (Exception ex)
{
Console.WriteLine($"Произошла ошибка: {ex.Message}");
}
// StreamWriter автоматически закрывается и освобождает ресурсы после выхода из блока using, даже если произошла ошибка.
Код, использующий блок
using, выглядит более чистым и упрощает понимание и сопровождение. using (MemoryStream memoryStream = new MemoryStream())
{
// Работа с MemoryStream
}
// MemoryStream автоматически освобождается после выхода из блока using.
Работа с файлами
using System;
using System.IO;
class Program
{
static void Main()
{
using (StreamReader reader = new StreamReader("example.txt"))
{
string content = reader.ReadToEnd();
Console.WriteLine(content);
}
// StreamReader автоматически закрывается и освобождает ресурсы после выхода из блока using.
}
}
Ставь 👍 и забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
👍9