Dimension 1 is fixed on the left-hand side but varies on the right ([1 x 1] ~= [:? x 1]).

조회 수: 5 (최근 30일)
Federico Avino
Federico Avino 2021년 1월 26일
답변: Aneela 2025년 3월 10일
Good morning,
I'm writing a piece of code in a simulink block. Here the code
function [otv1, otv2, otv3] = TCU(P_tank1, P_tank2, P_tank3, time)
%% input
toll = 1; %[bar]
% coder.varsize('otv1');
% coder.varsize('otv2');
% coder.varsize('otv3');
persistent OTV1 OTV2 OTV3
if isempty(OTV1); OTV1 = 0; end
if isempty(OTV2); OTV2 = 0; end
if isempty(OTV3); OTV3 = 0; end
if time <= 0
OTV1 = 0;
OTV2 = 0;
OTV3 = 0;
else
Tank_unsorted =[P_tank1 OTV1
P_tank2 OTV2
P_tank3 OTV3];
% ordine decrescente di pressione
[Tank, index] = sortrows(Tank_unsorted, 'descend');
if (Tank(1,1) - Tank(2,1)) >= toll
Tank(1,2) = 1;
Tank(2,2) = 0;
Tank(3,2) = 0;
else if (Tank(1,1) - Tank(2,1)) < toll
Tank(1,2) = 1;
Tank(2,2) = 1;
Tank(3,2) = 0;
else if (Tank(2,1) - Tank(3,1)) <= toll
Tank(1,2) = 1;
Tank(2,2) = 1;
Tank(3,2) = 1;
end
end
end
end
index_otv = find(index == 1);
OTV1 = Tank(index_otv, 2);
index_otv = find(index == 2);
OTV2 = Tank(index_otv, 2);
index_otv = find(index == 3);
OTV3 = Tank(index_otv, 2);
%% output
otv1 = OTV1;
otv2 = OTV2;
otv3 = OTV3;
I tried the logic of the code in a script, and it works: OTVn in the end contains the correct scalar. But during the execution i have the error I wrote in the title at the lines
OTV1 = Tank(index_otv, 2);
and similar. But actually OTVn contains a scalar, because it's extracted from the matrix Tank.
I have tried the solutions on the net, but none of them worked (including coder.varsize). How could i overcome the problem? Considering the goal is to generate code

답변 (2개)

Infinite_king
Infinite_king 2024년 5월 10일
편집: Infinite_king 2024년 5월 10일
Hi Federico Avino,
The MATLAB Coder was unable to determine that the variable 'index_otv' would always be a scalar, leading to the error.
While analyzing the expression 'index_otv = find(index == 1)', MATLAB Coder considers the type and size of the variable 'index' but does not account for the values stored in the variable. Since 'index' can contain any values, MATLAB Coder concludes that the function call 'find(index == 1)' may return a vector.
To resolve the issue, you can first store the result in a temporary variable, 'tOTV,' and then extract the value at index 1 and store it in the variable 'OTV'.
Please find the updated code below,
function [otv1, otv2, otv3] = TCU(P_tank1, P_tank2, P_tank3, time) %#codegen
%% input
toll = 1; %[bar]
% coder.varsize('otv1');
% coder.varsize('otv2');
% coder.varsize('otv3');
persistent OTV1 OTV2 OTV3
if isempty(OTV1); OTV1 = 0; end
if isempty(OTV2); OTV2 = 0; end
if isempty(OTV3); OTV3 = 0; end
if time <= 0
OTV1 = 0;
OTV2 = 0;
OTV3 = 0;
else
Tank_unsorted =[P_tank1 OTV1
P_tank2 OTV2
P_tank3 OTV3];
% ordine decrescente di pressione
[Tank, index] = sortrows(Tank_unsorted, 'descend');
if (Tank(1,1) - Tank(2,1)) >= toll
Tank(1,2) = 1;
Tank(2,2) = 0;
Tank(3,2) = 0;
else if (Tank(1,1) - Tank(2,1)) < toll
Tank(1,2) = 1;
Tank(2,2) = 1;
Tank(3,2) = 0;
else if (Tank(2,1) - Tank(3,1)) <= toll
Tank(1,2) = 1;
Tank(2,2) = 1;
Tank(3,2) = 1;
end
end
end
%------------------------------------------------------------------------------------%
% Change 1 - Moved the code inside the for loop.
% Change 2 - Values are stored in temporary variables
% and then transferred to main variables.
index_otv = find(index == 1);
tOTV1 = Tank(index_otv, 2);
index_otv = find(index == 2);
tOTV2 = Tank(index_otv, 2);
index_otv = find(index == 3);
tOTV3 = Tank(index_otv, 2);
OTV1 = tOTV1(1);
OTV2 = tOTV2(1);
OTV3 = tOTV3(1);
%------------------------------------------------------------------------------------%
end
%% output
otv1 = OTV1;
otv2 = OTV2;
otv3 = OTV3;

Aneela
Aneela 2025년 3월 10일
Assuming that you are using MATLAB Coder for code generation, it is essential to ensure that all variables have a fixed size in this scenario, as all the "OTVn" variables are scalars, as you mentioned.
The error arises due to the "find" function which introduces ambiguity as it returns all indices that meet a condition, potentially resulting in a vector rather than a scalar. This poses a problem for code generation as MATLAB coder cannot guarantee that "index_otv" will alwayas be a scalar.
To resolve the issue, consider implementing the following approaches:
1) Temporary Variables: By storing the result of "find" in a temporary variable and explicitly selecting the first element, it can be ensured that the final assignment to "OTVn" is always a scalar.
index_otv = find(index == 1);
tOTV1 = Tank(index_otv, 2);
OTV1 = tOTV1(1); % Ensure OTVn are scalars by taking the first element
2) Preallocation: Explicitly preallocate the variables as scalars you know that the condition results in a single true value. This ensures that their size is fixed from the start, which is beneficial for performance and clarity.
OTV1 = 0;
OTV2 = 0;
OTV3 = 0;
Please refer to the following MathWorks documentation for more information on variable-sized and fixed-sized arrays in Code Generation:

카테고리

Help CenterFile Exchange에서 Sources에 대해 자세히 알아보기

제품


릴리스

R2020b

Community Treasure Hunt

Find the treasures in MATLAB Central and discover how the community can help you!

Start Hunting!

Translated by