-
Notifications
You must be signed in to change notification settings - Fork 0
/
ICSolarThermalBalance.py
405 lines (334 loc) · 15.5 KB
/
ICSolarThermalBalance.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
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
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
# Written by Justin Shultz from Rensselaer Polytechnic Institute
# for Assad Oberai, with his assistance
import numpy
import math
import matplotlib.pyplot as plt
numpy.set_printoptions(linewidth=150)
def Cp(liquid, T):
# This function calculates the fSpecific Heat of Water and Air based on Temperature
# The equation for Water was a 6 degree polynomial calculated using a temperature range from 5-105C
# The specifc heat of air does not change over the temperature range of the system and can be set constant
if liquid == "w":
# Constants determined from a 5 degree fit polynomial using MatLab CurveFit Toolbox
# R = 0.9986 & SSE = 5.556e-06
p1 = -4.178e-11
p2 = 1.384e-08
p3 = -1.737e-06
p4 = 0.0001115
p5 = -0.003429
p6 = 4.218
Cp = p1*T**5 + p2*T**4 + p3*T**3 + p4*T**2 + p5*T + p6
elif liquid == "a":
# The specifc heat of air does not change over the temperature region of the system
Cp = 1.005
return Cp # Return the specific heat
def rho_a(T): # kg/m^3
p1 = 1.75e-05
p2 = -0.00483
p3 = 1.293
rho = p1*T**2 + p2*T + p3
return rho
def rho_w(T):
p1 = -0.003416
p2 = -0.09298
p3 = 1001
rho = p1*T**2 + p2*T + p3
return rho
def h(interface, T):
# This function returns the convection heat transfer coefficient for specific temperatures
# Interface, at which exchange is occuring. Options: w (water interior of pipe), pa (pipe to air), i (interior), e (exterior)
# L is the characteristic length, this changes based on which part of the system is being calculated
def k(liquid, T):
if liquid == "w":
# Equations and values acquired from "Standard Reference Data for the Thermal Conductivity of Water" by M.L.V. Ramires 1995
# Below equations accurate in temperature ranges from 274K to 370K
kAtRoom = 0.6065
dimensionless_k = -1.48445 + 4.12292*((T+274.15)/298.15) - 1.63866*((T+274.15)/298.15)**2
k = dimensionless_k * kAtRoom
elif liquid == "a":
# Constants determined from a 1 degree fit polynomial using MatLab CurveFit Toolbox
# R = 1
p1 = 7e-05
p2 = 0.0243
k = p1*T + p2
return k # Return the specific heat
def viscosity_a(T):
p1 = 7.5e-05
p2 = 0.0888
p3 = 13.3
viscosity = (p1*T**2 + p2*T + p3)*10**(-6)
return viscosity
def Pr_a(T):
p1 = -4.705e-19
p2 = -0.0001
p3 = 0.715
Pr = p1*T**2 + p2*T + p3
return Pr
if mode == 'forced':
# Characteristic length discussion: http://physics.stackexchange.com/questions/44185/what-is-the-characteristic-length-of-a-cylinder
if interface == 'w':
k = k('w', T)
L = diameter_inside # Diameter *NOTE* Check this number again had 0.3 before, that doesn't make sense because it should be milimeters
Nu = 3.66 # Nu_D for Reynold numbers < 2300 (laminar) and constant wall temperature. Could use Nu_D = 4.36 for constant heat transfer.
elif interface == 'pa':
k = k('a', T)
L = tubing_length * m # Length of system
Re = v_a * L / viscosity_a(T) # Re = velocity*Desnity*Length/viscosity
Nu = 0.037*Re**(4.0/5.0)*Pr_a(T)**(1.0/3.0)
elif interface == 'i':
k = k('a', T)
L = tubing_length * m
Re = v_a * L / viscosity_a(T)
Nu = 0.0296*Re**(4.0/5.0)*Pr_a(T)**(1.0/3.0)
elif interface == 'e':
k = k('a', T)
L = tubing_length * m
Re = v_a * L / viscosity_a(T)
Nu = 0.0296*Re**(4.0/5.0)*Pr_a(T)**(1.0/3.0)
h = k / L * Nu # L the characteristic length that should match the Nu L term
return h
def R(interface):
if interface == "pipe": # water to air
# Resistance = Convenction_interior + Silicone Pipe + Insulation + Convection_exterior
# A is cross sectional area
R.R_conv_wt = 1.0/(2.0 * math.pi * diameter_inside * tubing_length * h('w', (T[0]+T[m*4])/2.0 )) # Convection from water to tubing
R.R_cond_t = math.log((diameter_tubing)/(diameter_inside)) / (2 * math.pi * diameter_tubing * tubing_length * cond_tubing) # Conduction through tubing
R.R_cond_ins = math.log((diameter_insulation)/(diameter_tubing)) / (2.0 * math.pi * diameter_insulation * tubing_length * cond_insulation) # Conduction through insulation
R.R_conv_ta = 1.0/(2 * math.pi * diameter_insulation * tubing_length * h('pa',(T[0+1]+T[m*4+1])/2+(T[0+1]+T[m*4+1])/2)) # Convenction from tubing to air
Resistance = R.R_conv_wt + R.R_cond_t + R.R_cond_ins + R.R_conv_ta
if interface == 'int': # Resistance of the interior wall
# L/kA = thickness/(conductivity*Area)
# Double Layer Window with Argon Gap
# Resistance = First Layer of Glass + Argon Gap + Second Layer of Glass
Interior_Convection = 1/(A_wall * h('i',(((T[0+1]+T[m*4+1])/2)+T_int)/2))
FirstLayer = (6*10**(-3))/(1.05 * A_wall) # First layer = first layer of glass
SecondLayer = (6*10**-3)/(0.016 * A_wall) # Second layer = argon gap
ThirdLayer = (6*10**-3)/(1.05 * A_wall) # Third layer = second layer of glass
Resistance = FirstLayer + SecondLayer + ThirdLayer + Interior_Convection
if interface == 'ext': # Resistance of the exterior wall
# L/kA = thickness/(conductivity*area)
# Single layer exterior optically transparent glass layer for ideal optical performance on ICSolar
Interior_Convection = 1/(A_wall * h('e',(((T[0+1]+T[m*4+1])/2)+T_ext)/2))
FirstLayer = (6*10**-3)/(1.05 * A_wall) # First layer = first layer of glass
Resistance = FirstLayer + Interior_Convection
return Resistance
def GenTemperatureArray(m, T_wi):
# This function creates an array of temperature values (Even is air, Odd is water)
# The values are an initial guess for use in the residue function
count = 0
T = numpy.empty(m*4+2, dtype=float)
T_w_tmp = T_wi # Temperary value for the water
T_a_tmp = T_ai # Temperary value for the water
while count < 4*m+2:
if count%2 == 0:
T_tmp = T_w_tmp
T[count] = T_tmp # Add the first air temperature to the array
T_w_tmp += 2 # Increase each air temperature by 5 degrees
else:
T_tmp = T_a_tmp
T[count] = T_tmp # Add the first water temeperature to the end of the array
T_a_tmp += 1 # Increase each water temperature by 5 degrees
count += 1
return T
# # print T
def Residue(m,T):
# Calculates the temperature balance of each region of the system
# Inputs are temperature and number of modules
# Calculated values are the inbalance in the heat equations
# Inbalance will be solved using Newton's Iteration in a further step
q = numpy.empty(m*4, dtype=float)
j = 0
for i in range(0,m):
# print "Running module %d" %(i+1)
# Odd region (transfer between water, air, interior, and exterior)
# Calculate water temperature for region 1 (Transfer with air)
q_w1 = m_w * Cp('w', ((T[j+2]+T[j])/2)*(T[j+2]-T[j])) - ((T[j+3]-T[j+2]) / R('pipe'))
q[j] = q_w1
# Calculate air temperature for region 1 (Transfer with water, interior and exterior)
q_a1 = m_a * Cp('a',(T[j+3]+T[j+1])/2) * (T[j+3]-T[j+1]) + ((T[j+3]-T[j+2]) / R('pipe')) - (T_int-T[j+3])/R('int') - (T_ext-T[j+3])/R('ext')
q[j+1] = q_a1
# Even Region (Heat input)
q_w2 = m_w * Cp('w',(T[j+4]+T[j+2])/2) * (T[j+4]-T[j+2]) - q_receiver
q[j+2] = q_w2
q_a2 = m_a * Cp('a',(T[j+5]+T[j+3])/2) * (T[j+5]-T[j+3]) - q_module
q[j+3] = q_a2
j = j + 4
return (q)
def Differentiation():
dRdT = numpy.zeros((m*4,m*4), dtype=float)
j = 0
for i in range(0,m):
##### Region 1: Heat Balance in the Water #####
# q_w1 = m_w * Cp('w', ((T[j+2]+T[j])/2)*(T[j+2]-T[j])) - ((T[j+3]-T[j+2]) / R('pipe'))
# Derivative of R1 (water balance) with respect to T[j] (T[0])
dR1dT0_w = - m_w * Cp('w', (T[j+2]+T[j])/2)
if j == 0:
#Do nothing
pass
else:
dRdT[j,j-2] = dR1dT0_w
# Derivative of R1 (water balance) with respect to T[j+1] (T[1])
dR1dT1_w = 0
if j == 0:
#Do nothing
pass
else:
dRdT[j,j-1] = dR1dT1_w
# Derivative of R1 (water balance) with respect to T[j+2] (T[2])
dR1dT2_w = m_w * Cp('w', ((T[j+2]+T[j])/2)) + 1.0/R('pipe')
dRdT[j,j] = dR1dT2_w
# Derivative of R1 (water balance) with respect to T[j+3] (T[3])
dR1dT3_w = - 1.0/R('pipe')
dRdT[j,j+1] = dR1dT3_w
##### Region 1: Heat Balance in the Air #####
# q_a1 = m_a * Cp('a',(T[j+3]+T[j+1])/2) * (T[j+3]-T[j+1]) + ((T[j+3]-T[j+2]) / R('pipe')) - (T_int-T[j+3])/R('int') - (T_ext-T[j+3])/R('ext')
# Derivative of R1 (air balance) with respect to T[j] (T[0])
dR1dT0_a = 0
if j == 0:
#Do nothing
pass
else:
dRdT[j+1,j-2] = dR1dT0_a
# Derivative of R1 (air balance) with respect to T[j+1] (T[1])
dR1dT1_a = - m_a * Cp('a',(T[j+3]+T[j+1])/2)
if j == 0:
#Do nothing
pass
else:
dRdT[j+1,j-1] = dR1dT1_a
# Derivative of R1 (air balance) with respect to T[j+2] (T[2])
dR1dT2_a = -1.0/R('pipe')
dRdT[j+1,j] = dR1dT2_a
# Derivative of R1 (air balance) with respect to T[j+3] (T[3])
dR1dT3_a = m_a * Cp('a',(T[j+3]+T[j+1])/2.0) + 1.0/R('pipe') + 1.0/R('int') + 1.0/R('ext')
dRdT[j+1,j+1] = dR1dT3_a
##### Region 2: Heat Balance in the Water #####
# q_w2 = m_w * Cp('w',(T[j+4]+T[j+2])/2) * (T[j+4]-T[j+2]) + q_receiver
# Derivative of R2 (water balance) with respect to T[j] (T[0])
dR2dT0_w = 0
# Derivative of R2 (water balance) with respect to T[j+1] (T[1])
dR2dT1_w = 0
# Derivative of R2 (water balance) with respect to T[j+2] (T[2])
dR2dT2_w = - m_w * Cp('w',(T[j+4]+T[j+2])/2)
dRdT[j+2,j] = dR2dT2_w
# Derivative of R2 (water balance) with respect to T[j+3] (T[3])
dR2dT3_w = 0
# Derivative of R2 (water balance) with respect to T[j+4] (T[4])
dR2dT4_w = m_w * Cp('w',(T[j+4]+T[j+2])/2)
dRdT[j+2,j+2] = dR2dT4_w
# Derivative of R2 (water balance) with respect to T[j+5] (T[5])
dR2dT5_w = 0
##### Region 2: Heat Balance in the Air #####
# q_a2 = m_a * Cp('a',(T[j+5]+T[j+3])/2) * (T[j+5]-T[j+3]) + q_module
# Derivative of R2 (air balance) with respect to T[j] (T[0])
dR2dT0_a = 0
# Outside of Matrix
# Derivative of R2 (air balance) with respect to T[j+1] (T[1])
dR2dT1_a = 0
# Outside of Matrix
# Derivative of R2 (air balance) with respect to T[j+2] (T[2])
dR2dT2_a = 0
# Derivative of R2 (air balance) with respect to T[j+3] (T[3])
dR2dT3_a = - m_a * Cp('a',(T[j+5]+T[j+3])/2)
dRdT[j+3,j+1] = dR2dT3_a
# Derivative of R2 (air balance) with respect to T[j+4] (T[4])
dR2dT4_a = 0
# Derivative of R2 (air balance) with respect to T[j+5] (T[5])
dR2dT5_a = m_a * Cp('a',(T[j+5]+T[j+3])/2)
dRdT[j+3,j+3] = dR2dT5_a
j = j + 4
return dRdT
def UpdatedTemp(T):
def Increment(dQ, q):
# Using Newton Method
# Increment = q/dQ = dQ_inv * q
# T_new = T_old - dQ_inv * q = T_old - q/dQ
q_matrix = numpy.matrix(q)
dQ_inv = numpy.linalg.inv(dQ)
Increment = dQ_inv * numpy.transpose(q_matrix)
Increment = numpy.squeeze(numpy.asarray(Increment))
return Increment
# print "Inverse Derivative Matrix"
# print numpy.linalg.inv(dQ)
# print
Inc = Increment(dQ, q)
# print "This is the initial increment"
# print Inc
# print
T_new = T[2:]
count = 1
#while count < 2: # abs(numpy.amax(Inc)) > 10**(-4):
T_new = T_new - Inc
T_local = numpy.concatenate([T[:2], T_new])
# print "This is the temperature after %d increment" %(count)
# print T_local
print
q_new = Residue(m,T_local)
# print "Updating heat balance with new temperature"
# print q_new
print
Inc = Increment(dQ, q_new)
# print "Updated increment"
# print Inc
print
count =+ 1
# time.sleep(5)
T_final = numpy.concatenate([T[:2], T_new])
return T_final
# Before now all code is functions
# Code starts here to
m = 6 # Number of Modules
inlet = numpy.recfromcsv('jan15.csv',
delimiter=',')
out_filename = 'outlet_jan15.txt'
# numpy.loadtxt(out_filename)
outlet_T_water = numpy.empty(len(inlet['timestamp']), dtype=float)
T_int = 22.5 # Temperature on the interior of the building, degrees [C]
T_ext = 25.0 # Temperature on the exterior of the building, degrees [C]
T_ai = 20.0 # Inlet temperature of air, degrees [C]
# T_wi = specified below from input file
diameter_inside = 3.0*10**(-3)
diameter_tubing = diameter_inside + 1.675*10**(-3)
ins_thickness = 9.525*10**(-3) # Thickness of the insulation around the silicone pipes
diameter_insulation = diameter_tubing + ins_thickness
As_pipe = 2 * math.pi * (diameter_insulation) * 0.3 # Area m^2
cond_tubing = 0.145 # Conduction value of silicon tubing
cond_insulation = 0.037 # Conduction value of silicon insulation
Height_wall = 3
Width_wall = 0.3
tubing_length = 0.3
A_wall = Height_wall * Width_wall # Area of glass
P_wall = 2*Height_wall + 2*Width_wall # Perimeter of the wall
radius_cavity = 4 * 0.3*0.3 / (0.3*4) / 2 # Hydraulic diameter assuming cavity is 0.3 meters by 0.3 m (cross section)
count = 0
while count < len(inlet['timestamp']):
T_wi = inlet['exp_inlet'][count] # Temperature on the interior of the building, degrees [C]
Water_flowrate = inlet['exp_water_flowrate'][count]
T = GenTemperatureArray(m,T_wi)
# print "Initial Temperature Array Guess"
# print T
# print
q_receiver = (inlet['exp_heatgen'][count])*10**(-3) / m # 8.0*10**(-3) # Heat flow into water from Module Heat Receiver
q_module = 3.0*10**(-3) # Heat flow into air from Heat Loss from the Module
mode = 'forced' # Defines whether the air flow is 'forced' or 'natural'
m_w = Water_flowrate*10**(-7) * rho_w((T[0]+T[m*4])/2) # Mass flowrate of water = VolumetricFlowrate * DensityWater
v_a = 0.5 # Flow velocity [m/s]
m_a = v_a * rho_a((13+30)/2) * radius_cavity # Mass Flowrate of air [kg/s] = velocity [m/s] * DensityAir [kg/m^3] * cross section [m^2]
q = Residue(m,T)
# print "Residual heat balance array:"
# print q
# print
dQ = Differentiation()
# print "Derivative Matrix:"
# print dQ
# print
T_final = UpdatedTemp(T)
# print "Final Temperatures"
print T_final
outlet_T_water[count] = T_final[len(T_final)-2]
print count
count = count + 1
# timestamp = inlet['timestamp'].astye(float)
# plt.plot(inlet['timestamp'], inlet['tc_b2s3m4_outlet'], 'r--', inlet['timestamp'], outlet_T_water, 'bs')
# plt.show()
numpy.savetxt(out_filename, outlet_T_water, delimiter=',', newline='\n', header='', footer='', comments='# ')