Jak przypisać wartości na przekątnej?

Załóżmy, że mam macierz NxN A, wektor indeksowy V składający się z podzbioru liczb 1: N i wartości K, i chcę to zrobić:

 for i = V
     A(i,i) = K
 end

Czy jest sposób, aby to zrobić w jednym oświadczeniu w / wektoryzacji?

Np. A(coś) = K

Twierdzenie A(V,V) = K nie zadziała, przypisuje elementy poza przekątne i nie tego chcę. np.:

>> A = zeros(5);
>> V = [1 3 4];
>> A(V,V) = 1

A =

 1     0     1     1     0
 0     0     0     0     0
 1     0     1     1     0
 1     0     1     1     0
 0     0     0     0     0
Author: Cris Luengo, 2010-10-19

6 answers

Zwykle używam do tego oka :

A = magic(4)
A(logical(eye(size(A)))) = 99

A =
    99     2     3    13
     5    99    10     8
     9     7    99    12
     4    14    15    99

Alternatywnie, można po prostu utworzyć listę indeksów liniowych, ponieważ od jednego elementu diagonalnego do następnego, To trwa nRows+1 kroki:

[nRows,nCols] = size(A);
A(1:(nRows+1):nRows*nCols) = 101
A =
   101     2     3    13
     5   101    10     8
     9     7   101    12
     4    14    15   101

Jeśli chcesz uzyskać dostęp tylko do podzbioru elementów przekątnych, musisz utworzyć listę indeksów przekątnych:

subsetIdx = [1 3];
diagonalIdx = (subsetIdx-1) * (nRows + 1) + 1;
A(diagonalIdx) = 203
A =
   203     2     3    13
     5   101    10     8
     9     7   203    12
     4    14    15   101

Alternatywnie można utworzyć tablicę indeksów logicznych za pomocą diag (działa tylko dla tablic kwadratowych)

diagonalIdx = false(nRows,1);
diagonalIdx(subsetIdx) = true;
A(diag(diagonalIdx)) = -1
A =
    -1     2     3    13
     5   101    10     8
     9     7    -1    12
     4    14    15   101
 59
Author: Jonas,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2010-10-18 22:04:41
>> tt = zeros(5,5)
tt =
     0     0     0     0     0
     0     0     0     0     0
     0     0     0     0     0
     0     0     0     0     0
     0     0     0     0     0
>> tt(1:6:end) = 3
tt =
     3     0     0     0     0
     0     3     0     0     0
     0     0     3     0     0
     0     0     0     3     0
     0     0     0     0     3

I bardziej ogólne:

>> V=[1 2 5]; N=5;
>> tt = zeros(N,N);
>> tt((N+1)*(V-1)+1) = 3
tt =
     3     0     0     0     0
     0     3     0     0     0
     0     0     0     0     0
     0     0     0     0     0
     0     0     0     0     3

Jest to oparte na fakcie, że macierze mogą być dostępne jako tablice jednowymiarowe (wektory), gdzie 2 indeksy (m,n) są zastąpione liniowym odwzorowaniem m*N+N.

 22
Author: ysap,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2010-10-18 21:43:17
A = zeros(7,6);
V = [1 3 5];

[n m] = size(A);
diagIdx = 1:n+1:n*m;
A( diagIdx(V) ) = 1

A =
     1     0     0     0     0     0
     0     0     0     0     0     0
     0     0     1     0     0     0
     0     0     0     0     0     0
     0     0     0     0     1     0
     0     0     0     0     0     0
     0     0     0     0     0     0
 2
Author: Amro,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2010-10-18 21:39:10

Załóżmy, że K jest wartością. Polecenie

A=A-diag(K-diag(A))

May be a bit faster

>> A=randn(10000,10000);

>> tic;A(logical(eye(size(A))))=12;toc

Upłynął czas 0.517575 sekund.

>> tic;A=A+diag((99-diag(A)));toc

Upłynął czas 0.353408 sekund.

Ale zużywa więcej pamięci.
 2
Author: ,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2015-05-16 22:34:26
>> B=[0,4,4;4,0,4;4,4,0]

B =

     0     4     4
     4     0     4
     4     4     0

>> v=[1,2,3]

v =

     1     2     3

>> B(eye(size(B))==1)=v
%insert values from v to eye positions in B

B =

     1     4     4
     4     2     4
     4     4     3
 2
Author: Gabriel Cedillos,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2018-05-25 01:41:00

Użyłbym sub2ind i przekazałbym indeksy przekątnej jako parametry x i y:

A = zeros(4)
V=[2 4]

idx = sub2ind(size(A), V,V)
% idx = [6, 16]

A(idx) = 1

% A =
% 0     0     0     0
% 0     1     0     0
% 0     0     0     0
% 0     0     0     1
 1
Author: Eran W,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2014-05-17 08:43:00