Please help me how to fit the data with a power law function

115 views (last 30 days)
Hi every one, I have a small problem with data needed to fit it with the power law.
x = [30,29,21,15,9,1];
y = [0.05,2,3.4,6.18,7.56,10;
Fit the data with a power function y = b * x^m.
I will be grateful for any idea or any help. Thank you very much.

Answers (4)

Image Analyst
Image Analyst on 19 Nov 2017
OK, haven't heard from you so it seems like you must be having trouble. Here is the full code with your actual data that you provided:
% Uses fitnlm() to fit a non-linear model (an power law curve) through noisy data.
% Requires the Statistics and Machine Learning Toolbox, which is where fitnlm() is contained.
% Initialization steps.
clc; % Clear the command window.
close all; % Close all figures (except those of imtool.)
clear; % Erase all existing variables. Or clearvars if you want.
workspace; % Make sure the workspace panel is showing.
format long g;
format compact;
fontSize = 20;
X = [30,29,21,15,9,1];
Y = [0.05,2,3.4,6.18,7.56,10];
% Now we have noisy training data that we can send to fitnlm().
% Plot the noisy initial data.
plot(X, Y, 'b*', 'LineWidth', 2, 'MarkerSize', 15);
grid on;
% Convert X and Y into a table, which is the form fitnlm() likes the input data to be in.
tbl = table(X', Y');
% Define the model as Y = a * (x .^ b) + c
% Note how this "x" of modelfun is related to big X and big Y.
% x((:, 1) is actually X and x(:, 2) is actually Y - the first and second columns of the table.
modelfun = @(b,x) b(1) * x(:, 1) .^ + b(2) + b(3);
a = -1 % Close guesses based on what I see from the raw data plot.
b = 1
c = 10
beta0 = [a, b, c]; % Guess values to start with. Just make your best guess.
% Now the next line is where the actual model computation is done.
mdl = fitnlm(tbl, modelfun, beta0);
% Now the model creation is done and the coefficients have been determined.
% YAY!!!!
% Extract the coefficient values from the the model object.
% The actual coefficients are in the "Estimate" column of the "Coefficients" table that's part of the mode.
coefficients = mdl.Coefficients{:, 'Estimate'}
% Now we have the coefficients and we can plot y for ANY x, not just the training set.
% Let's make up a bunch of x (50) from the min to the max.
xFitted = linspace(min(X), max(X), 50);
% Create smoothed/regressed data using the model:
yFitted = coefficients(1) * xFitted .^ coefficients(2) + coefficients(3);
% Now we're done and we can plot the smooth model as a red line going through the noisy blue markers.
hold on;
plot(xFitted, yFitted, 'r*-', 'LineWidth', 2);
grid on;
title('Power Law Regression with fitnlm()', 'FontSize', fontSize);
xlabel('X', 'FontSize', fontSize);
ylabel('Y', 'FontSize', fontSize);
legendHandle = legend('Noisy Y', 'Fitted Y', 'Location', 'north');
legendHandle.FontSize = 25;
message = sprintf('Coefficients for Y = a * X ^ b + c:\n a = %8.5f\n b = %8.5f\n c = %8.5f',...
coefficients(1), coefficients(2), coefficients(3));
text(5, 3, message, 'FontSize', 23, 'Color', 'r', 'FontWeight', 'bold', 'Interpreter', 'none');
% Set up figure properties:
% Enlarge figure to full screen.
set(gcf, 'Units', 'Normalized', 'OuterPosition', [0, 0.04, 1, 0.96]);
% Get rid of tool bar and pulldown menus that are along top of figure.
% set(gcf, 'Toolbar', 'none', 'Menu', 'none');
% Give a name to the title bar.
set(gcf, 'Name', 'Demo by ImageAnalyst', 'NumberTitle', 'Off')

Image Analyst
Image Analyst on 20 Nov 2017
Regarding your edit an hour ago where y = a * x ^ b, see code below:
% Uses fitnlm() to fit a non-linear model (an power law curve) through noisy data.
% Requires the Statistics and Machine Learning Toolbox, which is where fitnlm() is contained.
% Initialization steps.
clc; % Clear the command window.
close all; % Close all figures (except those of imtool.)
clear; % Erase all existing variables. Or clearvars if you want.
workspace; % Make sure the workspace panel is showing.
format long g;
format compact;
fontSize = 20;
X = [30,29,21,15,9,1];
Y = [0.05,2,3.4,6.18,7.56,10];
% Now we have noisy training data that we can send to fitnlm().
% Plot the noisy initial data.
plot(X, Y, 'b*', 'LineWidth', 2, 'MarkerSize', 15);
grid on;
% Convert X and Y into a table, which is the form fitnlm() likes the input data to be in.
tbl = table(X', Y');
% Define the model as Y = a * (x .^ b)
% Note how this "x" of modelfun is related to big X and big Y.
% x((:, 1) is actually X and x(:, 2) is actually Y - the first and second columns of the table.
modelfun = @(b,x) b(1) * x(:, 1) .^ + b(2);
b = 10 % Close guesses based on what I see from the raw data plot.
m = -1
beta0 = [b, m]; % Guess values to start with. Just make your best guess.
% Now the next line is where the actual model computation is done.
mdl = fitnlm(tbl, modelfun, beta0);
% Now the model creation is done and the coefficients have been determined.
% YAY!!!!
% Extract the coefficient values from the the model object.
% The actual coefficients are in the "Estimate" column of the "Coefficients" table that's part of the mode.
coefficients = mdl.Coefficients{:, 'Estimate'}
% Now we have the coefficients and we can plot y for ANY x, not just the training set.
% Let's make up a bunch of x (50) from the min to the max.
xFitted = linspace(min(X), max(X), 50);
% Create smoothed/regressed data using the model:
yFitted = coefficients(1) * xFitted .^ coefficients(2);
% Now we're done and we can plot the smooth model as a red line going through the noisy blue markers.
hold on;
plot(xFitted, yFitted, 'r*-', 'LineWidth', 2);
grid on;
title('Power Law Regression with fitnlm()', 'FontSize', fontSize);
xlabel('X', 'FontSize', fontSize);
ylabel('Y', 'FontSize', fontSize);
legendHandle = legend('Noisy Y', 'Fitted Y', 'Location', 'north');
legendHandle.FontSize = 25;
message = sprintf('Coefficients for Y = b * X ^ m:\n b = %8.5f\n m = %8.5f',...
coefficients(1), coefficients(2));
text(5, 3, message, 'FontSize', 23, 'Color', 'r', 'FontWeight', 'bold', 'Interpreter', 'none');
% Set up figure properties:
% Enlarge figure to full screen.
set(gcf, 'Units', 'Normalized', 'OuterPosition', [0, 0.04, 1, 0.96]);
% Get rid of tool bar and pulldown menus that are along top of figure.
% set(gcf, 'Toolbar', 'none', 'Menu', 'none');
% Give a name to the title bar.
set(gcf, 'Name', 'Demo by ImageAnalyst', 'NumberTitle', 'Off')
  2 Comments
Vidhi B
Vidhi B on 3 Sep 2018
It did not work for me. I am trying the same power law equation. mdl gave error.
Image Analyst
Image Analyst on 3 Sep 2018
Vidhi, I just tried it - copied and pasted, and it ran beautifully. Please read this link and then explain in A LOT more detail what "did not work for me" means.

Sign in to comment.


Matt J
Matt J on 18 Nov 2017
Edited: Matt J on 18 Nov 2017
The method with polyfit is a good way to come up with an initial estimate of m and b, but it would also be a good idea to further refine that initial estimate with a proper non-linear fitting routine. For your problem, FMINSPLEAS (Download) would be a good choice, because you have only a single nonlinear unknown variable.
r = polyfit(log(x),log(y),1);
funlist={@(m,x) x.^m};
[m,b]=fminspleas(funlist, r(1),x,y);

Image Analyst
Image Analyst on 19 Nov 2017
Use the fitnlm() (fit non-linear model) function.
Been there, done that, made demo (for prior poster). Requires Statistics and Machine Learning Toolbox. See attached demo and adapt as needed. Let me know if you need help.

Tags

Community Treasure Hunt

Find the treasures in MATLAB Central and discover how the community can help you!

Start Hunting!