Форум программистов, компьютерный форум, киберфорум
C# Windows Forms
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
 
Рейтинг 4.56/9: Рейтинг темы: голосов - 9, средняя оценка - 4.56
44 / 7 / 1
Регистрация: 05.08.2013
Сообщений: 60

Потоки и форма

27.09.2019, 13:21. Показов 1984. Ответов 23
Метки нет (Все метки)

Студворк — интернет-сервис помощи студентам
C#
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
using System.Linq;
using System.Text;
using System.Threading;
using System.Threading.Tasks;
using System.Windows.Forms;
 
namespace Server
{
    public partial class Form1 : Form
    {
        private Thread thread;
        private bool start = true; // Флаг для выхода из цикла
        
        private void listener() //Потоковый метод
        {
            int count = 0;
 
            while (start)
            {
                count++;
 
                if (InvokeRequired) { //Инвок
                    Invoke(new Action(() =>
                    {
                        Log.Text += count.ToString() + "\r\n"; //Обращаемся к текстовому полю, что в основном потоке
                    }));
                }
            };             
 
           
        }
 
        public Form1()
        {
            InitializeComponent();
        }
 
        private void Form1_Load(object sender, EventArgs e) //Ставртуем поток в загрузке формы
        {
            thread = new Thread(listener); 
            thread.IsBackground = true;
            thread.Start();
        }        
 
        private void Form1_FormClosed(object sender, FormClosedEventArgs e) //Закрытие формы
        {
            start = false;
            if (thread.IsAlive) // Если поток жив, ликвидируем его.
            {
                thread.Abort();
            }
        }
 
     }

Вопрос такой.

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

Почему эта ошибка вообще выходит? Ведь это фоновый поток и он должен закрываться при закрытии?
Почему кнопкой получается поставить флаг и выйти из цикла, а событием закрытия формы - нет?

Всем огромное спасибо.
0
cpp_developer
Эксперт
20123 / 5690 / 1417
Регистрация: 09.04.2010
Сообщений: 22,546
Блог
27.09.2019, 13:21
Ответы с готовыми решениями:

serialPort, форма и потоки
Допустим, на форме имеется label и к проекту прикручен стандртный компонент "serialPort" По приему байта пытаемся изменить текст...

Форма и потоки
Добрый день, всегда работал без интерфейсов, теперь перешёл на интерфейсы и столкнулся с вопросом как запустить поток, который при закрытии...

Добавление записи в Listview(1 форма) через Диалоговое окно(3 форма) другой формы (2 форма)
Всем доброго дня и с наступающими праздниками! Знаю, тема 7 частых вопросов по WinForms уже не раз обсуждалась, и была прочитана, но или...

23
Эксперт .NET
6691 / 4102 / 1607
Регистрация: 09.05.2015
Сообщений: 9,575
27.09.2019, 13:23
Попробуйте событие Closing вместо Closed. Ну и Thread.Abort тут не нужен.
0
44 / 7 / 1
Регистрация: 05.08.2013
Сообщений: 60
27.09.2019, 13:28  [ТС]
Увы, не получается... Почему-то если я кнопку ставлю и в ней флаг меняю - все ажурно. И цикл останавливается, и поткок закрывается сам. А стоит в закрытии формы, прям беда. Closing - пробовал. Это было первое, что попробовал.

Почему ошибка вообще выходит? Ведь гибнет основной поток, фоновый должен сам уйти...

Больше в коде ничего нет...

Спасибо Вам.
0
44 / 7 / 1
Регистрация: 05.08.2013
Сообщений: 60
27.09.2019, 13:42  [ТС]
Вот я приложил скриншот, на котором вида ошибка и даже состояние флага цикла (false). Как же происходит попадание в цикл?
Миниатюры
Потоки и форма  
0
Эксперт .NET
6691 / 4102 / 1607
Регистрация: 09.05.2015
Сообщений: 9,575
27.09.2019, 13:47
Цитата Сообщение от PsiMagistr Посмотреть сообщение
Как же происходит попадание в цикл?
Например было true, зашло в цикл, стало false, а цикл еще не завершился...

У меня кстати ваша ошибка не воспроизводится (VS2019 16.4.0 Preview 1.0, .NET Framework 4.8), исключение не вылетает ни с Closing, ни с Closed... Возможно зависит от каких-то внешних факторов, например оборудования...
1
44 / 7 / 1
Регистрация: 05.08.2013
Сообщений: 60
27.09.2019, 13:54  [ТС]
Спасибо огромное. Вы не могли бы приложить Ваш код? Точно тот же самый?

От оборудования вряд ли...
0
Эксперт .NET
6691 / 4102 / 1607
Регистрация: 09.05.2015
Сообщений: 9,575
27.09.2019, 13:57
Цитата Сообщение от PsiMagistr Посмотреть сообщение
Вы не могли бы приложить Ваш код? Точно тот же самый?
Такой же код, ваш скопировал...
C#
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Globalization;
using System.Linq;
using System.Text;
using System.Threading;
using System.Threading.Tasks;
using System.Windows.Forms;
 
namespace WindowsFormsApp1234
{
    public partial class Form1 : Form
    {
        private Thread thread;
        private bool start = true; // Флаг для выхода из цикла
 
        private void listener() //Потоковый метод
        {
            int count = 0;
 
            while (start)
            {
                count++;
 
                if (InvokeRequired)
                { //Инвок
                    Invoke(new Action(() =>
                    {
                        textBox1.Text += count.ToString() + "\r\n"; //Обращаемся к текстовому полю, что в основном потоке
                    }));
                }
            };
        }
 
        public Form1()
        {
            InitializeComponent();
        }
 
        private void Form1_FormClosing(object sender, FormClosingEventArgs e)
        {
            //start = false;
            //if (thread.IsAlive) // Если поток жив, ликвидируем его.
            //{
            //    thread.Abort();
            //}
        }
 
        private void Form1_FormClosed(object sender, FormClosedEventArgs e)
        {
            start = false;
            if (thread.IsAlive) // Если поток жив, ликвидируем его.
            {
                thread.Abort();
            }
        }
 
        private void Form1_Load(object sender, EventArgs e)
        {
            thread = new Thread(listener);
            thread.IsBackground = true;
            thread.Start();
        }
    }
}
0
44 / 7 / 1
Регистрация: 05.08.2013
Сообщений: 60
27.09.2019, 14:10  [ТС]
Хм... У меня там просто искуственное прерывание потока, через Abort. Может оно и к Вам попало?

И что теперь делать - абортами бить по потоку?
0
Эксперт .NET
6691 / 4102 / 1607
Регистрация: 09.05.2015
Сообщений: 9,575
27.09.2019, 15:22
Цитата Сообщение от PsiMagistr Посмотреть сообщение
Хм... У меня там просто искуственное прерывание потока, через Abort. Может оно и к Вам попало?
Да, если убрать Thread.Abort то ошибка проявляется.
0
44 / 7 / 1
Регистрация: 05.08.2013
Сообщений: 60
27.09.2019, 15:28  [ТС]
Ну так Abort я и поставил как аварийный выход.... Можно еще в исключение, но как то это все не то...
0
 Аватар для Sanya_sa
912 / 816 / 333
Регистрация: 03.02.2015
Сообщений: 5,276
Записей в блоге: 9
27.09.2019, 15:52
PsiMagistr, Вы флаг start = false; выставили и по идеи поток сам закроется.
Для чего Abort()?
0
Эксперт .NET
6691 / 4102 / 1607
Регистрация: 09.05.2015
Сообщений: 9,575
27.09.2019, 15:52
Вот так вроде работает
C#
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Diagnostics;
using System.Drawing;
using System.Globalization;
using System.Linq;
using System.Text;
using System.Threading;
using System.Threading.Tasks;
using System.Windows.Forms;
 
namespace WindowsFormsApp1234
{
    public partial class Form1 : Form
    {
        private Thread thread;
        private bool start = true; // Флаг для выхода из цикла
 
        private void listener(object arg) //Потоковый метод
        {
            IProgress<int> progress = (IProgress<int>)arg;
            int count = 0;
 
            while (start)
            {
                count++;
 
                progress.Report(count);
 
                Thread.Sleep(10); // без слипа всё повиснет, т.к. этот код работает намного быстрее чем вариант с Invoke
            };
 
            Console.WriteLine("Thread Exit");
        }
 
        public Form1()
        {
            InitializeComponent();
        }
 
        private void Form1_FormClosing(object sender, FormClosingEventArgs e)
        {
            Console.WriteLine("Form1_FormClosing");
            start = false;
        }
 
        private void Form1_Load(object sender, EventArgs e)
        {
            var progress = new Progress<int>((p) => textBox1.Text += p + Environment.NewLine);
 
            thread = new Thread(listener);
            thread.IsBackground = true;
            thread.Start(progress);
        }
    }
}
1
44 / 7 / 1
Регистрация: 05.08.2013
Сообщений: 60
29.09.2019, 08:58  [ТС]
Именно потому что невзирая на флаг вылетает исключение при закрытии.

Добавлено через 3 минуты
Спасибо огромное. Но мне нужет поток непрерывный, увы. Слипить не вариант.

Добавлено через 40 минут
Потому что с флагом не работает. Посмотрите скриншот выше.
0
Эксперт .NET
6691 / 4102 / 1607
Регистрация: 09.05.2015
Сообщений: 9,575
29.09.2019, 13:35
Цитата Сообщение от PsiMagistr Посмотреть сообщение
Но мне нужет поток непрерывный, увы. Слипить не вариант.
По скорости работы примерно одинаково... Т.к. Invoke прокидывает ваш делегат через очередь, а это занимает время...
Это так же скорее всего является причиной ошибки (делегат поместили в очередь, форма задиспозилась, а очередь то все еще работает)
1
44 / 7 / 1
Регистрация: 05.08.2013
Сообщений: 60
29.09.2019, 14:03  [ТС]
Спасибо.

А вообще, что случается при закрытии потоков? Если например мы открыли фоновый поток, закрыли основной (форма), а если в фоновом потоке соединение с сервером и т.д.. Что случается с ресурсами?
0
44 / 7 / 1
Регистрация: 05.08.2013
Сообщений: 60
20.10.2019, 16:12  [ТС]
Ребят, никто не скажет, почему при закоытии сервера, клиент закоывается?
Ничего специально для этого я не делал.
0
5 / 5 / 1
Регистрация: 30.09.2015
Сообщений: 17
21.10.2019, 02:26
Фоновые потоки обязаны закрыться при завершении остальных потоков (не фоновых). При это не имеет значения, какую операцию они производят. Все ресурсы (как и ресурсы остальных потоков) - в мусор. Буквально. Есть такой инструмент - сборщик мусора.
Если Вам хочется корректно завершить все операции, придётся самому создавать не фоновый поток, который может жить дольше основного потока.
Недавно создавал такой класс. Для периодического запуска задачи в потоке с корректным выходом. Но там со "слипами" - так надо.

По теме.
Вероятно Invoke глючит от того, что Ваш поток назначает invoke в то время, как главный поток занят под завязку тем, что завершает свою работу - уничтожает объекты. Ему просто "некогда" выполнить заданный invoke. А когда он освобождается, пытается наконец выполнить последнюю задачу, но формы уже нет. Потому и не завершается рабочий поток, ведь главный поток ещё жив. То же и с start - параллельный поток ждёт, пока выполнится Invoke и просто не проверяет состояние start.

Вариант решения:
В Form1_FormClosing запретить закрывать форму. Форма не уничтожится, start=false отработает, поток завершится.
C#
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
private void listener() //Потоковый метод
        {
            int count = 0;
 
            while (start)
            {
                count++;
 
                if (InvokeRequired)
                { //Инвок
                    Invoke(new Action(() =>
                    {
                        textBox1.Text += count.ToString() + "\r\n"; //Обращаемся к текстовому полю, что в основном потоке
                    }));
                }
            };
                    Invoke(new Action(() =>
                    {
                        Close();
                    }));
        }
 
        private void Form1_FormClosing(object sender, FormClosingEventArgs e)
        {
            if (start)
           {
             e.Cancel = true;
             start = false;
           }
        }
Тут я отменяю закрытие формы при первом вызове Close. Поток завершается и повторно вызывает Close() уже с нормальным завершением.

Добавлено через 14 минут
И еще, если создать другой класс, обычный, не Form, то объект этого класса может прожить дольше, чем form. И применив Invoke к объекту такого класса, мы таки попадём внутрь Invoke и получим ошибку уже внутри метода. Разница в том, что эту ошибку уже можно будет корректно обработать через try...catch или проверкой существования объекта типа Form1.

К слову, я всегда предпочитаю создавать отдельные классы для логики приложения. Чтобы интерфейсы (окна, кнопки и т.п.) отдельно, обработка отдельно.
0
44 / 7 / 1
Регистрация: 05.08.2013
Сообщений: 60
23.10.2019, 15:20  [ТС]
C#
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Net;
using System.Net.Sockets;
using System.Text;
using System.Threading;
using System.Threading.Tasks;
using System.Windows.Forms;
 
namespace Client
{
    public partial class Form1 : Form
    {
        // private const string ip = "127.0.0.1";
        private bool start = true;
        private const int port = 8083;
        private Socket _serverSocket;
        private Thread thread;
        private bool Contacts = false;
        private void Connect(string ip)
        {
            try
            {
                IPEndPoint ipEndPoint = new IPEndPoint(IPAddress.Parse(ip), port);                
                 _serverSocket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);               
                 _serverSocket.Connect(ipEndPoint);
               
            }
            catch(Exception ex)
            {
                Chat.Text = ex.Message;
               // Chat.Text = Encoding.(Chat.Text, Encoding.Default);
            }
        }
 
        private void listener() //Тут
        {
            //int length = 256;
            while (start)
            {
                int length = 0;
                var size = 0;
                int recived = 0;
                var bt = new byte[4];
                do
                {
                    recived += _serverSocket.Receive(bt, recived, 4 - recived, SocketFlags.None);
 
                } while (recived < 4);
 
                length = BitConverter.ToInt32(bt, 0);                
                if (length > 0)
                {
                    try
                    {
                        MessageBox.Show(length.ToString());
                        var buffer = new byte[length];
                        size = 0;
                        var data = new StringBuilder();
                        do
                        {
                            size = _serverSocket.Receive(buffer);
                            data.Append(Encoding.UTF8.GetString(buffer, 0, size));
                        } while (_serverSocket.Available > 0);
 
                        //  MessageBox.Show("Вышел2");
 
                        if (InvokeRequired)
                        {
                            Invoke(new Action(() =>
                            {
                                if (data.ToString().Contains("<loader>"))
                                {
                                    var d = data.ToString().Remove(0, 8);
                                    users.Items.Clear();
                                    string[] names = d.Split('>');
                                    foreach (var name in names)
                                    {
                                        users.Items.Add(name);
                                    }
                                    var bytes = Encoding.UTF8.GetBytes("<shalom>");
                                    _serverSocket.Send(BitConverter.GetBytes(bytes.Length));
                                    _serverSocket.Send(bytes); //Тут                                                                                               
                                }
                                else /*if (data.ToString().Contains("<msg>"))*/
                                {
                                    Chat.Text = String.Empty;
                                    Chat.Text = data.ToString();
                                }
 
                            }));
                        }
 
                    }
                    catch (Exception ex)
                    {
                        if (InvokeRequired)
                        {
                            Invoke(new Action(() =>
                            {
                                Chat.Text = ex.Message;
                            }));
                        }
                    }
 
                }                      
            
            }
            MessageBox.Show("543");
           // Close();
        }
        public Form1()
        {
            InitializeComponent();
        }
 
       
 
        private void Form1_FormClosing(object sender, FormClosingEventArgs e)
        {
            MessageBox.Show(e.CloseReason.ToString());
            if (_serverSocket != null  && _serverSocket.Connected == true)
            {
                e.Cancel = true; //Запрет закрытия
                var bytes = Encoding.UTF8.GetBytes("<bye>" + Name);
                _serverSocket.Send(BitConverter.GetBytes(bytes.Length));
                _serverSocket.Send(Encoding.UTF8.GetBytes("<bye>" + Name));
                _serverSocket.Shutdown(SocketShutdown.Both);
                _serverSocket.Close();                
                start = false;
                MessageBox.Show(e.CloseReason.ToString());
            }
            
               
        }
 
        private void btnMessage_Click(object sender, EventArgs e)
        {
             var bytes = Encoding.UTF8.GetBytes(Name + " пишет: " + txtMessage.Text);             
            _serverSocket.Send(BitConverter.GetBytes(bytes.Length));
            _serverSocket.Send(bytes);
            txtMessage.Text = String.Empty;
            txtMessage.Focus();              
        }
 
        private void btnName_Click(object sender, EventArgs e)
        {
            Name = txtName.Text;
            Connect(txtIp.Text);
            if (_serverSocket != null &&  _serverSocket.Connected) {
                btnName.Enabled = false;
                txtName.Enabled = false;
                txtIp.Enabled = false;                
                txtMessage.Enabled = true;               
                thread = new Thread(listener);
                thread.IsBackground = true;
                thread.Start();                
                var bytes = Encoding.UTF8.GetBytes("<setname>" + txtName.Text);
               _serverSocket.Send(BitConverter.GetBytes(bytes.Length));               
               _serverSocket.Send((bytes));                
                txtMessage.Focus();
            }
        }
 
        private void txtMessage_TextChanged(object sender, EventArgs e)
        {
            btnMessage.Enabled = !String.IsNullOrWhiteSpace(((TextBox)sender).Text);
        }
 
        private void txtName_TextChanged(object sender, EventArgs e)
        {
            btnName.Enabled = !String.IsNullOrWhiteSpace(((TextBox)sender).Text);
        }
 
        private void Form1_Load(object sender, EventArgs e)
        {
          //  MessageBox.Show(System.Text.Encoding.Default.ToString());
        }
    }
}
Друзья, это полный код клиента. Несмотря на запрет стандартного закрытия, по совету уважаемого DmitryB84, при нажатии на крестик форма пропадает с экрана.
0
5 / 5 / 1
Регистрация: 30.09.2015
Сообщений: 17
23.10.2019, 17:29
А ошибки не выскакивают?
Что выводят MessageBoxы?

Проверьте через отладку, что внутрь условия Form1_FormClosing компьютер попадает. Мало ли что...

Повторный вызов Close у Вас закоментирован - видимо для проверки. Но работать он не должен. Это метод формы, значит нужен Invoke.

Немного не по теме. А точно надо посылать данные сокетом в Invoke? Ещё и не асинхронным методом. Форму не подвешивает такой подход? Вроде как Socket не заблокирован формой и его можно использовать во всех потоках.

Добавлено через 7 минут
Да и, на всякий случай, событие Form1_FormClosing в конструкторе назначено форме или так просто процедура в классе?
0
44 / 7 / 1
Регистрация: 05.08.2013
Сообщений: 60
24.10.2019, 13:40  [ТС]
Благодарю за ответ.

1) Ошибки не выскакивают.

2) MessageBox выводит UserClosing. При том почему-то несколько раз.
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
raxper
Эксперт
30234 / 6612 / 1498
Регистрация: 28.12.2010
Сообщений: 21,154
Блог
24.10.2019, 13:40
Помогаю со студенческими работами здесь

Форма 1 вызывает форму 2, форма 2 закрывается, форма 1 продолжает работать дальше
Вынес предстартовые действия формы1 в форму2 (там у меня проверка существования файлов, их закачка если нету и проверка обновления), но как...

Что такое потоки ввода, потоки вывода?
Здарова всем! Не так давно уже прогаю на С++ и все НИКАК не могу понять, что такое потоки ввода, потоки вывода..! вот допустим...

Байтовые потоки и потоки символов
Объясните, пожалуйста подробно что имелось ввиду На самом низком уровне все операции ввода/вывода в С# оперируют байтами. Подобный...

Не закрывается форма 1, а после вызова формы 2, форма 1 дублируется несколько раз
имеется форма, на ней кнопка. нажимаю и открывается новая форма, но старая снова появляется неограниченное количество раз код 1...

Форма авторизации - вне зависимости от выбранного пользака, открывается одна и та же форма
Форма авторизации - вне зависимости от выбранного пользака, открывается одна и та же форма Есть два пользака - главбух и ведущий бух В...


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
20
Ответ Создать тему
Новые блоги и статьи
Отправка уведомления на почту при изменении наименования справочника
Maks 24.03.2026
Программная отправка письма электронной почты на примере изменения наименования типового справочника "Склады" в конфигурации БП3. Перед реализацией необходимо выполнить настройку системной учетной. . .
модель ЗдравоСохранения 5. Меньше увольнений- больше дохода!
anaschu 24.03.2026
Теперь система здравосохранения уменьшает количество увольнений. 9TO2GP2bpX4 a42b81fb172ffc12ca589c7898261ccb/ https:/ / rutube. ru/ video/ a42b81fb172ffc12ca589c7898261ccb/ Слева синяя линия -. . .
Midnight Chicago Blues
kumehtar 24.03.2026
Такой Midnight Chicago Blues, знаешь?. . Когда вечерние улицы становятся ночными, а ты не можешь уснуть. Ты идёшь в любимый старый бар, и бармен наливает тебе виски. Ты смотришь на пролетающие. . .
SDL3 для Desktop (MinGW): Вывод текста со шрифтом TTF с помощью библиотеки SDL3_ttf на Си и C++
8Observer8 24.03.2026
Содержание блога Финальные проекты на Си и на C++: finish-text-sdl3-c. zip finish-text-sdl3-cpp. zip
Жизнь в неопределённости
kumehtar 23.03.2026
Жизнь — это постоянное существование в неопределённости. Например, даже если у тебя есть список дел, невозможно дойти до точки, где всё окончательно завершено и больше ничего не осталось. В принципе,. . .
Модель здравоСохранения: работники работают быстрее после её введения.
anaschu 23.03.2026
geJalZw1fLo Корпорация до введения программа здравоохранения имела много невыполненных работниками заданий, после введения программы количество заданий выросло. Но на выплатах по больничным это. . .
Контроль уникальности заводского номера
Maks 23.03.2026
Алгоритм контроля уникальности заводского (или серийного) номера на примере нетипового документа выдачи шин для спецтехники с табличной частью, разработанного в конфигурации КА2. Данные берутся из. . .
Хочу заставить корпорации вкладываться в здоровье сотрудников: делаю мат модель здравосохранения
anaschu 22.03.2026
e7EYtONaj8Y Z4Tv2zpXVVo https:/ / github. com/ shumilovas/ med2. git
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru