Форум программистов, компьютерный форум, киберфорум
Delphi для начинающих
Войти
Регистрация
Восстановить пароль
Карта форума Темы раздела Блоги Сообщество Поиск Заказать работу  
 
Рейтинг 4.95/21: Рейтинг темы: голосов - 21, средняя оценка - 4.95
0 / 0 / 0
Регистрация: 23.10.2019
Сообщений: 8
1

Критические секции

19.12.2019, 14:34. Показов 4387. Ответов 2
Метки нет (Все метки)

Author24 — интернет-сервис помощи студентам
Delphi
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
unit Unit1;
 
interface
 
uses
  Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls,
  Forms, Dialogs, StdCtrls, ExtCtrls, SyncObjs;
 
type
  TForm1 = class(TForm)
    Button1: TButton;
    Button2: TButton;
    Edit2: TEdit;
    Memo1: TMemo;
    Edit1: TEdit;
    Memo3: TMemo;
    Label1: TLabel;
    Label2: TLabel;
    Label3: TLabel;
    Memo2: TMemo;
    Splitter1: TSplitter;
    Splitter2: TSplitter;
    procedure Button1Click(Sender: TObject);
    procedure Button2Click(Sender: TObject);
    procedure FormCreate(Sender: TObject);
    procedure FormDestroy(Sender: TObject);
 
  private
    { Private declarations }
  public
    { Public declarations }
  end;
 
var
  Form1: TForm1;
 
implementation
 
{$R *.dfm}
var
  n1, n2, j, i10, i20 : integer;
  p1, p2: ^integer;
  mas: array of integer;
  hThread1, hThread2: integer;
  ThreadID1, ThreadID2: cardinal;
  C_S: TCriticalSection;
 
function ThreadFunc(lp: PInteger): integer; stdcall;
var
  i: integer;
  n: integer;
begin
  while j < n2 do
    begin
      C_S.Enter;
 
      for i := 1 to n1 do
        begin
          n := Length(mas);
          if lp^ = 1 then
            Form1.Memo2.Lines.Add('mas[' + IntToStr(i+j*n1) + ']= ' + IntToStr(mas[n-1]))
          else
            Form1.Memo3.Lines.Add('mas[' + IntToStr(i+j*n1) + ']= ' + IntToStr(mas[n-1]));
          Setlength(mas,n+1);
          mas[n] := mas[n-1]+1;
        end;
 
      Inc(j);
      Sleep(5);
      C_S.Leave;
      Sleep(100);
    end;
  Result:= 1;
end;
 
procedure TForm1.Button1Click(Sender: TObject);
begin
  if (Edit1.Text ='') or (Edit2.Text = '') then
    begin
      ShowMessage('Не заданы длина порции чисел' + chr(13) + chr(10) +
                  'или их количество');
      Exit;
    end;
   n1:= StrToInt(Edit1.Text);
   n2:= StrToInt(Edit2.Text);
 
  i10:= 1;
  p1:= @i10;
  hThread1:= CreateThread(nil, // Указатель на атрибуты безопасности(по умолчанию)
               0,          // Размер стека, 0 - размер по умолчанию - 1 мегабайт
               @ThreadFunc,       // Указатель на процедуру нити
               p1,                // Указатель на аргумент этой процедуры
               CREATE_SUSPENDED,  // Нить не стартует сразу, 0 - стартует сразу
               ThreadID1);        //  Идентификатор нити, выходной параметр
if hThread1 =0 then
    begin
      ShowMessage('Не создана первая нить');
      Exit;
    end;
 
  i20:= 2;
  p2:= @i20;
  hThread2:= CreateThread(nil, // Указатель на атрибуты безопасности(по умолчанию)
               0,          // Размер стека, 0 - размер по умолчанию - 1 мегабайт
               @ThreadFunc,       // Указатель на процедуру нити
               p2,                // Указатель на аргумент этой процедуры
               CREATE_SUSPENDED,  // Нить не стартует сразу, 0 - стартует сразу
               ThreadID2);        // Идентификатор нити, выходной параметр
  if hThread2 =0 then
    begin
      ShowMessage('Не создана вторая нить');
      Exit;
    end;
end;
 
 
procedure TForm1.Button2Click(Sender: TObject);
begin
     inherited;
  ResumeThread(hThread1);
  ResumeThread(hThread2);
end;
 
procedure TForm1.FormCreate(Sender: TObject);
begin
  SetLength(mas, 1);
  mas[0]:= 1;
  // Создаем критическую секцию
  C_S:= TCriticalSection.Create;
end;
 
 
procedure TForm1.FormDestroy(Sender: TObject);
begin
// Удаляем критическую секцию
  C_S.Free;
end;
 
 
 
 
end.
Проблема в критической секции, почему то не хочет туда входить. Помогите, пожалуйста, разобраться.(я вообще так сделала или нет)
0
Лучшие ответы (1)
Programming
Эксперт
94731 / 64177 / 26122
Регистрация: 12.04.2006
Сообщений: 116,782
19.12.2019, 14:34
Ответы с готовыми решениями:

Синхронизация потоков и критические секции
Всем привет. Нужно из списка TStringList вывести в Memo все строки по порядку в несколько потоков...

Доступ к одному методу из двух потоков, критические секции
Вопрос такой: имеется объект MyObject класса TExample, содержащего метод DrawSomething(Cnv:...

не выводит секции
Привет всем! Такая сейчас проблема процедура записи в INI:...

Странности секции Initialization
Не совсем понимаю работу секции Initialization и не могу понять как так происходит. Есть...

2
884 / 404 / 174
Регистрация: 20.10.2016
Сообщений: 1,828
20.12.2019, 03:53 2
Лучший ответ Сообщение было отмечено LisaU как решение

Решение

Сама по себе концепция, в общем, неверная, на мой взгляд. Но использование CreateThread вместо TThread обусловлено условием задачи или это личные предпочтения? Думаю, в этом у вас и проблема

Теперь про неверность:
Первое правило работы с потоками: никогда не обращаться к компонентам формы из потока без синхронизации.
Второе правило работы с потоками: вообще никогда не обращаться к компонента формы из потока без синхронизации.

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

Но на самом деле не ок. Вы упускаете один важный момент: на самом деле потоков не два, а три. Один - hThread1, второй hThread2, третий - основной поток вашей программы. И как раз основной поток будет работать с компонентами формы без всякой синхронизации. То есть, вызывать различные перерисовки компонентов, их обновления и т.д., и все это будет вне критических секций. Так что, фактически, это будет аналогом работы с элементами формы без всякой синхронизации. А тут начинаем смотреть на первое правило работы с потоками.
1
0 / 0 / 0
Регистрация: 23.10.2019
Сообщений: 8
20.12.2019, 18:31  [ТС] 3
Спасибо, я совершенно упустила основную нить. Моя проблема решилась тем, что создание критической секции было при запуске формы, поэтому я перекинула ее создание в кнопку, тогда треды по очереди и выполняются, а так было, что секция "вечно" занята. И да, по сути это и было заданием использовать именно CreateThread.
0
20.12.2019, 18:31
IT_Exp
Эксперт
87844 / 49110 / 22898
Регистрация: 17.06.2006
Сообщений: 92,604
20.12.2019, 18:31
Помогаю со студенческими работами здесь

Использование критической секции
CriticalSection.Enter; Inc(Acc); if Acc &lt; Accounts.Count then CurAcc := Acc else Work...

Запись ID INI секции
Добрый день форумчане! Вот появился один вопросик как сделать запись ID секции INI,то есть номер...

Парсинг xml и вывод секции
Здравствуйте уважаемые форумчане, в общем передо мной стоит такая задача: есть кусок кода, где ...

Создание секции в ini файле
Здравствуйте. Нужно программно записать в файл секцию. Есть для этого функции? Или просто, как...


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

Или воспользуйтесь поиском по форуму:
3
Ответ Создать тему
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2024, CyberForum.ru