На этом занятии мы познакомимся с понятиями конкурентность, параллельность, многопоточность, асинхронность в языке программирования C#. Научимся создавать новые потоки (thread), а также работать с задачами (task) с помощью асинхронных операций async и await.
@GOGOGO GOGOGO Может быть оговорился. По сути да, мы берем готовый поток из пула и помещаем туда задачу на выполнение. Но это можно также расценивать не как физическое создание, а использование уже созданного, введение в использование, по сути добавление нового асинхронного потока, скорее всего я имел ввиду это под созданием.
52:30 Можно же сделать Main асинхронным. Объяви async static Task Main(string[] args) и поставь await перед DoWorkAsync(); Тогда вывод будет более последовательный Begin Main Begin Async DoWork DoWork DoWork DoWork DoWork DoWork DoWork DoWork DoWork DoWork End Async Continue Main Main Main Main Main Main Main Main Main Main Main End Main
25:40 - Начало отладки видео. Дальше 3 минуты можно вообще не смотреть. Прошу прощения за технические неполадки 28:30 - Перемотать сюда. Лаги видеокарты закончились. Оказалось SLI с OBS вообще не дружит...
Кроме того не стоит путать concurrent и parallel. Parallel означает, что код выполняется фактически одновременно, например, двумя процессорами, а concurrent переключает ветки, в то время как каждая из них работает только в отведенное ей время, а потом засыпает, уступая место другой ветке.
Про асинхронность, извини, ты совсем не о том говоришь что нужно людям знать. Асинхронные задачи вполне могут быть важными и связанными со всей остальной программой. Дело не в том, что где-то "параллельно" что-то делается в другом потоке из thread pool - совершенно не важно как технически виртуальная машина это делает. Смысл в том, что в синхронном режиме все команды выполняются последовательно, и если будет команда которая выполняется долго (типа получения данных по сети), то это будет замедлять работу программы, так как нужно дождаться выполнения. Асинхронность в свою очередь предполагает, что не дожидаясь выполнения функции программа выйдет из нее и пойдет дальше, затем вернется к выполнению когда данные будут готовы. Поэтому проблем с общими ресурсами, доступом к общей памяти и т.п. нет, потому что выполняется НЕ ПАРАЛЛЕЛЬНО, а просто в другой последовательности.
2 года уже прошло, не знаю, увидел ли ты уже, но код в vs можно форматировать сочетанием ctrl + k + d. Он сам ставит пробелы (внутри циклов, например), отступы добавляет или убирает, в зависимости от вложенности
Кстати, метод Main можно сделать Асинхронным! Только нужно заменить возращаемое значение void на Таску)) Пример: static async Task Main() { await Task.Delay(2000); Console.WriteLine("Ждал 2 секунды, лол"); await Task.Delay(5000); Console.WriteLine("А теперь 5 сек)"); Console.ReadKey(); } Что-то на подобии этого)
@@CODEBLOG Не знаю даже на сколько Райзены глючные. Смотрел пару обзоров, вроде все нормально. AMD своей ценой при сопоставимой мощности сейчас хорошо напихала среднему классу интеловских процов. А интеловские топы на мой взгляд всегда были сильно оверпрайсд. Я сделал вывод, что серия Ryzen вышла очень хорошей учитывая стоимость и потому когда понадобится обращу на них внимание. Да и есть во мне дух борьбы против монополистов. Особенно когда оппозиция делает качественный продукт)
Здравствуйте, мастер. Я разбираюсь с клиент серв приложениями и параллельно с многопоточностью. Сделал тестовую программу, обращающуюся в к элементу формы в потоке. В результате при закрытии попадаю в исключение "Доступ к ликвидированному объекту невозможен. Имя объекта: "Form1"." . Я могу конечно его отловить ,но.... Пытался заключать invoke в проверку на IsDisposed (Существование формы)... Не помогло. Еще раз спасибо Вам большое за все.
Да, есть такая ошибка часто. И решить её достаточно сложно. в целом нужно правильно выстроить процесс завершения и прерывания потока. Смысл в том, что поток еще выполняется, а форма уже закрыта, но из памяти еще не удалена. нужно сначала остановить поток, а потом уже закрывать форму
@@CODEBLOG У меня в отдельном потоке запускается бесконечный цикл. Я завел переменную флаг, при закрытии формы ставлю ее в false, что как будто бы должно обеспечить выход из цикла и соответственно закрытие потока, так как в нем больше ничего нет. Но увы все равно получаю ошибку при закрытии, о том что объект уничтожен. Помогает только остановка потока -
Блин, перепроверял себя много раз! Цикл выполняется, пока условие верно! А значит м2 не выполнится и никакой борьбы не будет! Не знаю, что там на гитхабе, но в конце видео м2 неверный!
Чувак, если приходится много работать со строками, то лучше, нет, даже необходимо(!) юзать StringBuilder. В твоём примере это относится к тому месту, где ты хотел 100000 раз прибавлять рандомное число к строке
А всё из-за того, что строка это значимый тип и при каждом изменении строки происходит выделение памяти и прочие расходные операции. Стрингбилдер работает иначе, поэтому он быстрее, но быстрее, только в том случае, если строк реально много. Если строк мало, в скорости он может и проиграть.
Доброго времени суток! Подскажите каким образом можно исправить ошибку ниже... используя Асинхронный метод (async/await) когда Запускаем поток таким образом - все работает new Thread(() => { Invoke((MethodInvoker)(() => { //тут выполняем код })); }).Start(); ******************************* Когда вызываю асинхронный метод выдается ошибка, как ее исправить? async Task MetodAsync() { await Task.Run(()=> { //тут выполняем код } ); } Выходит ошибка: System.InvalidOperationException: "Недопустимая операция в нескольких потоках: попытка доступа к элементу управления 'label1' не из того потока, в котором он был создан."
К вечеру вроде разобрался ) Но при всем великом уважении к Вадиму Примеры в уроки далеко не самые удачные ((( ВАДИМ И МОЖНО ВОПРОС что значит повышенный приоритет ? р нем в видео рассказывалось ?
Ну да, возможно и не лучшие ) потом может быть еще сниму, да и в CLR Эта тема будет рассматриваться. а по поводу приоритета, то важно понимать, что ресурсы системы ограничены, и доступ осуществляется на конкурентной основе. Например, вычисления процессора. ОС решает сама, какой именно процесс выполнять в каждый момент времени, управляет очередями и т.д. и если задать приоритет, то вероятность первичного выполнения - повышается, но не гарантируется.
Подскажите с lock запутался,два класса записывают в один фаил разные данные в 2 потока, так как одновременно их обратиться к файлу они не могут по идее надо один залочить. Я для этого объявляю как на вашем примере object lock в первом классе и object lock во втором классе и выделяю локом цикл который записывает данные в фаил... вот как я понимаю- кто первый из потоков добрался до лока тот и за лочился до тех пор пока не закончит... правильно????
должен быть один объект синхронизации. когда один начал записывать в файл, второй будет ждать завершения. как в пословице - кто первый встал, того и тапки :)
Быстро обернуть кусок кода в регион - выделить кусок, ПКМ - Snippet - Surround With - region. Там много во что можно обернуть: в регион, в if, for, forech, class и тд Вопрос - сколько потоков можно создавать? сколько хочется? это имеет смысл для ускорения работы приложения? ну типа у нас же 4-8 ядерные процессоры, сколько то потоков могут ускорить работу? Или потоки это не об том?
Привет, потоки можно создавать по желанию, но это медленный процесс, поэтому его нужно делать осознано. Лучше использовать threadpool, оттуда потоки вызываются значительно быстрее. Иногда распараллеливание может даже замедлить выполнение, поэтому лучше хорошо все обдумать, оценить и протестировать
Добрый день. Подскажите мне, пожалуйста, можно ли метод сразу описывать как асинхронный? Или всегда сначала нужно создавать синхронный метод, а потом при необходимости делать его асинхронным?
Спасибо за ответ. А как в данном случае будет выглядеть структура метода? Если здесь мы вызываем уже существующий метод через await, то где будет это слово стоять в рассматриваемом случае? И ответьте, пожалуйста, еще на один вопрос: есть ли у Вас занятие, посвященное работе с Git? Пролистал плейлист, но не увидел. Может, просто плохо смотрел...
Ну....сумбурненько как-то получилось....выглядит неподготовленным. Очень много времени, а информации охвачено не так много. Но в целом, спасибо. В русскоязычном ютубе информации не так и много!
У вас работаю 2 камеры через USB . У одной камеры контакт USB не достаточно плотный . У меня так отключается внешний диск в момент копирования (наверно USB шнур износился или гнездо USB . Можно поменять гнездо) .
очень тихо говоришь, наушники беспроводные, радиоволнами сигнал передается, если громкость выкрутить, появляется шум из помех, смотреть не очень удобно, до этого выпуска все было лучше
а можно же было просто сделать метод записи в файл асинхронным, а саму запись отдать таску в теле и не писать лишнего на 53 строке? или это плохая практика?) async bool SaveFile(string path) { await Task.Run(() => { // запись }); return true; }
Зависит от того, где ты будешь это использовать. Вообще уже есть готовые методы async записи в файл ) а также нужно подумать, чтобы не было конфликтов при записи файл из разных потоков. но сильно страшного в таком коде я не вижу, за исключением того, что никогда не вернется false )
20 минут в видео руками размахивал и трепался. Это как 20 минут рассказывать какие тортики вкусные, а потом только приступать пробовать. Зачем тратить время?
На этом занятии мы познакомимся с понятиями конкурентность, параллельность, многопоточность, асинхронность в языке программирования C#. Научимся создавать новые потоки (thread), а также работать с задачами (task) с помощью асинхронных операций async и await.
@GOGOGO GOGOGO Может быть оговорился. По сути да, мы берем готовый поток из пула и помещаем туда задачу на выполнение. Но это можно также расценивать не как физическое создание, а использование уже созданного, введение в использование, по сути добавление нового асинхронного потока, скорее всего я имел ввиду это под созданием.
0:00 введение
4:58 конкурентность
6:27 параллельность
8:37 многопоточность
11:33 асинхронное программирование 1
13:07 чат 1
15:36 асинхронное программирование 2
21:04 практика
22:35 чат 2
24:07 практика многопоточность 1
25:36 перезапуск трансляции
29:41 практика многопоточность 2
какие же понятные уроки просто кайф!
Вадим, у тебя классные уроки! Хоть и давно это уже было. Спасибо тебе!
52:30 Можно же сделать Main асинхронным. Объяви async static Task Main(string[] args) и поставь await перед DoWorkAsync(); Тогда вывод будет более последовательный
Begin Main
Begin Async
DoWork
DoWork
DoWork
DoWork
DoWork
DoWork
DoWork
DoWork
DoWork
DoWork
End Async
Continue Main
Main
Main
Main
Main
Main
Main
Main
Main
Main
Main
End Main
25:40 - Начало отладки видео. Дальше 3 минуты можно вообще не смотреть. Прошу прощения за технические неполадки
28:30 - Перемотать сюда. Лаги видеокарты закончились. Оказалось SLI с OBS вообще не дружит...
Хороший урок, хорошо что неполадки как раз выпали на теорию) Спасибо что выложили!
1070 в слай и обс не дружат? в ходе последних событий с серией 2080 вопрос о покупке второй 1070 для стрима отпал :D
Кроме того не стоит путать concurrent и parallel. Parallel означает, что код выполняется фактически одновременно, например, двумя процессорами, а concurrent переключает ветки, в то время как каждая из них работает только в отведенное ей время, а потом засыпает, уступая место другой ветке.
каждое видео смотрю с интересом
Про асинхронность, извини, ты совсем не о том говоришь что нужно людям знать. Асинхронные задачи вполне могут быть важными и связанными со всей остальной программой. Дело не в том, что где-то "параллельно" что-то делается в другом потоке из thread pool - совершенно не важно как технически виртуальная машина это делает. Смысл в том, что в синхронном режиме все команды выполняются последовательно, и если будет команда которая выполняется долго (типа получения данных по сети), то это будет замедлять работу программы, так как нужно дождаться выполнения. Асинхронность в свою очередь предполагает, что не дожидаясь выполнения функции программа выйдет из нее и пойдет дальше, затем вернется к выполнению когда данные будут готовы. Поэтому проблем с общими ресурсами, доступом к общей памяти и т.п. нет, потому что выполняется НЕ ПАРАЛЛЕЛЬНО, а просто в другой последовательности.
Спасибо за труд!
лукас и коммент в поддержку канала!
Спасибо за обучающий стрим! Ваш блог очень сильно мне помог)) Респект)
2 года уже прошло, не знаю, увидел ли ты уже, но код в vs можно форматировать сочетанием ctrl + k + d. Он сам ставит пробелы (внутри циклов, например), отступы добавляет или убирает, в зависимости от вложенности
Вадим уже стал разработчиком Microsoft, думаю он уже узнал об этом)) Жаль, что видео перестал делать (
А откуда информация?@@OlegGalysh
@@73километр в его соц сетях было давно
он устроился в майкрософт, сам видел
Кто смотрит на 29:45, после перезапуска, перестанет мерцать. Не расходимся!)))
Кто первый встал - того и папки! Тот и папка! Тьфу, того и тапки!!!
Да, поржал ) даже у себя в вк этот момент отдельно опубликовал )
Спасибо.
29:02 Примеры кода
Для епилептиков можете после когда вывода поставить Thread.Sleep(400); будет по медленее все выводиться
Резануло по ушам: "Что бы не думали что я богатый"
Быть богатым это нормально! Спасибо за урок!
Особенно духовно богатым быть нормально, но разве кто-то сейчас под словом «богатство» это подразумевает...
@@benjaminBTN Глубина мысли на уровне "как жаль, что под словом велосипед никто не подразумевает табуретку"
Кстати, метод Main можно сделать Асинхронным!
Только нужно заменить возращаемое значение void на Таску))
Пример:
static async Task Main()
{
await Task.Delay(2000);
Console.WriteLine("Ждал 2 секунды, лол");
await Task.Delay(5000);
Console.WriteLine("А теперь 5 сек)");
Console.ReadKey();
}
Что-то на подобии этого)
Асинхроность != Паралелльность
Именно так, и насколько я помню я об этом и говорил в начале видео
Ищу работу на Джуна, решил изучить вопрос о многопоточности, думаю это сыграет мне в плюс на собеседовании)
Ну как, сыграло?))
@@richarddaniel7088 да, 5ый месяц пошел в должности младшего-фуллстека)
@@АртемЧешихин-з3л кросс
@@АртемЧешихин-з3л как успехи?
@@satanwave охуенно, 9 дней без выходных, с переработками под 20 часов и все равно дедлайн на пару дней просрали, благо заказчик не ругался совсем
Вадим тебе нужно перепаять контакты ! у меня тоже так было на старом ламповом телевизоре в детстве когда любимый ну погоди смотрел ...
Не, это не контакты. потом уже разобрался )
Ryzen one love. Буду покупать себе новый комп, возьму райзен.
почему? он не глючит так?
@@CODEBLOG Не знаю даже на сколько Райзены глючные. Смотрел пару обзоров, вроде все нормально. AMD своей ценой при сопоставимой мощности сейчас хорошо напихала среднему классу интеловских процов. А интеловские топы на мой взгляд всегда были сильно оверпрайсд. Я сделал вывод, что серия Ryzen вышла очень хорошей учитывая стоимость и потому когда понадобится обращу на них внимание. Да и есть во мне дух борьбы против монополистов. Особенно когда оппозиция делает качественный продукт)
Здравствуйте, мастер. Я разбираюсь с клиент серв приложениями и параллельно с многопоточностью. Сделал тестовую программу, обращающуюся в к элементу формы в потоке. В результате при закрытии попадаю в исключение "Доступ к ликвидированному объекту невозможен.
Имя объекта: "Form1"."
. Я могу конечно его отловить ,но.... Пытался заключать invoke в проверку на IsDisposed (Существование формы)... Не помогло. Еще раз спасибо Вам большое за все.
Бывает
Да, есть такая ошибка часто. И решить её достаточно сложно. в целом нужно правильно выстроить процесс завершения и прерывания потока. Смысл в том, что поток еще выполняется, а форма уже закрыта, но из памяти еще не удалена. нужно сначала остановить поток, а потом уже закрывать форму
@@CODEBLOG У меня в отдельном потоке запускается бесконечный цикл. Я завел переменную флаг, при закрытии формы ставлю ее в false, что как будто бы должно обеспечить выход из цикла и соответственно закрытие потока, так как в нем больше ничего нет. Но увы все равно получаю ошибку при закрытии, о том что объект уничтожен. Помогает только остановка потока -
Блин, перепроверял себя много раз! Цикл выполняется, пока условие верно! А значит м2 не выполнится и никакой борьбы не будет! Не знаю, что там на гитхабе, но в конце видео м2 неверный!
28:47 закончились лаги и начались обьяснение наконец
35:36 чем плох такой вот вариант?
Thread thread2 = new Thread(() => DoWork2(int.MaxValue));
Чувак, если приходится много работать со строками, то лучше, нет, даже необходимо(!) юзать StringBuilder. В твоём примере это относится к тому месту, где ты хотел 100000 раз прибавлять рандомное число к строке
делал и так и так. StringBuilder сработал даже не в 2 или 3 раза, а на порядок быстрее.
А всё из-за того, что строка это значимый тип и при каждом изменении строки происходит выделение памяти и прочие расходные операции.
Стрингбилдер работает иначе, поэтому он быстрее, но быстрее, только в том случае, если строк реально много.
Если строк мало, в скорости он может и проиграть.
@@anonym1548 строка это ссылочный тип, не значимый.
@@alex-rr5mt спасибо за поправку.
Верно, конечно же ссылочный.
Сложновато , мой мозг с первого раза не особо проникся данной темой (
А тема очень важная. Прям очень-очень!
Доброго времени суток! Подскажите каким образом можно исправить ошибку ниже... используя Асинхронный метод (async/await)
когда Запускаем поток таким образом - все работает
new Thread(() =>
{
Invoke((MethodInvoker)(() =>
{
//тут выполняем код
}));
}).Start();
*******************************
Когда вызываю асинхронный метод выдается ошибка, как ее исправить?
async Task MetodAsync()
{
await Task.Run(()=>
{
//тут выполняем код
}
);
}
Выходит ошибка:
System.InvalidOperationException: "Недопустимая операция в нескольких потоках: попытка доступа к элементу управления 'label1' не из того потока, в котором он был создан."
Есть ли ролик, по требованиям от работодателей к программистам С# разных уровней ?И по каким параметрам определяется уровень программиста ?
Пока нет, но будет )
@@CODEBLOG есть уже?)
@@artur8904 Вадим принял ислам, уже ничего не будет
@@VyacheslaveDev69 ушёл работать в Microsoft)
К вечеру вроде разобрался )
Но при всем великом уважении к Вадиму
Примеры в уроки далеко не самые удачные (((
ВАДИМ И МОЖНО ВОПРОС
что значит повышенный приоритет ? р нем в видео рассказывалось ?
Ну да, возможно и не лучшие ) потом может быть еще сниму, да и в CLR Эта тема будет рассматриваться.
а по поводу приоритета, то важно понимать, что ресурсы системы ограничены, и доступ осуществляется на конкурентной основе. Например, вычисления процессора. ОС решает сама, какой именно процесс выполнять в каждый момент времени, управляет очередями и т.д. и если задать приоритет, то вероятность первичного выполнения - повышается, но не гарантируется.
Подскажите с lock запутался,два класса записывают в один фаил разные данные в 2 потока, так как одновременно их обратиться к файлу они не могут по идее надо один залочить. Я для этого объявляю как на вашем примере object lock в первом классе и object lock во втором классе и выделяю локом цикл который записывает данные в фаил... вот как я понимаю- кто первый из потоков добрался до лока тот и за лочился до тех пор пока не закончит... правильно????
должен быть один объект синхронизации. когда один начал записывать в файл, второй будет ждать завершения. как в пословице - кто первый встал, того и тапки :)
Почему метод , где возвращаемое значение Task не требует return ?
Судя по тому что в одном из примеров, где возвращаемое значение должно быть Task, а по факту было просто bool, то просто Task = void
@@Упырь-н7б да , я понял
Эпилептики покинули чат
Быстро обернуть кусок кода в регион - выделить кусок, ПКМ - Snippet - Surround With - region. Там много во что можно обернуть: в регион, в if, for, forech, class и тд
Вопрос - сколько потоков можно создавать? сколько хочется? это имеет смысл для ускорения работы приложения? ну типа у нас же 4-8 ядерные процессоры, сколько то потоков могут ускорить работу? Или потоки это не об том?
Привет, потоки можно создавать по желанию, но это медленный процесс, поэтому его нужно делать осознано. Лучше использовать threadpool, оттуда потоки вызываются значительно быстрее. Иногда распараллеливание может даже замедлить выполнение, поэтому лучше хорошо все обдумать, оценить и протестировать
Добрый день. Подскажите мне, пожалуйста, можно ли метод сразу описывать как асинхронный? Или всегда сначала нужно создавать синхронный метод, а потом при необходимости делать его асинхронным?
конечно же можно ) если заранее знаешь, что он таким будет, то можно писать сразу )
Спасибо за ответ. А как в данном случае будет выглядеть структура метода? Если здесь мы вызываем уже существующий метод через await, то где будет это слово стоять в рассматриваемом случае? И ответьте, пожалуйста, еще на один вопрос: есть ли у Вас занятие, посвященное работе с Git? Пролистал плейлист, но не увидел. Может, просто плохо смотрел...
как заблочить поток в асинхронном методе?
Ничего себе лысину отполировал, хорошо объясняешь.
круто!
Спасибо
А у меня метод Main может быть асинхронным. :)
Проблематично, конечно, интегрировать асинхронность и многопоточность в свою предметную область.
C# 7.1 добавили асинхронный метод мейн
Ну....сумбурненько как-то получилось....выглядит неподготовленным.
Очень много времени, а информации охвачено не так много.
Но в целом, спасибо. В русскоязычном ютубе информации не так и много!
Застрял на этой теме. Чтобы разобраться, сейчас читаю "Современные операционные системы" 😒
Очень сложно но все равно спасибо
)))) паинт зло)
У вас работаю 2 камеры через USB . У одной камеры контакт USB не достаточно плотный . У меня так отключается внешний диск в момент копирования (наверно USB шнур износился или гнездо USB . Можно поменять гнездо) .
Не, там было дело не в камере, а в sli он с приложением для трансляций не дружит к сожалению
♥ Спасибо за видео! Нашел видео с примером создания программы с многопоточностью ruclips.net/video/X5AxuT5tbZ4/видео.html ☻
В примере блокировки никто никого не блокирует.
очень тихо говоришь, наушники беспроводные, радиоволнами сигнал передается, если громкость выкрутить, появляется шум из помех, смотреть не очень удобно, до этого выпуска все было лучше
вроде пучком
Во всех уроках ничего не готово, подаваемый материал и примеры не продуманы, куча времени впустую.
Эпилептикам не смотреть!!1
пример конечно не очень, про ящик и сотку
а можно же было просто сделать метод записи в файл асинхронным, а саму запись отдать таску в теле и не писать лишнего на 53 строке? или это плохая практика?)
async bool SaveFile(string path)
{
await Task.Run(() => { // запись });
return true;
}
Зависит от того, где ты будешь это использовать. Вообще уже есть готовые методы async записи в файл ) а также нужно подумать, чтобы не было конфликтов при записи файл из разных потоков. но сильно страшного в таком коде я не вижу, за исключением того, что никогда не вернется false )
Человек совершенно не понимает разницу между асинхронностью и многопоточностью...
++++
Если бы ты потратил бы 15 минут перед трансляцией, то ты бы предоставил бы весь материл за 30 минут вместо 1.5 часа. Экономия в два раза.
Не тратьте время, найдите чей ни будь другой урок на эту тему... товарищ всё рожает на лету из головы, к материалу не готов и плавает.
Мне всё нравится. Это у тебя что-то с головой
Согласен. Много ляпов, которые если воспроизвести на собеседовании, то сразу станет понятно, что вопросом не владеешь.
не реально смотреть с етим миганием
Ни подготовки, ни плана, ни сценария.
Очень тошно
20 минут в видео руками размахивал и трепался.
Это как 20 минут рассказывать какие тортики вкусные, а потом только приступать пробовать.
Зачем тратить время?
+++