How to write a function to determine if a date entered is valid
Mostrar comentarios más antiguos
I am to Write a function called valid_date that takes three positive integer scalar inputs year, month, day. If these three represent a valid date, return a logical true, otherwise false. The name of the output argument is valid. If any of the inputs is not a positive integer scalar, return false as well. Here is the code i have tried but keeps returning true for all dates, I have editted it a lot and now i just need a fresh eye to point out what i am doing wrong:
NB: I am not to use any built in MatLab function.
function valid = valid_date (year, month, date)
if (nargin==3)
valid=true;
elseif (isinteger(year)) && year>0 && (isscalar(year))
valid=true;
elseif (isinteger(month))&& month>0 && month <=12 && (isscalar(month))
valid=true;
elseif (isinteger(date)) && date > 0 && (isscalar(date))
valid=true;
elseif (month==1||month==3||month==5||month==7||month==8||month==10||month==12)&& date<=31
valid=true;
elseif date <=30
valid=true;
elseif (isinteger(year/4)) && month == 2 && (~(isinteger(year/100))||(isinteger(year/400))) && date <= 29
valid=true;
else
valid = false;
end
9 comentarios
Image Analyst
el 28 de Dic. de 2019
isinteger() and isscalar() are built-in functions.
Also, you can't just say it's valid after checking the year alone or the number of inputs. You have to check the months and days also.
Elliot Ogbechina
el 28 de Dic. de 2019
Stephen23
el 28 de Dic. de 2019
"so do i need to take out all the "valid = true" lines?"
No, you need to revise your logic. For example, the very first thing your function does is check if three inputs were provided:
function valid = valid_date (year, month, date)
if (nargin==3)
valid=true;
elseif ...
...
end
and if there are three inputs then you set valid=true and exit the function. So I could provide any three inputs and your function would set valid=true ... is that what your assignment required?
Walter Roberson
el 28 de Dic. de 2019
You can arrange the code as a cascade of rejections, each step declaring that valid is false if conditions were not met. Then at the very end when all possible rejections had been processed, anything left over must be valid.
Elliot Ogbechina
el 28 de Dic. de 2019
Elliot Ogbechina
el 31 de Dic. de 2019
Walter Roberson
el 31 de Dic. de 2019
TF = isinteger(A) returns logical 1 (true) if A is an array of integer type. Otherwise, it returns logical 0 (false).
Not integer value, integer type
Elliot Ogbechina
el 31 de Dic. de 2019
"so what can i use in place of isinteger"
Take a look at this answer you were given six hours ago:
Does it use isinteger? (hint: no)
What does it use? (hint: mod)
Also read my comment following that answer.
Respuestas (4)
aniket GIRI
el 27 de Mayo de 2020
function isvalid = valid_date(y, m, d)
% Check if the inputs are valid
% Check that they are scalars
if ~(isscalar(y) && isscalar(m) && isscalar(d))
isvalid = false;
% Check that inputs are positive
elseif ~all([y, m, d] > 0)
isvalid = false;
% Check that inputs are integers (not the data type)
elseif any(rem([y, m, d], 1))
isvalid = false;
% Check that m and d are below the max possible
elseif (m > 12) || (d > 31)
isvalid = false;
% The inputs could be a valid date, let's see if they actually are
else
% Vector of the number of days for each month
daysInMonth = [31 28 31 30 31 30 31 31 30 31 30 31];
% If leap year, change days in Feb
if isequal(rem(y, 4), 0) && (~isequal(rem(y, 100), 0) || isequal(rem(y, 400), 0))
daysInMonth(2) = 29;
end
maxDay = daysInMonth(m);
if d > maxDay
isvalid = false;
else
isvalid = true;
end
end
end
MUHAMMAD USMAN BIN AHMED
el 1 de Jun. de 2020
function valid = valid_date(year,month,day)
if nargin ~=3
valid=false;
elseif ((~isscalar(year))||(mod(year,1)~=0)||year<=0)
valid=false;
elseif ((~isscalar(month))||(mod(month,1)~=0)||month<=0||month>12)
valid=false;
elseif ((~isscalar(day))||(mod(day,1)~=0)||day<=0)
valid=false;
elseif (any(month==[1,3,5,7,8,10,12]) && day>31)
valid=false;
elseif (any(month==[4,6,9,11]) && day>30)
valid=false;
else
valid=true;
if (rem(year,4)==0 && month==2 && day<=29)
valid=true;
if (rem(year,100)==0)
valid=false;
if rem(year,400)==0
valid=true;
return
end
end
elseif month==2 && day>28
valid=false;
end
end
end
4 comentarios
MUHAMMAD USMAN BIN AHMED
el 1 de Jun. de 2020
this is the final code.
Walter Roberson
el 1 de Jun. de 2020
if (rem(year,4)==0 && month==2 && day<=29)
valid=true;
if (rem(year,100)==0)
valid=false;
No, that would declare (for example) February 15 1900 to be invalid. rem(year,4) would be 0, month would be 2, day <= 29 would be true, so you would set valid true, but then you test rem(year,100)=0 which would be true so you would set valid = false. Then rem(year,400) would not be 0 so you would leave valid false. Then since you found true on the year/month/day test you would not execute the elseif so valid will still be false. Then you would return, with valid still false.
MUHAMMAD USMAN BIN AHMED
el 1 de Jun. de 2020
function valid = valid_date_2(year,month,day)
if nargin ~=3
valid=false;
elseif ((~isscalar(year))||(mod(year,1)~=0)||year<=0)
valid=false;
elseif ((~isscalar(month))||(mod(month,1)~=0)||month<=0||month>12)
valid=false;
elseif ((~isscalar(day))||(mod(day,1)~=0)||day<=0)
valid=false;
elseif (any(month==[1,3,5,7,8,10,12]) && day>31)
valid=false;
elseif (any(month==[4,6,9,11]) && day>30)
valid=false;
else
valid=true;
if (rem(year,4)==0 && month==2 && day<=29)
valid=true;
if ((rem(year,100)==0)&& day>28)
valid=false;
if (rem(year,400)==0 && day<=29)
valid=true;
return
end
end
elseif month==2 && day>28
valid=false;
end
MUHAMMAD USMAN BIN AHMED
el 1 de Jun. de 2020
is this the correct one?
Jalaj Gambhir
el 31 de Dic. de 2019
Hi,
You can try the following function and see if it works for you:
function valid = valid_date(year, month, date)
if(nargin ~= 3)
valid = false;
elseif (~(mod(year,1)==0) || (year<0) || ~isscalar(year))
valid = false;
elseif (~(mod(month,1)==0) || (month<=0) || (month>12) || ~isscalar(month))
valid = false;
elseif (~(mod(date,1)==0) || (date<=0) || ~isscalar(date))
valid = false;
elseif( (month==1||month==3||month==5||month==7||month==8||month==10||month==12)&& date>31 )
valid = false;
elseif ((month==4||month==6||month==9||month==11) && date>30)
valid = false;
elseif(~leapyear(year) && month==2 && date>28)
valid=false;
elseif(leapyear(year) && month==2 && date>29)
valid=false;
else
valid = true;
end
6 comentarios
This answer has a few bugs in it, for example, the order of these arguments:
(~(mod(year,1)==0) || (year<0) || ~isscalar(year)
means that if year is non-scalar the code will throw an error (as inputs to the short-circuiting || must be scalar). The solution is to read the || documentation and check isscalar first, then the code will work as specified by the assignment:
~isscalar(year) || ...
The function leapyear is in the Aerospace Toolbox, which is unlikely to be available to most MATLAB users.
Rather than this obfuscation
~(mod(year,1)==0)
prefer the simpler
mod(year,1)~=0
Rather than repeating code:
month==1||month==3||month==5||month==7||month==8||month==10||month==12
just use basic MATLAB vector operations:
any(month==[1:2:7,8:2:12])
Elliot Ogbechina
el 31 de Dic. de 2019
Nice work. Your code is developing nicely.
Leap years are a bit fiddly. One simple approach would be to calculate the date limit number based on the year (i.e. not have a hard-coded 28 or 29), e.g. replace your two leapyear elseif conditions/statements with something like this:
elseif month==2 && date>(28+(mod(year,400)==0 || (mod(year,100)~=0 && mod(year,4)==0)))
Elliot Ogbechina
el 31 de Dic. de 2019
Anil Muradiya
el 1 de Jun. de 2020
Code doesn't work
Walter Roberson
el 1 de Jun. de 2020
Anil Muradiya which code are you finding is failing, and what inputs are you finding that it is failing on?
Sravani Kurma
el 17 de Jul. de 2020
Editada: Sravani Kurma
el 17 de Jul. de 2020
function valid=valid_date(year,month,date)
if nargin == 3;
if ~isscalar(year) || year<1|| year~=fix(year)||~isscalar(month) || month<1|| month~=fix(month)||~isscalar(date) || date<1|| date~=fix(date);
valid=false;
else
if ((year/100)~=fix(year/100)&&(year/4)==fix(year/4))||((year/400)==fix(year/400));
%leap yr
valid=mmm(month,date);
else
%non leap year
valid=sss(month,date);
end
end
end
function valid=mmm(month,date)
m30=[4,6,9,11];
m31=[1,3,5,7,8,10,12];
m29=2;
m28=2;
if max((month==m31)')==1
valid=(date>=1)&&date<=31;
elseif max(( month==m30)')==1
valid=(date>=1)&&date<=30;
elseif max((month==m29)')==1
valid=(date>=1)&&date<=29;
else
valid=false;
end
function valid=sss(month,date)
m30=[4,6,9,11];
m31=[1,3,5,7,8,10,12];
m29=2;
m28=2;
if max((month==m31)')==1
valid=(date>=1)&&date<=31;
elseif max(( month==m30)')==1
valid=(date>=1)&&date<=30;
elseif max((month==m28)')==1
valid=(date>=1)&&date<=28;
else
valid=false;
end
1 comentario
Walter Roberson
el 17 de Jul. de 2020
What will your code return if nargin is not 3?
if max((month==m31)')==1
Hint: any()
Categorías
Más información sobre Time Series Objects en Centro de ayuda y File Exchange.
Community Treasure Hunt
Find the treasures in MATLAB Central and discover how the community can help you!
Start Hunting!