-
-
Notifications
You must be signed in to change notification settings - Fork 318
/
Copy pathSortThds.pas
139 lines (116 loc) · 3.65 KB
/
SortThds.pas
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
unit SortThds;
interface
uses
Types, Classes,
Graphics, ExtCtrls,
PythonEngine;
type
{ TSortThread }
PSortArray = ^TSortArray;
TSortArray = array[0..MaxInt div SizeOf(Integer) - 1] of Integer;
TSortThread = class(TPythonThread)
private
FModule: TPythonModule;
FScript: TStrings;
FBox: TPaintBox;
FSortArray: PSortArray;
FSize: Integer;
FI, FJ: Integer;
fpyfuncname: string;
procedure DoVisualSwap;
function getvalue(i: integer): integer;
protected
procedure ExecuteWithPython; override;
public
property value[i: integer]: integer read getvalue; default;
constructor Create( AThreadExecMode: TThreadExecMode; script: TStrings;
module: TPythonModule; apyfuncname: string;
Box: TPaintBox; var SortArray: array of Integer);
procedure VisualSwap(I, J: Integer);
end;
procedure PaintLine(Canvas: TCanvas; I, Len: Integer);
implementation
procedure PaintLine(Canvas: TCanvas; I, Len: Integer);
begin
Canvas.PolyLine([Point(0, I * 2 + 1), Point(Len, I * 2 + 1)]);
end;
{ TSortThread }
constructor TSortThread.Create( AThreadExecMode: TThreadExecMode; script: TStrings;
module: TPythonModule; apyfuncname: string;
Box: TPaintBox; var SortArray: array of Integer);
begin
fpyfuncname := apyfuncname;
fScript := script;
FModule := module;
FBox := Box;
FSortArray := @SortArray;
FSize := High(SortArray) - Low(SortArray) + 1;
FreeOnTerminate := True;
ThreadExecMode := AThreadExecMode;
inherited Create(False);
end;
{ Since DoVisualSwap uses a VCL component (i.e., the TPaintBox) it should never
be called directly by this thread. DoVisualSwap should be called by passing
it to the Synchronize method which causes DoVisualSwap to be executed by the
main VCL thread, avoiding multi-thread conflicts. See VisualSwap for an
example of calling Synchronize. }
procedure TSortThread.DoVisualSwap;
type
pinteger = ^integer;
var t: integer;
pi,pj: pinteger;
begin
with FBox do
begin
Canvas.Pen.Color := clBtnFace;
pi := @(FSortArray^[FI]);
pj := @(FSortArray^[FJ]);
PaintLine(Canvas, FI, pi^);
PaintLine(Canvas, FJ, pj^);
Canvas.Pen.Color := clRed;
PaintLine(Canvas, FI, pj^);
PaintLine(Canvas, FJ, pi^);
t := pi^;
pi^ := pj^;
pj^ := t;
end;
end;
{ VisusalSwap is a wrapper on DoVisualSwap making it easier to use. The
parameters are copied to instance variables so they are accessable
by the main VCL thread when it executes DoVisualSwap }
procedure TSortThread.VisualSwap(I, J: Integer);
begin
Py_BEGIN_ALLOW_THREADS;
if Terminated then
raise EPythonError.Create( 'Pythonthread terminated');
FI := I;
FJ := J;
Synchronize(DoVisualSwap);
Py_END_ALLOW_THREADS;
end;
{ The Execute method is called when the thread starts }
procedure TSortThread.ExecuteWithPython;
var pyfunc: PPyObject;
begin
with GetPythonEngine do
begin
if Assigned(FModule) and (ThreadExecMode = emNewInterpreter) then
FModule.InitializeForNewInterpreter;
if Assigned(fScript) then
ExecStrings(fScript);
pyfunc := FindFunction( ExecModule, AnsiString(fpyfuncname));
if Assigned(pyfunc) then
try
EvalFunction(pyfunc,[NativeInt(self),0,FSize]);
finally
Py_DecRef(pyfunc);
end;
end;
end;
function TSortThread.getvalue(i: integer): integer;
begin
if Terminated then
raise EPythonError.Create( 'Pythonthread terminated');
Result := FSortArray^[i];
end;
end.