Results for
Untapped Potential for Output-arguments Block
MATLAB has a very powerful feature in its arguments blocks. For example, the following code for a function (or method):
- clearly outlines all the possible inputs
- provides default values for each input
- will produce auto-complete suggestions while typing in the Editor (and Command Window in newer versions)
- checks each input against validation functions to enforce size, shape (e.g., column vs. row vector), type, and other options (e.g., being a member of a set)
function [out] = sample_fcn(in)
	arguments(Input)
		in.x (:, 1) = []
		in.model_type (1, 1) string {mustBeMember(in.model_type, ...
			["2-factor", "3-factor", "4-factor"])} = "2-factor"
		in.number_of_terms (1, 1) {mustBeMember(in.number_of_terms, 1:5)} = 1
		in.normalize_fit (1, 1) logical = false
	end
	% function logic ...
end
If you do not already use the arguments block for function (or method) inputs, I strongly suggest that you try it out.
The point of this post, though, is to suggest improvements for the output-arguments block, as it is not nearly as powerful as its input-arguments counterpart. I have included two function examples: the first can work in MATLAB while the second does not, as it includes suggestions for improvements. Commentary specific to each function is provided completely before the code. While this does necessitate navigating back and forth between functions and text, this provides for an easy comparison between the two functions which is my main goal.
Current Implementation
The input-arguments block for sample_fcn begins the function and has already been discussed. A simple output-arguments block is also included. I like to use a single output so that additional fields may be added at a later point. Using this approach simplifies future development, as the function signature, wherever it may be used, does not need to be changed. I can simply add another output field within the function and refer to that additional field wherever the function output is used.
Before beginning any logic, sample_fcn first assigns default values to four fields of out. This is a simple and concise way to ensure that the function will not error when returning early.
The function then performs two checks. The first is for an empty input (x) vector. If that is the case, nothing needs to be done, as the function simply returns early with the default output values that happen to apply to the inability to fit any data.
The second check is for edge cases for which input combinations do not work. In this case, the status is updated, but default values for all other output fields (which are already assigned) still apply, so no additional code is needed.
Then, the function performs the fit based on the specified model_type. Note that an otherwise case is not needed here, since the argument validation for model_type would not allow any other value.
At this point, the total_error is calculated and a check is then made to determine if it is valid. If not, the function again returns early with another specific status value.
Finally, the R^2 value is calculated and a fourth check is performed. If this one fails, another status value is assigned with an early return.
If the function has passed all the checks, then a set of assertions ensure that each of the output fields are valid. In this case, there are eight specific checks, two for each field.
If all of the assertions also pass, then the final (successful) status is assigned and the function returns normally.
function [out] = sample_fcn(in)
	arguments(Input)
		in.x (:, 1) = []
		in.model_type (1, 1) string {mustBeMember(in.model_type, ...
			["2-factor", "3-factor", "4-factor"])} = "2-factor"
		in.number_of_terms (1, 1) {mustBeMember(in.number_of_terms, 1:5)} = 1
		in.normalize_fit (1, 1) logical = false
	end
	arguments(Output)
		out struct
	end
	%% 
	out.fit = [];
	out.total_error = [];
	out.R_squared = NaN;
	out.status = "Fit not possible for supplied inputs.";
	%% 
	if isempty(in.x)
		return
	end
	%% 
	if ((in.model_type == "2-factor") && (in.number_of_terms == 5)) || ... % other possible logic
		out.status = "Specified combination of model_type and number_of_terms is not supported.";
		return
	end
	%%
	switch in.model_type
		case "2-factor"
			out.fit = % code for 2-factor fit
		case "3-factor"
			out.fit = % code for 3-factor fit
		case "4-factor"
			out.fit = % code for 4-factor fit
	end
	%%
	out.total_error = % calculation of error
	if ~isfinite(out.total_error)
		out.status = "The total_error could not be calculated.";
		return
	end
	%%
	out.R_squared = % calculation of R^2
	if out.R_squared > 1
		out.status = "The R^2 value is out of bounds.";
		return
	end
	%% 
	assert(iscolumn(out.fit), "The fit vector is not a column vector.");
	assert(size(out.fit) == size(in.x), "The fit vector is not the same size as the input x vector.");
	assert(isscalar(out.total_error), "The total_error is not a scalar.");
	assert(isfinite(out.total_error), "The total_error is not finite.");
	assert(isscalar(out.R_squared), "The R^2 value is not a scalar.");
	assert(isfinite(out.R_squared), "The R^2 value is not finite.");
	assert(isscalar(out.status), "The status is not a scalar.");
	assert(isstring(out.status), "The status is not a string.");
	%% 
	out.status = "The fit was successful.";
end
Potential Implementation
The second function, sample_fcn_output_arguments, provides essentially the same functionality in about half the lines of code. It is also much clearer with respect to the output. As a reminder, this function structure does not currently work in MATLAB, but hopefully it will in the not-too-distant future.
This function uses the same input-arguments block, which is then followed by a comparable output-arguments block. The first unsupported feature here is the use of name-value pairs for outputs. I would much prefer to make these assignments here rather than immediately after the block as in the sample_fcn above, which necessitates four more lines of code.
The mustBeSameSize validation function that I use for fit does not exist, but I really think it should; I would use it a lot. In this case, it provides a very succinct way of ensuring that the function logic did not alter the size of the fit vector from what is expected.
The mustBeFinite validation function for out.total_error does not work here simply because of the limitation on name-value pairs; it does work for regular outputs.
Finally, the assignment of default values to output arguments is not supported.
The next three sections of sample_fcn_output_arguments match those of sample_fcn: check if x is empty, check input combinations, and perform fit logic. Following that, though, the functions diverge heavily, as you might expect. The two checks for total_error and R^2 are not necessary, as those are covered by the output-arguments block. While there is a slight difference, in that the specific status values I assigned in sample_fcn are not possible, I would much prefer to localize all these checks in the arguments block, as is already done for input arguments.
Furthermore, the entire section of eight assertions in sample_fcn is removed, as, again, that would be covered by the output-arguments block.
This function ends with the same status assignment. Again, this is not exactly the same as in sample_fcn, since any failed assertion would prevent that assignment. However, that would also halt execution, so it is a moot point.
function [out] = sample_fcn_output_arguments(in)
	arguments(Input)
		in.x (:, 1) = []
		in.model_type (1, 1) string {mustBeMember(in.model_type, ...
			["2-factor", "3-factor", "4-factor"])} = "2-factor"
		in.number_of_terms (1, 1) {mustBeMember(in.number_of_terms, 1:5)} = 1
		in.normalize_fit (1, 1) logical = false
	end
	arguments(Output)
		out.fit (:, 1) {mustBeSameSize(out.fit, in.x)} = []
		out.total_error (1, 1) {mustBeFinite(out.total_error)} = []
		out.R_squared (1, 1) {mustBeLessThanOrEqual(out.R_squared, 1)} = NaN
		out.status (1, 1) string = "Fit not possible for supplied inputs."
	end
	%% 
	if isempty(in.x)
		return
	end
	%% 
	if ((in.model_type == "2-factor") && (in.number_of_terms == 5)) || ... % other possible logic
		out.status = "Specified combination of model_type and number_of_terms is not supported.";
		return
	end
	%%
	switch in.model_type
		case "2-factor"
			out.fit = % code for 2-factor fit
		case "3-factor"
			out.fit = % code for 3-factor fit
		case "4-factor"
			out.fit = % code for 4-factor fit
	end
	%%
	out.status = "The fit was successful.";
end
Final Thoughts
There is a significant amount of unrealized potential for the output-arguments block. Hopefully what I have provided is helpful for continued developments in this area.
What are your thoughts? How would you improve arguments blocks for outputs (or inputs)? If you do not already use them, I hope that you start to now.
                    Too small
                
 
                
                    22%
                
  
            
                    Just right
                
 
                
                    38%
                
  
            
                    Too large
                
 
                
                    40%
                
  
            
            2648 votes
        
    
                    Always!
                
 
                
                    29%
                
  
            
                    It depends
                
 
                
                    14%
                
  
            
                    Never!
                
 
                
                    21%
                
  
            
                    I didn't know that was possible
                
 
                
                    36%
                
  
            
            1810 votes
        
    Base case:
Suppose you need to do a computation many times. We are going to assume that this computation cannot be vectorized. The simplest case is to use a for loop:
number_of_elements = 1e6;
test_fcn = @(x) sqrt(x) / x;
tic
for i = 1:number_of_elements
    x(i) = test_fcn(i);
end
t_forward = toc;
disp(t_forward + " seconds")
Preallocation:
This can easily be sped up by preallocating the variable that houses results:
tic
x = zeros(number_of_elements, 1);
for i = 1:number_of_elements
    x(i) = test_fcn(i);
end
t_forward_prealloc = toc;
disp(t_forward_prealloc + " seconds")
In this example, preallocation speeds up the loop by a factor of about three to four (running in R2024a). Comment below if you get dramatically different results.
disp(sprintf("%.1f", t_forward / t_forward_prealloc))
Run it in reverse:
Is there a way to skip the explicit preallocation and still be fast? Indeed, there is. 
clear x
tic
for i = number_of_elements:-1:1
    x(i) = test_fcn(i);
end
t_backward = toc;
disp(t_backward + " seconds")
By running the loop backwards, the preallocation is implicitly performed during the first iteration and the loop runs in about the same time (within statistical noise):
disp(sprintf("%.2f", t_forward_prealloc / t_backward))
Do you get similar results when running this code? Let us know your thoughts in the comments below.
Beneficial side effect:
Have you ever had to use a for loop to delete elements from a vector? If so, keeping track of index offsets can be tricky, as deleting any element shifts all those that come after. By running the for loop in reverse, you don't need to worry about index offsets while deleting elements.
                    quick / easy
                
 
                
                    21%
                
  
            
                    themed / in a group
                
 
                
                    20%
                
  
            
                    challenge (e.g., banned functions)
                
 
                
                    13%
                
  
            
                    puzzle / game
                
 
                
                    16%
                
  
            
                    educational
                
 
                
                    28%
                
  
            
                    other (comment below)
                
 
                
                    3%
                
  
            
            117 votes
        
    
                    isstring
                
 
                
                    11%
                
  
            
                    ischar
                
 
                
                    7%
                
  
            
                    iscellstr
                
 
                
                    13%
                
  
            
                    isletter
                
 
                
                    21%
                
  
            
                    isspace
                
 
                
                    9%
                
  
            
                    ispunctuation
                
 
                
                    37%
                
  
            
            2455 votes
        
    
                    Don't use / What are Projects?
                
 
                
                    26%
                
  
            
                    1–10
                
 
                
                    31%
                
  
            
                    11–20
                
 
                
                    15%
                
  
            
                    21–30
                
 
                
                    9%
                
  
            
                    31–50
                
 
                
                    7%
                
  
            
                    51+ (comment below)
                
 
                
                    12%
                
  
            
            4070 votes
        
    
                    2
                
 
                
                    17%
                
  
            
                    3
                
 
                
                    12%
                
  
            
                    4
                
 
                
                    59%
                
  
            
                    6
                
 
                
                    4%
                
  
            
                    8
                
 
                
                    5%
                
  
            
                    Other (comment below)
                
 
                
                    3%
                
  
            
            6419 votes
        
    
                    numel(v)
                
 
                
                    6%
                
  
            
                    length(v)
                
 
                
                    13%
                
  
            
                    width(v)
                
 
                
                    14%
                
  
            
                    nnz(v)
                
 
                
                    8%
                
  
            
                    size(v, 1)
                
 
                
                    27%
                
  
            
                    sum(v > 0)
                
 
                
                    31%
                
  
            
            2537 votes
        
    
                    ismissing( { [ ] } )
                
 
                
                    26%
                
  
            
                    ismissing( NaN )
                
 
                
                    18%
                
  
            
                    ismissing( NaT )
                
 
                
                    11%
                
  
            
                    ismissing( missing )
                
 
                
                    21%
                
  
            
                    ismissing( categorical(missing) )
                
 
                
                    9%
                
  
            
                    ismissing( { '' } ) % 2 apostrophes
                
 
                
                    16%
                
  
            
            896 votes
        
    
                    isempty( [ ] )
                
 
                
                    10%
                
  
            
                    isempty( { } )
                
 
                
                    13%
                
  
            
                    isempty( '' ) % 2 single quotes
                
 
                
                    13%
                
  
            
                    isempty( "" ) % 2 double quotes
                
 
                
                    24%
                
  
            
                    c = categorical( [ ] ); isempty(c)
                
 
                
                    18%
                
  
            
                    s = struct("a", [ ] ); isempty(s.a)
                
 
                
                    22%
                
  
            
            1324 votes
        
    
                    sort(v)
                
 
                
                    8%
                
  
            
                    unique(v)
                
 
                
                    16%
                
  
            
                    union(v, [ ])
                
 
                
                    17%
                
  
            
                    intersect(v, v)
                
 
                
                    14%
                
  
            
                    setdiff(v, [ ])
                
 
                
                    12%
                
  
            
                    All return sorted output
                
 
                
                    33%
                
  
            
            1193 votes
        
    
                    s = ['M','A','T','L','A','B']
                
 
                
                    9%
                
  
            
                    char([77,65,84,76,65,66])
                
 
                
                    7%
                
  
            
                    "MAT" + "LAB"
                
 
                
                    21%
                
  
            
                    upper(char('matlab' - '0' + 48))
                
 
                
                    17%
                
  
            
                    fliplr("BALTAM")
                
 
                
                    17%
                
  
            
                    rot90(rot90('BALTAM'))
                
 
                
                    30%
                
  
            
            2929 votes
        
    
                    eye(3) - diag(ones(1,3))
                
 
                
                    11%
                
  
            
                    0 ./ ones(3)
                
 
                
                    9%
                
  
            
                    cos(repmat(pi/2, [3,3]))
                
 
                
                    16%
                
  
            
                    zeros(3)
                
 
                
                    20%
                
  
            
                    A(3, 3) = 0
                
 
                
                    32%
                
  
            
                    mtimes([1;1;0], [0,0,0])
                
 
                
                    12%
                
  
            
            3009 votes
        
    
                    1
                
 
                
                    33%
                
  
            
                    2
                
 
                
                    34%
                
  
            
                    3
                
 
                
                    18%
                
  
            
                    4
                
 
                
                    5%
                
  
            
                    5
                
 
                
                    3%
                
  
            
                    6+
                
 
                
                    6%
                
  
            
            1643 votes
        
    
                    <= 6 GB
                
 
                
                    10%
                
  
            
                    7–12 GB
                
 
                
                    26%
                
  
            
                    13–22 GB
                
 
                
                    34%
                
  
            
                    23–46 GB
                
 
                
                    19%
                
  
            
                    47–90 GB
                
 
                
                    6%
                
  
            
                    >= 91 GB
                
 
                
                    6%
                
  
            
            15925 votes
        
    
                    Yes, the available tools are great
                
 
                
                    12%
                
  
            
                    Yes, the available tools need help
                
 
                
                    6%
                
  
            
                    No, but I would like to
                
 
                
                    14%
                
  
            
                    No, it is not important to me
                
 
                
                    7%
                
  
            
                    What is test-driven development?
                
 
                
                    61%
                
  
            
            1955 votes
        
    
                    Always
                
 
                
                    12%
                
  
            
                    Sometimes
                
 
                
                    11%
                
  
            
                    In the past, but not now
                
 
                
                    3%
                
  
            
                    Never
                
 
                
                    20%
                
  
            
                    What is Simulink Project?
                
 
                
                    53%
                
  
            
            2443 votes
        
    
                    Always
                
 
                
                    8%
                
  
            
                    Sometimes
                
 
                
                    9%
                
  
            
                    In the past, but not now
                
 
                
                    2%
                
  
            
                    Never
                
 
                
                    23%
                
  
            
                    What is MATLAB Project?
                
 
                
                    58%
                
  
            
            4533 votes
        
    
                    A bunch of quick, simple problems
                
 
                
                    16%
                
  
            
                    A structured, educational set
                
 
                
                    15%
                
  
            
                    Brain-teasers
                
 
                
                    16%
                
  
            
                    Random miscellany
                
 
                
                    3%
                
  
            
                    Something else
                
 
                
                    2%
                
  
            
                    What are Problem Groups?
                
 
                
                    49%
                
  
            
            1777 votes
        
    