it-roy-ru.com

Как мне сделать множественное назначение в MATLAB?

Вот пример того, что я ищу:

>> foo = [88, 12];
>> [x, y] = foo;

Я бы ожидал что-то вроде этого потом:

>> x

x =

    88

>> y

y =

    12

Но вместо этого я получаю ошибки, такие как:

??? Too many output arguments.

Я думал, что deal() может сделать это, но, похоже, работает только на ячейках.

>> [x, y] = deal(foo{:});
??? Cell contents reference from a non-cell array object.

Как мне решить мою проблему? Должен ли я постоянно индексировать 1 и 2, если я хочу иметь дело с ними отдельно?

43
Benjamin Oakes

Вам вообще не нужен deal (отредактируйте: для Matlab 7.0 или новее), и, для вашего примера, вам не нужен mat2cell; Вы можете использовать num2cell без других аргументов ::

foo = [88, 12];
fooCell = num2cell(foo);
[x y]=fooCell{:}

x =

    88


y =

    12

Если вы хотите использовать deal по какой-либо другой причине, вы можете:

foo = [88, 12];
fooCell = num2cell(foo);
[x y]=deal(fooCell{:})

x =

    88


y =

    12
42
Ramashalanka

Обратите внимание, что deal принимает «список» в качестве аргумента, а не массив ячеек. Таким образом, следующее работает как ожидалось:

> [x,y] = deal(88,12)
x = 88

y = 12

Синтаксис c{:} преобразует массив ячеек в список, а список представляет собой значения, разделенные запятыми , как в аргументах функции. Это означает, что вы можете использовать синтаксис c{:} в качестве аргумента для других функций, кроме deal. Чтобы увидеть это, попробуйте следующее:

> z = plus(1,2)
z = 3

> c = {1,2};
> z = plus(c{:});
z = 3
16
jullybobble

Чтобы использовать решение num2cell в одной строке, определите вспомогательную функцию list:

function varargout = list(x)
% return matrix elements as separate output arguments
% example: [a1,a2,a3,a4] = list(1:4)

varargout = num2cell(x);

end
8
Sander Evers

Что MTRW сказал. По сути, вы хотите использовать сделку с массивом ячеек (хотя сделка (88,12) также работает).

Предполагая, что вы начинаете с массива foo n-by-2 и хотите присвоить первому столбцу значение x, а второму значение y, вы сделаете следующее:

foo = [88,12;89,13;90,14];
%# divide the columns of foo into separate cells, i.e. do mat2cell(foo,3,[1,1])
fooCell = mat2cell(foo,size(foo,1),ones(size(foo,2),1));
[x,y] = deal(fooCell{:});
4
Jonas

Я не могу комментировать другие ответы, поэтому отдельное дополнение.

вы можете использовать сделку (88,12), если вы начинаете со скаляров

deal может использоваться как однострочник и для не скаляров, конечно, если они у вас уже есть в отдельных переменных, скажем:

a = 123;
b = Rand(3);
c = {a, b};
d = struct('field','val')

и теперь вы разбираете их одной строкой:

>> [x,y,z,w] = deal(a,b,c,d)
x =
   123
y =
    0.6370    0.2165    0.6711
    0.2945    0.8803    0.2705
    0.7633    0.1537    0.0767
z = 
    [123]    [3x3 double]
w = 
    field: 'val'

Однако, если они упакованы в одну переменную, вы можете deal их, только если они находятся в массиве ячейки или структуры - с deal(X{:}) для массива ячеек и deal(S.field) для массива структуры. (В последнем случае обрабатывается только одно поле, но из всех структур в массиве.) В Matlab v.7 + вы можете использовать X{:} и S.field без deal, как отмечено в других ответах.

1
Victor K

Создать функцию arr2vars для удобства

function varargout = arr2vars(arr)
% Distribute elements over variables

N = numel(arr);
if nargout ~= N
    error('Number of outputs does not match number of elements')
end
for k = 1:N
    varargout{k} = arr(k);
end

Вы можете использовать его тогда, как это

[~,roi] = imcrop(im);
[x,w,y,h] = arr2vars(roi);
1
Tigliottu

СДЕЛКА действительно полезна и очень запутана. Я считаю, что foo должен быть сам массив ячеек. Следующее, кажется, работает в Octave, если я правильно помню, это будет работать и в MATLAB:

> foo = {88, 12}
foo =

{
  [1,1] =  88
  [1,2] =  12
}

> [x,y] = deal(foo{:})
x =  88
y =  12
1
mtrw

Возможно, вы ищете

>>> foo = [88, 12];
>>> [x, y] = deal(foo(1), foo(2))

в результате чего

x = 
    88

y = 
    12

Итак, у вас есть работающий однострочник.

0
luckydonald