StepOne | Степан Минин – Telegram
StepOne | Степан Минин
3.42K subscribers
247 photos
35 videos
6 files
312 links
StepOne by Степан Минин @ststphn

Твой первый шаг к успеху в программировании

Закрытый тг канал https://news.1rj.ru/str/tribute/app?startapp=slOA

По вопросам рекламы @Spiral_Yuri

Ютуб https://www.youtube.com/@steponeit
Download Telegram
StepOne | Степан Минин
Примитивы синхронизации доступно 🙏 1️⃣ Monitor Простейший в использовании примитив синхронизации в .NET, который используется для обеспечения взаимоисключающего доступа к распределённому ресурсу путём получения или высвобождения блокировки на специальном…
Продолжение по просьбе подписчика 🔥

6️⃣ ReaderWriterLock

С помощью этого класса в любой момент времени можно получить или конкурентный доступ на чтение распределённого ресурса для множества потоков, или экслюзивный доступ на запись в него для одного, поэтому, в случае, когда ресурс нечасто меняется, будет обеспечена лучшая пропускная способность, чем с Монитором.

У читателей и писателей разные очереди. В момент времени, когда поток высвобождает писательскую блокировку, все потоки, ожидавшие блокировку на чтение, получают её. И, наоборот, когда все эти читательские блокировки отпускаются, поток в читательской очереди обрабатывается. Получается, что примитив чередует обработку коллекции потоков читателей и одного потока писателя.

7️⃣ ReaderWriterLockSlim

Этот примитив синхронизации очень похож на предыдущий, и рекомендуется к использованию в версиях .NET новее .NET Framework 3.5

Класс доработали с разных сторон: упрощены правила рекурсии и изменения состояния блокировки, исправлены многие случаи потенциального дедлока, а также общая производительность выше, чем у предшественника.

А вообще, помните друзья: лучший примитив синхронизации тот, который вам не понадобится, другими словами, не делите данные между потоками 😁
Please open Telegram to view this post
VIEW IN TELEGRAM
🔥21👍113❤‍🔥1
StepOne | Степан Минин
Всё, как я и ожидал Обобщённая математика слишком "непонятный фрукт" для рядового C# разработчика - в проде это не используют, и при этом не планируют из-за отсутствия понимания инструмента. Благо, вы подписаны на этот канал, и в этом месяце будет анонс…
Как в C# 11 появилась обобщённая математика

7 ноября при поддержке CUSTIS состоится MskDotNet Meetup #55, где я выступлю с докладом.

В рамках доклада рассмотрим с нуля концепцию обобщённой математики, как она выглядит в C# 11, как могла выглядеть в предыдущих версиях языка и почему вообще появилась.

Также зароемся в кишки System.Numerics, и узнаем, в какой продакшн в скором времени заделиверится эта фича.

Чтобы посетить митап оффлайн зарегистрируйтесь по ссылке, там вся необходимая информация.

Для онлайн слушателей будет организована трансляция.
Please open Telegram to view this post
VIEW IN TELEGRAM
🔥1254👍2
Как разобраться в новом проекте: пошаговая инструкция 🤩

Каждый разработчик при выходе на новую работу, скачав код из корпоративного репозитория, сталкивался с мыслью о том, что он не понимает написанного на экране.

Дело тут не в компетенции, а в исторически сложившихся особенностях системы, которые отличаются от компании к компании.

При использовании следующих рекомендаций появится шанс приносить ценность ещё до окончания испытательного срока.

1️⃣ Общайтесь с аналитиками, расспрашивайте их про проект, попросите показать документацию и так далее.

Именно эти люди превращают непонятные бизнес-требования в структурированные технические задачи.

2️⃣ Если нет аналитиков, документации или что-то из этого в плохом состоянии, то смело бросайтесь выполнять много разноплановых мелких задач.

В большинстве проектов есть задачи техдолга, покрывающие разные участки системы: работа с ними позволит сформировать поверхностное представление, которое со временем станет проще углублять.

3️⃣ О качественной аналитике на проекте никогда не слышали, задач для погружения не дают - тогда бегите писать тесты!

Во-первых, тесты нужны всем. Во-вторых, их написание вырабатывает понимание того, как выполняется конкретный код, а, следовательно, какую задачу он решает.
Please open Telegram to view this post
VIEW IN TELEGRAM
🔥16👍83🤯3
Baza 🥶 ☠️ 🥵
😁11💯2🤩1
StepOne | Степан Минин
Как в C# 11 появилась обобщённая математика 7 ноября при поддержке CUSTIS состоится MskDotNet Meetup #55, где я выступлю с докладом. В рамках доклада рассмотрим с нуля концепцию обобщённой математики, как она выглядит в C# 11, как могла выглядеть в предыдущих…
unmanaged constraint

Позавчера на докладе меня попросили прокомментировать этот фрагмент кода из ML NET.

В частности слушателя заинтересовало, что означает ограничение unmanaged.

Тогда ответить не вышло, честно не знал 😅

Однако я не я, если не разберусь в теме, и не напишу об этом пост 😁

Запоминаем - where T : unmanaged означает, что тип T это что-то из списка ниже:

⭐️ sbyte, byte, short, ushort, int, uint, long, ulong, nint, nuint, char, float, double, decimal, bool

🌴 любой enum тип

🎁 любой тип указателя (например void*)

👽 любой пользовательский struct с полями только из unmanaged типов

Иными словами такое ограничение запрещает типу иметь внутри себя ссылки, управляемые GC, то есть это не ссылочный тип, или тип без членов ссылочного типа на любом уровне вложенности.
Please open Telegram to view this post
VIEW IN TELEGRAM
👍462🔥2
Лучший совет по интернет безопасности в 2023 году
😁33💯7🔥2
Чуть больше двух лет назад я проходил собеседование в OZON, где в рамках технического этапа нужно было провести ревью кода.

Я не знал, что они так проверяли исключительно знание примитивов синхронизации и многопоточности в .NET.

И начал набрасывать, как плохо, не по ООП, спроектирован код, заявленным DDD не пахнет вообще и домен вдобавок смоделирован некорректно, а ещё принципы SOLID нарушаются.

Естественно, сказать, что тогда не прошёл то собеседование - не сказать ничего.

И вот алгоритмы ютуба решили напомнить мне о неудачном опыте, предложив видео паренька, вышедшее 29 октября этого года.

Конечно, разбирает правильно, правда немного неуверенно и нечётко, но почему за два года задача не изменилась совсем?

Буквально, код повторяется посимвольно.

Почему так, для меня загадка 🤷‍♂️
👍11😁6🤯1
ActionFilterAttribute


Это очень крутая штука, которую надо знать всем C# специалистам, разрабатывающим сервисы с помощью ASP NET Core.

Есть обычные фильтры, предоставляющие механизм построения пайплайна вокруг запроса (действия до и после, выполнение сквозного функционала и так далее).

Но простые операции для отдельных ручек можно реализовывать прямо в атрибутах, чтобы не тратить ресурсы на вычисление эндпоинта с помощью рефлексии.

Например, при необходимости проверять наличие определённого заголовка в запросе можно написать следующий атрибут.

[AttributeUsage(validOn: AttributeTargets.Method)]
public class HeaderRequiredAttribute : ActionFilterAttribute
{
private readonly string _headerName;

public HeaderRequiredAttribute(string headerName) =>
_headerName = headerName;

public override void OnActionExecuting(ActionExecutingContext context)
{
var httpContext = context.HttpContext;
var headerValue = httpContext.Request.Headers[_headerName].ToString();
if (string.IsNullOrWhiteSpace(headerValue))
throw new MissingRequiredHeaderException(_headerName);
}
}
👍14🤯4🔥3🤩1
Лайфхак для айтишников 🔥
😁30🍌4🤩2🌭2
Task vs ValueTask

Что такое Task знают все C# разработчики - класс, олицетворяющий длительную операцию в асинхронном программировании.

А вот в 7 версии языка появилась структура ValueTask, и до сих пор не всем ясно её назначение.

Из интересного в ней два поля - первое для хранения задачи, второе для хранения результата (при наличии).

Поскольку Task это класс, то его экземпляр хранится в куче, а значит высвобождением памяти будет заниматься GC.

ValueTask это структура, а значит вероятнее всего окажется на стеке, и расходов на высвобождение памяти будет потрачено существенно меньше.

Соответственно, стоит использовать эту структуру, когда мы ожидаем, что задача будет выполняться синхронно большинство времени (например, закэшированный результат) или расходы на вызов и ожидание обычной задачи будут больше.

Однако, есть ряд ограничений, то есть, нельзя делать следующие вещи:

⭐️ Ожидать задачу конкурентно или несколько раз

⭐️ Вызывать метод AsTask несколько раз

⭐️ Использовать .Result или .GetAwaiter().GetResult(), если задача не завершена

⭐️ Использовать задачу в комбинаторах WhenAll() и WhenAny()
Please open Telegram to view this post
VIEW IN TELEGRAM
👍193👏3
StepOne | Степан Минин
Внедрение конкретной реализации в стандартном контейнере Допустим, у нас есть некоторый интерфейс, который имеет несколько реализаций: public interface IDependency {} public class DependencyImplOne : IDependency {} public class DependencyImplTwo : IDependency…
ASP NET 8 наконец-то добавит многообразие зависимостей 🥳

Реализовано это будет через ключи (например строковые).

Возвращаясь к старому примеру, он преобразится следующим образом:

public interface IDependency {}

public class DependencyImplOne : IDependency {}
public class DependencyImplTwo : IDependency {}

builder.Services.AddKeyedSingleton<IDependency, DependencyImplOne>("one");
builder.Services.AddKeyedSingleton<IDependency, DependencyImplTwo>("two");


Далее, использовать вот так, с помощью атрибута [FromKeyedServices]:

public class BarService : IBarService
{
public BarService([FromKeyedServices("one")] IDependency dependency) // DependencyImplOne
{
}
}

public class BazService : IBazService
{
public BazService([FromKeyedServices("two")] IDependency dependency) // DependencyImplTwo
{
}
}


Для меня, как поклонника ООП, это знаковая веха в развитии платформы 🔥

Считаю, что ради этой киллер фичи можно смело планировать переезд на новый LTS релиз!
🔥24👍8🤯4😱2
Почему я всегда сообщаю рекрутеру зарплатные ожидания? 🤔

Есть распространённое мнение о зарплатных переговорах: «кто первый назвал цифру, тот и проиграл».

Однако это не совсем так.

Во-первых, сама культура подобных переговоров в РФ находится в достаточно зачаточном состоянии по сравнению с той же Америкой, например.

Во-вторых, разработчики это такие же рядовые сотрудники, как и другой персонал: для них существует вполне конкретный ФОТ.

То есть торги - это инструмент не растяжения мешка с деньгами, а прощупывания его границ.

И вот однажды я собеседовался в дом рф, попробовав всеми правдами и неправдами скрыть цифру 💵

Изворачивался и уходил от ответа так, как никогда в жизни 🤯

При этом тех собес был очень приятный и даже нетипичный для индустрии, перспективный проект и интересными задачами, да и ребята показались вполне адекватными…

Но когда дело дошло до оффера, меня брали по верху вилки, но оказался таким, что смысл туда идти пропал абсолютно 🥵

Сказать, что чувствовал себя тогда идиотом - не сказать ничего 😁

Да, мои зарплатные ожидания достаточно высокие, поэтому отвечаю на вопрос HR о деньгах, чтобы было понятно сразу - удовлетворит компания мои потребности или это пустая трата времени.

А терять время не хочет никто.
Please open Telegram to view this post
VIEW IN TELEGRAM
👍21🔥4💯4
StepOne | Степан Минин
Немного о pet project В прошлом посте я написал про то, как опыт коммерческой разработки повлиял на мой стиль его ведения. Однако, может возникнуть закономерный вопрос: «Так чем ты там занимаешься?» Отвечаю. Сейчас мой основной open source contribute -…
Правильность архитектуры проверяется при внедрении нового функционала

Кажется, написал что-то очевидное, но столкнулся с этим лично на своём pet project.

В рамках него я пишу интерпретатор TypeScript подобного языка.

Этот проект родился, как дипломная работа в бакалавриате бауманки.

Но поскольку сроки были сильно сжаты, то изначально всё сделано на скорую руку.

Год назад хотел добавить в язык аналог выражения with из C#, но столкнулся с тем, что из-за спешки сделать это будет больно.

Одно из самых неправильных решений в проекте - обработка абстрактного синтаксического дерева интрузивным методом.

Подсветить то, насколько важно перейти на паттерн Visitor, помогла моя попытка внедрить DDD и выделить домен системы.

Сейчас, когда рефакторинг близится к концу, все детали пазла собираются в одну картину, новые фичи собираются на глазах за счёт правильной траектории архитектурной эволюции.

При этом стали видны новые точки роста - ошибки в реализации «Посетителя», изоляция домена, изменение знания одних частей системы о других и так далее.
👍10🔥4👏21
В EF Core 8 узаконили Value Object 🤩

Одно из тех обновлений, которое не следует пропускать 😱

Например, есть следующая сущность заказа Order вместе с сущностью клиента Customer:

public class Customer
{
public int Id { get; set; }
public required string Name { get; set; }
public required Address Address { get; set; }
public List<Order> Orders { get; } = new();
}

public class Order
{
public int Id { get; set; }
public required string Contents { get; set; }
public required Address ShippingAddress { get; set; }
public required Address BillingAddress { get; set; }
public Customer Customer { get; set; } = null!;
}


Кажется, что уже существует семантика работы с типами значениями, ведь никто не отменял атрибут [Owned].

Однако, при попытке сохранить сущность, которая использует один инстанс на несколько полей возникнет ошибка:

var address = GetAddress();

customer.Orders.Add(
new Order { Contents = "Tesco Tasty Treats", BillingAddress = address, ShippingAddress = address, });

await context.SaveChangesAsync();


Кажется, что всё будет хорошо, но получим исключение InvalidOperationException с сообщением

«Cannot save instance of 'Order.ShippingAddress#Address' because it is an owned entity without any reference to its owner. Owned entities can only be saved as part of an aggregate also including the owner entity.
at Microsoft.EntityFrameworkCore.ChangeTracking.Internal.InternalEntityEntry.PrepareToSave()»

Теперь, достаточно разметить тип данных атрибутом [ComplexType].
При этом он может быть иммутабельным!

[ComplexType]
public record Address(string Line1, string? Line2, string City, string Country, string PostCode);


Соответственно, такое обновление сущности:

customer.Address = customer.Address with { Line1 = "Peacock Lodge" };

await context.SaveChangesAsync();


Сгенерирует следующий SQL:

UPDATE [Customers] SET [Address_Line1] = @p0
OUTPUT 1
WHERE [Id] = @p1;


То есть, это собственный тип на стероидах 💉
Please open Telegram to view this post
VIEW IN TELEGRAM
🔥11👍2🎉1🤣1🏆1
partial взрывает мне мозг 🤯

Недавно решил изучить такую возможность C#, как объявление типов данных в разных файлах с помощью ключевого слова partial.

В коммерческой практике не так часто это встречалось.

И я обнаружил, что с этой фичей можно делать просто супер странные вещи.

Например, у нас есть простенький интерфейс IConract:

public interface IContract
{
void Foo();
}


Сделаем раздробленный класс MyPartialClass:

public partial class MyPartialClass
{
// ...
}


И оказывается, что реализацию интерфейса этим классом можно написать вообще в другом файле, и если иметь перед глазами только первый, то об этом и не догадаешься!

public partial class MyPartialClass : IContract
{
public virtual void Foo()
{
throw new NotImplementedException();
}
}


При этом виртуальный метод Foo можно спокойно переопределять в той же манере в наследниках:

public partial class MyPartialClassDerived : MyPartialClass
{
public override void Foo()
{
Console.WriteLine(this);
base.Foo();
}
}


Опять же, догадаться о существовании подобных членов типа будет достаточно проблематично.

Злоупотреблять таким в командной работе конечно же не рекомендую.

Однако, подобный функционал будет полезен разработчикам решений на основе Source Generators.

Таким образом, можно докидывать компилятором различные бойлерплейты, которые не хочется писать руками.
💯9🤯3🙏2
StepOne | Степан Минин
Отлаживать код в Rider смогут даже младенцы 👶 JetBrains продолжают развивать лучшую IDE для C# разработчиков и удивлять их новыми обновлениями. В грядущей версии 2023.3 завезут, на мой взгляд, настоящую killer feature, которая называется Debugger Data Flow…
Релизнулся Rider 2023.3 😱

Завезли много всего интересного: поддержку .NET 8 SDK, одновременный запуск нескольких проектов, предиктивный дебаггинг, вытащили из превью AI ассистента и многое другое

Например, добавили отрисовку диаграм зависимостей типов 🎁

Теперь можно воочию увидеть архитектуру своего проекта, и убедиться, что её нужно переделать 😱

Короче говоря, побежали обновляться! 🏃‍♂️
Please open Telegram to view this post
VIEW IN TELEGRAM
🔥17😁31🤯1