⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 mds.m

📁 计量工具箱
💻 M
字号:
% MDS:  Nonmetric multidimensional scaling of a square symmetric distance matrix
%       in a specified number of dimensions, p.
%
%     Usage: [crd,stress,mapdist] = 
%                     mds(dist,{labels},{p},{restarts},{normflag},{noplot})
%
%       dist =     [n x n] square symmetric distance matrix.
%       labels =   optional [n x 1] vector of integer point labels.
%       p =        optional number of dimensions into which points are to be mapped
%                    [default = 2].
%       restarts = optional number of restarts, after finding a new minimum stress,
%                    needed to end search [default=0].
%       normflag = optional boolean flag indicating that stress is to be normalized 
%                    to a range [0-1] if true (=1), or unnormalized
%                    sum-of-squared deviations if false (=0) [default = 0].
%       noplot =   optional boolean flag indicating, if true (=1), that the first 
%                    two coordinate axes are not to be plotted [default = 0].
%       ---------------------------------------------------------------------------
%       crd =      [n x p] matrix of map coordinates.
%       stress =   final stress.
%       mapdist =  [n x ] square symmetric distance of mapped euclidean distances.
%

% Krzanowski and Marriott (1994), pp. 108-114.

% RE Strauss, 9/20/97
%   8/20/99 - changed plot colors and other miscellaneous changes for Matlab v5.
%   1/4/00 -  changed usage of sqplot().
%   2/22/00 - open new figure window for plot.
%   3/15/02 - change fmins() to fminsearch(); use default fminsearch options.
%   11/1/02 - changed 'plotflag' to 'noplot'.

function [crd,stress,mapdist] = mds(dist,labels,p,restarts,normflag,noplot)
  if (nargin < 2) labels = []; end;
  if (nargin < 3) p = []; end;
  if (nargin < 4) restarts = []; end;
  if (nargin < 5) normflag = []; end;
  if (nargin < 6) noplot = []; end;

  n = size(dist,1);

  if (isempty(labels))                   % Default argument values
    labels = [];
  end;
  if (isempty(p))
    p = 2;
  end;
  if (isempty(restarts))                     
    restarts = 0;
  end;
  if (isempty(normflag))
    normflag = 0;
  end;
  if (isempty(noplot))
    noplot = 0;
  end;

  if (nargout < 3)
    make_mapdist = 0;
  else
    make_mapdist = 1;
  end;

  if (make_mapdist)
    distsave = dist;
  end;

  no_norm = 1;
  if (normflag)
    no_norm = 0;
  end;

  options = foptions;                 % FMINS options
  options(2) = 1.e-3;                 % Set tolerance for pt-coords
  options(3) = 1.e-3;                 % Set tolerance for obj-fn

  % Get initial configuration from principal coordinates analysis
  [evects,evals] = pcoa(dist);
  if (length(evals)<p)                % Increase dimensions if necessary
    evects = [evects, zeros(n,p-length(evals))];
  end;
  init_pts = reshape(evects(:,1:p),n*p,1);  % Initial configuration

  % Repeat optimization until have 'restart' attempts with no better solution
  iter = restarts + 1;
  best_stress = 10e6;
  best_crd = init_pts;

  while (iter > 0)
    init_stress = mdsfunc(init_pts,dist,no_norm);

    if (init_stress > options(3))
%       crd = fminsearch('mdsfunc',init_pts,options,dist,no_norm); % Refine the configuration
      crd = fminsearch('mdsfunc',init_pts,[],dist,no_norm); % Refine the configuration
    else
      crd = init_pts;
    end;

    stress = mdsfunc(crd,dist,no_norm);              % Calculate stress
    if (stress < best_stress)                        % If new best stress,
      if ((best_stress-stress) > 0.001*best_stress)
        iter = restarts;                             %   restart if much better
      else
        iter = iter-1;                               %   else decrement repeats
      end;
      best_stress = stress;                          %   new stress
      best_crd = crd;                                %     along with crds
    else
      init_pts = crd + (2*rand-1)*std(crd);          % Else jiggle points
      iter = iter-1;                                 %   and decrement repeats
    end;
  end;

  stress = best_stress;
  crd = best_crd;

  crd = reshape(crd,n,p);
  crd = crd - ones(n,1)*mean(crd);    % Zero-center the axes

  if (~noplot)
    if (p>1)                          % Plot first two dimensions
      x = crd(:,1);
      y = crd(:,2);
      figure;
      plot(x,y,'k.');
      hold on;
      axis('square');

      sqplot(x,y,10);

      if (isempty(labels))
        for i=1:n
          text(crd(i,1),crd(i,2),[' ',int2str(i)]);
        end;
      else
        if (~ischar(labels))
          labels = tostr(labels);
        end;
        for i=1:n
          text(crd(i,1),crd(i,2),[' ',labels(i,:)]);
        end;
      end;
      hold off;
    end;
  end;

  if (make_mapdist)                   % Mapped euclidean distances
    mapdist = eucl(crd);
    f = mean(trilow(distsave))/mean(trilow(mapdist));
    mapdist = mapdist * f;            % Rescale to same mean as original dists
  end;

  return;

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -