-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathpid_control.py
107 lines (87 loc) · 2.95 KB
/
pid_control.py
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
import numpy as np
import matplotlib.pyplot as plt
from cstr_reactor import simulate_cstr, cstr # Ensure cstr is imported
from scipy.integrate import odeint
# PID parameters and initial conditions
Kc = 9.23461230362
tauI = 0.22836124114
# Control loop
def pid_control(T_ss, u_ss, t, Tf, Caf, x0):
op = np.ones(len(t)) * u_ss
e = np.zeros(len(t))
ie = np.zeros(len(t))
P = np.zeros(len(t))
I = np.zeros(len(t))
# Initialize Ca and T arrays
Ca = np.ones(len(t)) * x0[0]
T = np.ones(len(t)) * x0[1]
# Upper and Lower limits on OP
op_hi = 350.0
op_lo = 250.0
# Define the setpoint ramp or steps
sp = np.ones(len(t)) * T_ss
for i in range(15):
sp[i * 20:(i + 1) * 20] = 300 + i * 7.0
sp[300] = sp[299]
# Create plot
plt.figure(figsize=(10, 7))
plt.ion()
plt.show()
for i in range(len(t) - 1):
# i is current, i + 1 is future, i - 1 is past
delta_t = t[i + 1] - t[i]
e[i] = sp[i] - T[i]
if i >= 1:
ie[i] = ie[i - 1] + e[i] * delta_t
P[i] = Kc * e[i]
I[i] = Kc / tauI * ie[i]
op[i] = 300.0 + P[i] + I[i]
if op[i] > op_hi:
op[i] = op_hi
ie[i] = ie[i] - e[i] * delta_t
if op[i] < op_lo:
op[i] = op_lo
ie[i] = ie[i] - e[i] * delta_t
x0 = [Ca[i], T[i]]
ts = [t[i], t[i+1]]
y = simulate_cstr(x0,ts,op[i],Tf,Caf)
Ca[i + 1] = y[-1][0]
T[i + 1] = y[-1][1]
# Debugging information
# if i % 50 == 0:
# print(f"Time: {t[i]:.2f}, Setpoint: {sp[i]:.2f}, PV: {pv[i]:.2f}, OP: {op[i]:.2f}, Ca: {Ca[i]:.2f}, T: {T[i]:.2f}")
# Plotting
plt.clf()
plt.subplot(3, 1, 1)
plt.plot(t[:i + 1], sp[:i + 1], 'r--', label='Setpoint')
plt.plot(t[:i + 1], T[:i + 1], 'b-', label='Process Variable (Reactor Temp)')
plt.ylabel('Reactor Temperature (C)')
plt.legend(loc='best')
plt.subplot(3, 1, 2)
plt.plot(t[:i + 1], op[:i + 1], 'k-', label='Control Output (Cooling Jacket Temp)')
plt.ylabel('Cooling Jacket Temperature (C)')
plt.xlabel('Time (sec)')
plt.legend(loc='best')
plt.subplot(3, 1, 3)
plt.plot(t[:i + 1], Ca[:i + 1], 'g-', label='Concentration Ca')
plt.ylabel('Concentration Ca')
plt.xlabel('Time (sec)')
plt.legend(loc='best')
plt.pause(0.01)
op[len(t) - 1] = op[len(t) - 2]
ie[len(t) - 1] = ie[len(t) - 2]
P[len(t) - 1] = P[len(t) - 2]
I[len(t) - 1] = I[len(t) - 2]
# Save data to file
data = np.vstack((t, u, T, Ca, op, ie )).T
np.savetxt('cstr_output.txt', data, delimiter=',')
plt.ioff()
plt.show()
return u, T
# Main execution
if __name__ == "__main__":
t = np.linspace(0, 10, 301)
x0 = [0.87725294608097, 324.475443431599]
u_ss = 300.0
T_ss = 324.475443431599
u, T = pid_control(T_ss, u_ss, t, 350, 1, x0)