forked from sandialabs/MATLAB_PV_LIB
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathpvl_est_Rs_IEC60891_2.m
126 lines (105 loc) · 4.6 KB
/
pvl_est_Rs_IEC60891_2.m
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
function [Rs, a] = pvl_est_Rs_IEC60891_2(IVCurves, aIsc, bVoc)
% PVL_EST_RS_IEC60891_2 estimates Rs using method 2 in IEC 60891.
%
% Syntax
% [Rs, a] = pvl_est_Rs_IEC60891_1(IVCurves, aIsc, bVoc)
%
% Description
% Method 2 in IEC60891 translates a point on an IV curve measured at
% irradiance G1 and cell temperature Tc1 to the corresponding point on an
% unobserved IV curve at irradiance G2 and cell temperature Tc2. The
% translation reduces the voltage value by a voltage drop across the
% ‘internal series resistance’ of the module ([IEC 60891], Eq. 2).
% The resistance Rs is found by an iterative search over a set of IV
% curves measured at different irradiance levels and constant cell
% temperature: first, current for each IV curve is translated linearly
% to a common irradiance, then a fitting parameter is found by translating
% Voc to a common value, and finally Rs is found by minimizing the
% variance of Pmp of the translated IV curves. Method 2 differs from
% Method 1 by allowing greater variation in irradiance among the IV
% curves.
%
% pvl_est_Rs_IEC60891_2 assumes that the IV curves are measured at, or
% have been translated to a common cell temperature.
%
% Input:
% IVCurves - A structure array with the following fields:
% IVCurves.Isc - short circuit current in amperes.
% IVCurves.Voc - open circuit voltage in volts.
% IVCurves.Imp - current at maximum power point in amperes.
% IVCurves.Vmp - voltage at maximum power point in volts.
% IVCurves.Pmp - power at maximum power point in watts.
% IVCurves.V - vector of voltage in volts.
% IVCurves.I - vector of current in amperes.
% IVCurves.Ee - Effective irradiance (W/m2)
% IVCurves.Tc - cell temperature (C)
% aIsc - relative temperature coefficient for short circuit current
% in 1/C (not A/C).
% bVoc - relative temperature coefficient for open circuit voltage
% in 1/C (not V/C).
%
% Output:
% Rs - the series resistance value in ohms.
% a - curve correction factor related to the dependence of Voc on the
% logarithm of irradiance, unitless.
%
% References
% [1] IEC60891 Ed. 2 2009. Procedures for temperature and irradiance
% corrections to measured I-V characteristics of crystalline silicon
% photovoltaic (PV) devices.
%
% [2] C. Hansen and B. King, "Determining series resistance for
% equivalent circuit models of a PV module", in 45th IEEE Photovoltaic
% Specialist Conference, Waikoloa, HI, 2018.
Tc = unique(round([IVCurves(:).Tc], 2));
G = max([IVCurves(:).Ee]);
if length(Tc)>1
warning('est_Rs_IEC60891_1: IV curves must have equal cell temperature')
Rs = NaN;
return
else
% figure
% hold all
% for i=1:length(IVCurves)
% plot(IVCurves(i).V, IVCurves(i).I)
% end
% title('Input IV curves')
% first find parameter a
objfun = @(a) max_Voc_diff(a, 0, IVCurves, G, Tc, aIsc, bVoc);
options = optimset('TolX', 1e-6);
a = fminsearch(objfun, -0.1, options);
['Estimated a : ' num2str(a) ', variance ' num2str(max_Voc_diff(a, 0, IVCurves, G, Tc, aIsc, bVoc))];
% figure
% hold all
% for i=1:length(IVCurves)
% res = translate_IV_curve_IEC60891_2(IVCurves(i), G, Tc, aIsc, bVoc, 0, 0, a);
% plot(res.V, res.I)
% end
% title('After translation by a')
% now find Rs using parameter a
objfun = @(Rs) max_Pmp_diff(Rs, a, IVCurves, G, Tc, aIsc, bVoc);
options = optimset('TolX', 1e-6);
Rs = fminsearch(objfun, 0.0, options);
['Estimated Rs : ' num2str(Rs) ', variance ' num2str(max_Pmp_diff(Rs, a, IVCurves, G, Tc, aIsc, bVoc))];
% figure
% hold all
% for i=1:length(IVCurves)
% res = translate_IV_curve_IEC60891_2(IVCurves(i), G, Tc, aIsc, bVoc, Rs, 0, a);
% plot(res.V, res.I)
% end
% title('After translation by a and Rs')
end
function val = max_Pmp_diff(Rs, a, IVCurves, G, Tc, aIsc, bVoc)
% calculates maximum absolute difference in Pmp among translated IVCurves
for i=1:length(IVCurves)
tmpIVCurves(i) = pvl_translate_IV_curve_IEC60891_2(IVCurves(i), G, Tc, aIsc, bVoc, Rs, 0, a);
end
Pmp = [tmpIVCurves(:).Pmp];
val = max(abs((Pmp - median(Pmp))./median(Pmp)));
function val = max_Voc_diff(a, Rs, IVCurves, G, Tc, aIsc, bVoc)
% calculates maximum absolute difference in Pmp among translated IVCurves
for i=1:length(IVCurves)
tmpIVCurves(i) = pvl_translate_IV_curve_IEC60891_2(IVCurves(i), G, Tc, aIsc, bVoc, Rs, 0, a);
end
Voc = [tmpIVCurves(:).Voc];
val = max(abs((Voc - median(Voc))./median(Voc)));