How to put a legend automatically in PCA 3D Scatter Plot?
    8 visualizaciones (últimos 30 días)
  
       Mostrar comentarios más antiguos
    
    hello_world
      
 el 22 de Sept. de 2021
  
    
    
    
    
    Comentada: Rena Berman
    
 el 18 de Oct. de 2021
            Here is my solution which successfully creates a 3D scatter plot, but I am unable to add legend to it.  I will appreciate any help:
% data = mxn matrix; with all real numbers, no nan, inf or missing value. 
% label = mx1 vector, e.g., label = [0;0;...;0;1;1;...;1;2;2;...;2;3;3;...;3]
myClrMap = [      1 1 0;                      % Yellow
                  1 0 0;                      % Red
                  0 1 0;                      % Green
                  0 0 1;                      % Blue
                  ];
 [coeff_PCA, score_PCA, latent, tsquared, explained, mu] = pca(data);
 h = figure();
    subplot(1,2,1)    
    scatter3(score_PCA(:,1), score_PCA(:,2), score_PCA(:,3), 15, label, 'filled')
    colormap(myClrMap)    
    class = unique(label);
    % Below is my attempt to add legend which does not work.
    [r, c] = size(myClrMap);
    Legend = cell(r,1);
    for i = 1:r 
        Legend{i} = strcat(length(class), num2str(i));
    end
    legend(Legend, 'location', 'best')
    subplot(1,2,2)
    ...
    ...
    ...   
2 comentarios
  Rik
      
      
 el 24 de Sept. de 2021
				Original question:
How to put a legend automatically in PCA 3D Scatter Plot?
Here is my solution which successfully creates a 3D scatter plot, but I am unable to add legend to it.  I will appreciate any help:
% data = mxn matrix; with all real numbers, no nan, inf or missing value. 
% label = mx1 vector, e.g., label = [0;0;...;0;1;1;...;1;2;2;...;2;3;3;...;3]
myClrMap = [      1 1 0;                      % Yellow
                  1 0 0;                      % Red
                  0 1 0;                      % Green
                  0 0 1;                      % Blue
                  ];
 [coeff_PCA, score_PCA, latent, tsquared, explained, mu] = pca(data);
 h = figure();
    subplot(1,2,1)    
    scatter3(score_PCA(:,1), score_PCA(:,2), score_PCA(:,3), 15, label, 'filled')
    colormap(myClrMap)    
    class = unique(label);
    % Below is my attempt to add legend which does not work.
    [r, c] = size(myClrMap);
    Legend = cell(r,1);
    for i = 1:r 
        Legend{i} = strcat(length(class), num2str(i));
    end
    legend(Legend, 'location', 'best')
    subplot(1,2,2)
    ...
    ...
    ...   
Respuesta aceptada
  Walter Roberson
      
      
 el 22 de Sept. de 2021
        
      Editada: Walter Roberson
      
      
 el 22 de Sept. de 2021
  
          scatter3(score_PCA(:,1), score_PCA(:,2), score_PCA(:,3), 15, label, 'filled')
That generates one graphical object
   legend(Legend, 'location', 'best')
That can only label graphical objects. You only have one graphical object, so it can only create one legend entry.
Before telling you how to fix the problem within the code you already have:
Have you considered using one of the File Exchange gscatter3() instead ?
3 comentarios
  Walter Roberson
      
      
 el 23 de Sept. de 2021
				It generated 40, 41, 42, 43 for the legend because that is what you asked for. Your code had
    for i = 1:r 
        Legend{i} = strcat(length(class), num2str(i));
    end
Notice the length(class) in there. You have 4 classes, so you are asking that the number 4 be inserted into the legend before each of the four classes. I do not know why you wanted that, so I just reproduced the behaviour.
If you want grid lines and 3D then add in
    grid(ax, 'on');
    view(ax, 3)
  Walter Roberson
      
      
 el 23 de Sept. de 2021
				lab is a "dummy variable" name for the anonymous function
@(score_PCA,lab) scatter3(score_PCA(:,1), score_PCA(:,2), score_PCA(:,3), 15, lab, 'filled', 'displayname', NC + lab(1) )
When splitapply(FUN, score_PCA, label, G) is executed, then splitapply() will divide up the input variables score_PCA and label by rows according to the group number in the parameter G. All rows of score_PCA and label that have the same group number will be passed to the function. The selected rows of score_PCA will be passed as the first parameter, and the selected rows of label will be passed to the second parameter.
The anonymous function has to have some name to use for the (subset) of labels that was passed in, so I used the arbitary variable name lab . Because of the way we divided the groups according to label, all of the entries in lab are going to be the same, but there will be one row for each row in the selected rows from score_PCA .
I probably should also have used a different variable name for the first parameter, such as
splitapply(@(sP,lab) scatter3(sP(:,1), sP(:,2), sP(:,3), 15, lab, 'filled', 'displayname', NC + lab(1) ), score_PCA, label, G);
to emphasize that what is received as the first parameter is not the full set of score_PCA values.
Más respuestas (0)
Ver también
Categorías
				Más información sobre Data Distribution Plots en Help Center y File Exchange.
			
	Community Treasure Hunt
Find the treasures in MATLAB Central and discover how the community can help you!
Start Hunting!



