- Mark as New
- Bookmark
- Subscribe
- Mute
- Subscribe to RSS Feed
- Permalink
- Report Inappropriate Content
This simple program is supposed to calculate principal and interest as you make payments.
If the payment is <= the remaining principal, the payment amount is reduced accordingly.
When I run it , it does NOT show that I set the principal (ypr) = to 0.
In other words, line 19 gets skipped in the compiled code, where I set ypr= 0.
Am I missing something here? Why would the compiler ignore that line?
Link Copied
- Mark as New
- Bookmark
- Subscribe
- Mute
- Subscribe to RSS Feed
- Permalink
- Report Inappropriate Content
- Mark as New
- Bookmark
- Subscribe
- Mute
- Subscribe to RSS Feed
- Permalink
- Report Inappropriate Content
It's just waiting for input (by default, from keyboard).
Surprisingly, all this IBM-360ish non-Fortran stuff seems to work with several current compilers.
It doesn't look like you ever satisfied the condition ypr <= pay
- Mark as New
- Bookmark
- Subscribe
- Mute
- Subscribe to RSS Feed
- Permalink
- Report Inappropriate Content
The DO loop runs the specified number of times without the IF..ENDIF block being executed. The IF condition is not satisfied until after the 360th iteration of the loop.
- Mark as New
- Bookmark
- Subscribe
- Mute
- Subscribe to RSS Feed
- Permalink
- Report Inappropriate Content
I guess you did not actually run the program.
You can put in a payment amount that will make it quit early.
You still see the same problem.
- Mark as New
- Bookmark
- Subscribe
- Mute
- Subscribe to RSS Feed
- Permalink
- Report Inappropriate Content
I put in a PRINT statement BEFORE line 19, and then it
DOES execute that AND the next line.
So putting in the PRINT statement somehow fixes the compiler problem.
I see what you mean about the 360th time it executes the Do LOOP.
But I don't see why putting in a PRINT statement would change anything.
- Mark as New
- Bookmark
- Subscribe
- Mute
- Subscribe to RSS Feed
- Permalink
- Report Inappropriate Content
There is no problem that I can see. Add a statement such as
[fortran]write(*,*)' IF TEST SUCCEEDED'[/fortran]
after the IF statement, put in the payment amount that you want, and rerun the program.
If you still think that there is a problem, you would need to provide details such as compiler version, options used, etc.
- Mark as New
- Bookmark
- Subscribe
- Mute
- Subscribe to RSS Feed
- Permalink
- Report Inappropriate Content
That READ(*,*) statement is just to keep the program from quitting before I can read the window contents.
Surprisingly, in a GRAPHICS program, we don't need to do that.
It still allows you to examine the window contents before it actually quits,
but that is not true for CONSOLE programs.
- Mark as New
- Bookmark
- Subscribe
- Mute
- Subscribe to RSS Feed
- Permalink
- Report Inappropriate Content
The condition would be satisfied the 361st time around the loop but you stop at 360.
For console you can use the PAUSE command to wait at the end. I really would not recomment " implicit real*8 (A-H,O-Z)", use implicit none and declare everthing. It saves time in the long run.
I write some software and customers contact me sometimes saying there is a bug. More than 90% of the time they are doing something wrong or the software is doing what it should but they think it should be doing something different. It irritates me a bit on some occaisons. When I have a problem I start with "I have a problem with... " or "I don't undertstand why...." . Bug may be the terms that is appropraite having been through the issue in detail but isn't the starting point IMO.
- Mark as New
- Bookmark
- Subscribe
- Mute
- Subscribe to RSS Feed
- Permalink
- Report Inappropriate Content
That's why I put a question mark after "bug."
Cause I wasn't sure - - -
One of the other guys suggested the READ(*,*) approach.
They said PAUSE was an obsolete feature, so I tried that.
- Mark as New
- Bookmark
- Subscribe
- Mute
- Subscribe to RSS Feed
- Permalink
- Report Inappropriate Content
Bill,
The repayment for 360 monthly payments at nominal 4.5%; principal = 195,000, n = 360, r = .045/12 = 0.375% )
monthly repayment = principal * r * (1+r)^n / ( (1+r)^n - 1) = 988.04
where the repayment ratio = (1+r)^n / ( (1+r)^n - 1)
This formula ignores the rounding policy for interest calculation and also assumes monthly interest = annual interest / 12.
If you estimate a monthly compounding interest rate equivalent to an annual rate of 4.5%, this is 0.3675%, with a resulting monthly repayment of 977.6. ( my apologies if I have these numbers wrong, but it demonstrates the problem.) You calculated rate is really paying nearly 4.6%.
You need to read the fine print on how monthly interest is defined.
Your estimate of 987 extended the pain by 1 month !!
John
ps: I increased the repayment to 998 and it stopped at payment 353, with the if test being activated.
- Mark as New
- Bookmark
- Subscribe
- Mute
- Subscribe to RSS Feed
- Permalink
- Report Inappropriate Content
Actually, this first came up when I put in a payment of $988.04
That's what the bank is charging me for my mortgage.
But this is a tiny amount too small.
If I change it to $988.05, the last payment (#360) will in fact pay it off.
I was experimenting with different payment amounts to see what would happen.
- Mark as New
- Bookmark
- Subscribe
- Mute
- Subscribe to RSS Feed
- Permalink
- Report Inappropriate Content
THanks to all who responded.
- Mark as New
- Bookmark
- Subscribe
- Mute
- Subscribe to RSS Feed
- Permalink
- Report Inappropriate Content
Some comments of my own:
The statement:
xi=xint/100.0/12.0
is using both single-precision and double-precision in the calculation. In this case it may be harmless, but in general using mixed-mode arithmetic can lead to loss of precision.
Also, when computing with currency, use of binary floating point can be problematic due to rounding issues. I've attached an amortization program I wrote many years back that uses scale factors to convert the input values to integer-valued floats so that there's no loss of precision.
Edit: Hmm, attach not working. I'll insert it inline for now and see what's going on:
[fortran]
PROGRAM AMORT
!+
	! This program performs loan amortization calculations for loans which
	! are paid monthly.  It uses scaled double-precision values in order to
	! avoid losing fractions of cents.  The P format is used to scale input
	! values to cents on input and back to dollars on output.
	!
	! Author: Steve Lionel
	!
	!-
IMPLICIT NONE
INTEGER, PARAMETER :: DP = SELECTED_REAL_KIND(15)
	REAL(DP) :: RATE,RATE_PER_PAYMENT
	INTEGER  :: NUMBER_OF_PAYMENTS
	REAL(DP) :: REMAINING_PRINCIPAL
	REAL(DP) :: MONTHLY_PAYMENT,SCHEDULED_PAYMENT
	REAL(DP) :: MONTHLY_PRINCIPAL
	REAL(DP) :: MONTHLY_INTEREST
	REAL(DP) :: PRINCIPAL_FACTOR
	REAL(DP) :: YTD_PRINCIPAL,YTD_INTEREST
	REAL(DP) :: TOTAL_INTEREST
	INTEGER  :: YEAR_NUMBER, MONTH_NUMBER, DAY_NUMBER, PAYMENT_NUMBER, DAYTEMP
	LOGICAL  :: DONE
	TYPE T_M
	  INTEGER :: DAYS_IN_MONTH
	  CHARACTER(3) :: MONTHNAME
	  END TYPE T_M
	TYPE(T_M), DIMENSION(12), PARAMETER :: MONTHS = (/ &
	  T_M(31,'Jan'),T_M(28,'Feb'),T_M(31,'Mar'),T_M(30,'Apr'),T_M(31,'May'), T_M(30,'Jun'), &
	  T_M(31,'Jul'),T_M(31,'Aug'),T_M(30,'Sep'),T_M(31,'Oct'),T_M(30,'Nov'), T_M(31,'Dec') /)
	                       
	CHARACTER(200) FILESPEC
	CHARACTER YN
11  FORMAT (A)
	12  FORMAT (BN,2PG10.0)
	13  FORMAT (BN,I10)
	14  FORMAT (BN,-2PG12.0)
	15  FORMAT (A)
Mainloop: DO
	    WRITE (*,11,ADVANCE='NO') ' Enter interest rate in percent, ^Z to exit: '
	    READ (*,12,END=999) RATE
	    RATE_PER_PAYMENT = RATE/12.0D0
	    WRITE (*,11,ADVANCE='NO') ' Enter number of months: '
	    READ (*,13,END=999) NUMBER_OF_PAYMENTS
	    WRITE (*,11,ADVANCE='NO') ' Enter principal amount: '
	    READ (*,14,END=999) REMAINING_PRINCIPAL
    PRINCIPAL_FACTOR = 1.0D0 - ((RATE_PER_PAYMENT+1.0D0)** (-NUMBER_OF_PAYMENTS))
	    PRINCIPAL_FACTOR = RATE_PER_PAYMENT / PRINCIPAL_FACTOR
    SCHEDULED_PAYMENT = DNINT(REMAINING_PRINCIPAL * PRINCIPAL_FACTOR)
	    TOTAL_INTEREST = (NUMBER_OF_PAYMENTS * SCHEDULED_PAYMENT) - &
	    REMAINING_PRINCIPAL
101 FORMAT ( ' Principal amount = $',-2PF9.2)
	102 FORMAT ( ' Interest rate    =  ',2PF9.3,'%')
	103 FORMAT ( ' Duration of loan =  ',I9,' months')
	104 FORMAT (/' Monthly payment  = $',-2PF9.2)
	105 FORMAT ( ' Total interest   = $',-2PF9.2)
    WRITE (*,101) REMAINING_PRINCIPAL
	    WRITE (*,102) RATE
	    WRITE (*,103) NUMBER_OF_PAYMENTS
	    WRITE (*,104) SCHEDULED_PAYMENT
	    WRITE (*,105) TOTAL_INTEREST
    DO
	        WRITE (*,11,ADVANCE='NO') ' Do you wish an amortization report? 
	        READ (*,15,END=999) YN
	        IF (YN == 'Y' .OR. YN == 'y' .OR. YN == ' ') EXIT
	        IF (YN == 'N' .OR. YN == 'n') EXIT Mainloop
	        WRITE (*,11) ' Please respond with Y or N'
	        END DO
    WRITE (*,11,ADVANCE='NO') ' Enter loan year number (2005, etc.): '
	    READ (*,13,END=999) YEAR_NUMBER
	    WRITE (*,11,ADVANCE='NO') ' Enter loan month number (1-12): '
	    READ (*,13,END=999) MONTH_NUMBER
	    WRITE (*,11,ADVANCE='NO') ' Enter loan day number (1-31): '
	    READ (*,13,END=999) DAY_NUMBER
	    WRITE (*,11,ADVANCE='NO') ' Enter file specification for report: '
	    READ (*,15) FILESPEC
	    OPEN (UNIT=1,FILE=FILESPEC,STATUS='REPLACE',FORM='FORMATTED')
	    WRITE (1,101) REMAINING_PRINCIPAL
	    WRITE (1,102) RATE
	    WRITE (1,103) NUMBER_OF_PAYMENTS
	    WRITE (1,104) SCHEDULED_PAYMENT
	    WRITE (1,105) TOTAL_INTEREST
	106 FORMAT (///,T30,'Calendar year ',I4// &
	        '     Payment',T25,'Remaining',T37,'Principal',T48, &
	        'Interest',T61,'Total'/ &
	        '   #     Date',  T25,'Principal',T39,'Payment', T49, &
	        'Payment', T59,'Payment'/ &
	        1X,3('-------------------------'))
	107    FORMAT (1X,I3,1X,I2,'-',A3,'-',I4,2X,-2PF15.2,2X,3(-2PF10.2))
	109    FORMAT (//' Year-to-date interest paid    = $',-2PF9.2/ &
	              ' Year-to-date principal paid   = $',-2PF9.2/ &
	              ' End-of-year principal balance = $',-2PF9.2)
    DONE = .FALSE.
	    DO WHILE (PAYMENT_NUMBER < NUMBER_OF_PAYMENTS)
	        IF (MONTH_NUMBER < 12) WRITE (1,106) YEAR_NUMBER
        YTD_INTEREST = 0
	        YTD_PRINCIPAL = 0
	        
	        ! Do up until the end of this calendar year
	        DO WHILE ((MONTH_NUMBER < 12) .AND. .NOT. DONE)
	            PAYMENT_NUMBER = PAYMENT_NUMBER + 1
	            MONTH_NUMBER = MONTH_NUMBER + 1
	            MONTHLY_INTEREST = DNINT(REMAINING_PRINCIPAL * RATE_PER_PAYMENT)
	            MONTHLY_PRINCIPAL = SCHEDULED_PAYMENT - MONTHLY_INTEREST
	            ! Last payment?
	            IF (PAYMENT_NUMBER == NUMBER_OF_PAYMENTS) THEN
	                MONTHLY_PRINCIPAL = REMAINING_PRINCIPAL
	                DONE = .TRUE.
	                END IF
	            MONTHLY_PAYMENT = MONTHLY_INTEREST + MONTHLY_PRINCIPAL
	            DAYTEMP = MIN(DAY_NUMBER, MONTHS(MONTH_NUMBER)%DAYS_IN_MONTH)
	            IF ((MONTH_NUMBER == 2) .AND. (MOD(YEAR_NUMBER,4) == 0)) &
	              DAYTEMP = MIN(DAY_NUMBER, 29)
	            WRITE (1,107) PAYMENT_NUMBER, DAYTEMP, &
	              MONTHS(MONTH_NUMBER)%MONTHNAME,YEAR_NUMBER, &
	              REMAINING_PRINCIPAL, MONTHLY_PRINCIPAL, &
	              MONTHLY_INTEREST, MONTHLY_PAYMENT
	            REMAINING_PRINCIPAL = REMAINING_PRINCIPAL - MONTHLY_PRINCIPAL
	            YTD_INTEREST = YTD_INTEREST + MONTHLY_INTEREST
	            YTD_PRINCIPAL = YTD_PRINCIPAL + MONTHLY_PRINCIPAL
	            END DO
        ! Print year-to-date summary
	        IF (PAYMENT_NUMBER > 0) WRITE (1,109) YTD_INTEREST, &
	            YTD_PRINCIPAL, REMAINING_PRINCIPAL
	        YEAR_NUMBER = YEAR_NUMBER + 1
	        MONTH_NUMBER = 0
	        END DO
	    CLOSE (UNIT=1)
	    END DO Mainloop
	999 END[/fortran]
- Mark as New
- Bookmark
- Subscribe
- Mute
- Subscribe to RSS Feed
- Permalink
- Report Inappropriate Content
Hi steve;
I am going to try this code, see what it does.
A questyion did come up about the use of those constants.
Since 12 is numerically identical to 12.0 or even 12.D0,
would it be necessary to say something like:
Xi =xint /dble(12.0) / dble(100.0)
or xi=xint/12.d0 /100.d0
In other words, would the compiler automatically know to convert all quantiites to DBL
precision before doing the divides?
I could have said: xi=xint/12/100 couldn't I?
Maybe I am assuming too much here.
- Mark as New
- Bookmark
- Subscribe
- Mute
- Subscribe to RSS Feed
- Permalink
- Report Inappropriate Content
If you said xint/12/100 the compiler would be free to evaluate this as xint / dble(12/100) which would end up giving you a zerodivide error, since 12/100 is 0 in integer. This is the sort of thing that should make you avoid mixed-mode arithmetic everywhere.
- Mark as New
- Bookmark
- Subscribe
- Mute
- Subscribe to RSS Feed
- Permalink
- Report Inappropriate Content
I disagree that the compiler is free to change xint/12/100 (assuming xint is real or dp)
to xint/dble(12/100). xint/12/100 and xint/(12/100) are *not* mathematically
equivalent, because of the difference between real and integer division.
Semantics of the language say interpretation must be left to right with
mulitple /s. After the interpretation is established, then the compiler
can make changes that are mathematically equivalent. So it could
change it to xint/100/12, for example, or even xint/1200.0.
- Mark as New
- Bookmark
- Subscribe
- Mute
- Subscribe to RSS Feed
- Permalink
- Report Inappropriate Content
Ah, yes. Walt is correct here - my apologies. Nevertheless, mixed-mode arithmetic is to be avoided.
- Mark as New
- Bookmark
- Subscribe
- Mute
- Subscribe to RSS Feed
- Permalink
- Report Inappropriate Content
but xint/(12/100) is algebraically = to xint *100 /12
when you divide by a fraction, that is the same as inverting it and multiplying it.78
I was told that the compiler proceeds from left to right when it determines the precision of an operation,
so if two numbers have a different precision, it picks the higher precision.
Now would the compiler first convert 100 to a double precision number, THEN multiply it?
Since the result is now Double precision, wouldn't it then convert 12 to a D. P. number before multiplying it?
I agree that putting the 12/100 inside a parenthesis is asking for trouble, cause you are forcing
the compiler to use integer arithmetic.
xint/12/100 is algebraically = to (xint/12)/100
I don't think the compiler is supposed to ignore the rules of 9th grade algebra, is it?
All these give the EXACT same answer, BTW.
real*8 xint/4.5D0/
	  print *,xint/12/100
	  print *,xint/12./100.
	  print *,xint/(12*100)
	  print *,xint/(12.d0*100)
	  print *,xint/1200.D0
	  read(*,*)
	  end
- Mark as New
- Bookmark
- Subscribe
- Mute
- Subscribe to RSS Feed
- Permalink
- Report Inappropriate Content
When you have xint/100/12, the compiler does xint/100 first. According to the rules of mixed-mode arithmetic in the standard, the 100 is converted to the type and kind of xint (real(8) here. Then the divide by 12 is done, and again the 12 is converted to real(8) for the division.
- Mark as New
- Bookmark
- Subscribe
- Mute
- Subscribe to RSS Feed
- Permalink
- Report Inappropriate Content
I agree there, and I also see why xint/(12/100) gives you.a divide by zero.
You are forcing integer arithmetic there, too.
but since 12*100 =1200, and 12.D0 *100.D0 =1200.d0 ,then
multiply by (12*100) gives the same answer as multiply by 1200.D0
Sloppy programming, perhaps - - - -
You might get into trouble with integer overflow, since
the compiler does NOT check for that.
For example 257*257 would possibly give a wrong answer.
How would the compiler know that the result wont fit into a 2 byte integer?
Does it automatically assume the result will be integer(8)?
 
					
				
				
			
		
- Subscribe to RSS Feed
- Mark Topic as New
- Mark Topic as Read
- Float this Topic for Current User
- Bookmark
- Subscribe
- Printer Friendly Page
