www.gusucode.com > distcomp 案例源码程序 matlab代码 > distcomp/paralleldemo_quadpi_mpi.m

    %% Numerical Estimation of Pi Using Message Passing
% This example shows the basics of working with spmd statements, and how they
% provide an interactive means of performing parallel computations.  We do
% this by performing relatively simple computations to approximate pi.
%
% Related Documentation:
% 
% * <matlab:doc('spmd') spmd reference page> in the 
% Parallel Computing Toolbox(TM) User's Guide
%
% Related Examples:
%
% * <docid:distcomp_examples.example-ex58432651 Using GOP to Achieve MPI_Allreduce Functionality>

%   Copyright 2007-2013 The MathWorks, Inc.

%%
% The code shown in this example can be found in this function:
function paralleldemo_quadpi_mpi

%% Introduction
% We intend to use the fact that 
%
% $$ \int_0^1 {4 \over {1 + x^2}} dx = 4(atan(1) - atan(0)) = \pi$$
%
% to approximate pi by approximating the integral on the left.
%
% We intend to have the parallel pool perform the calculations in
% parallel, and to use the |spmd| keyword to mark the parallel blocks of
% code.  We first look at the size of the parallel pool that is currently
% open. 
p = gcp;
p.NumWorkers

%% Parallelize the Computations
% We approximate pi by the numerical integral of |4/(1 + x^2)| from 0
% to 1. 
type pctdemo_aux_quadpi.m


%% 
% We divide the work between the workers (labs) by having each worker calculate 
% the integral of the function over a subinterval of [0, 1] as shown in the
% picture. 
%
% <<../paralleldemo_quadpi_mpi_01.png>>

%%
% We define the variables |a| and |b| on all the workers, but let their values
% depend on |labindex| so that the intervals [a, b] correspond to the
% subintervals shown in the figure. We then verify that the intervals are
% correct.  Note that the code in the body of the spmd statement is
% executed in parallel on all the workers in the parallel pool.
spmd
    a = (labindex - 1)/numlabs;
    b = labindex/numlabs;
    fprintf('Subinterval: [%-4g, %-4g]\n', a, b);
end

%% 
% We let all the workers now use a MATLAB quadrature method to approximate
% each integral.  They all operate on the same function, but on the
% different subintervals of [0,1] shown in the figure above.
spmd
    myIntegral = integral(@pctdemo_aux_quadpi, a, b);
    fprintf('Subinterval: [%-4g, %-4g]   Integral: %4g\n', ...
            a, b, myIntegral);
end


%% Add the Results
% The workers have all calculated their portions of the integral of the
% function, and we add the results together to form the entire integral
% over [0, 1].  We use the |gplus| function to add |myIntegral| across all
% the workers and return the sum on all the workers.
spmd
    piApprox = gplus(myIntegral);
end

%% Inspect Results in the Client  
% Since the variable |piApprox| was assigned to inside an spmd statement,
% it is accessible on the client as a Composite.  Composite objects
% resemble cell arrays with one element for each worker.  Indexing into a
% Composite brings back the corresponding value from the worker to the client.
approx1 = piApprox{1};   % 1st element holds value on worker 1.
fprintf('pi           : %.18f\n', pi);
fprintf('Approximation: %.18f\n', approx1);
fprintf('Error        : %g\n', abs(pi - approx1))