cobol - 在 cobol 中添加两个整数会产生不需要的结果
问题描述
我正在将文件读入表格,请注意第一行不是表格的一部分。
1000
MS 1 - Join Grps Group Project 5 5
Four Programs Programming 15 9
Quiz 1 Quizzes 10 7
FORTRAN Programming 25 18
Quiz 2 Quizzes 10 9
HW 1 - Looplang Homework 20 15
在代码中,表格表示如下:
01 GRADES.
05 GRADE OCCURS 1 TO 100 TIMES DEPENDING ON RECORD-COUNT INDEXED BY J.
10 ASSIGNMENT-NAME PIC X(20).
10 CATEGORY PIC X(20).
10 POINTS-POSSIBLE PIC 9(14).
10 POINTS-EARNED PIC 9(14).
我还有一些其他累加器变量指定用于稍后计算总和/百分比。
01 RECORD-COUNT PIC 9(8) VALUE 0.
01 TOTAL-EARNED-POINTS PIC 9(14).
01 TOTAL-POSSIBLE-POINTS PIC 9(14) VALUE 0.
我的问题是,当我逐行阅读记录时,我想做以下事情:
ADD POINTS-EARNED(RECORD-COUNT) TO TOTAL-EARNED-POINTS
RECORD-COUNT
迭代中的当前位置在哪里。我希望TOTAL-EARNED-POINTS
第一次迭代后的值只是5
,对吗?但是,当我DISPLAY
计算 TOTAL-EARNED-POINTS 的值时,控制台会显示:
50000000000000
这是50万亿吗?还是数字 5 看起来很有趣?我怎样才能写这个,以便我可以用它做正确的数学来打印正确的成绩报告?
编辑:我知道可能有更好的方法来编写这个程序,但在尝试编写这个程序之前我从未使用过 cobol,而且我可能不会再使用它,或者至少在很长一段时间内都不会使用它。这是一个班级,所以只要我能正确打印我的输出,我就很好。完整的代码,到目前为止:
IDENTIFICATION DIVISION.
PROGRAM-ID. GRADEREPORT.
AUTHOR. JORDAN RENAUD.
DATE-WRITTEN. 09/18/2020.
ENVIRONMENT DIVISION.
INPUT-OUTPUT SECTION.
FILE-CONTROL.
SELECT GRADES-FILE ASSIGN TO "bill"
ORGANIZATION IS LINE SEQUENTIAL
ACCESS IS SEQUENTIAL.
DATA DIVISION.
FILE SECTION.
FD GRADES-FILE.
01 INPUT-TOTAL-POINTS PIC 9(4).
01 INPUT-GRADES.
05 INPUT-GRADE OCCURS 1 to 100 TIMES DEPENDING ON RECORD-COUNT INDEXED BY I.
10 INPUT-ASSIGNMENT-NAME PIC X(20).
10 INPUT-CATEGORY PIC X(20).
10 INPUT-POINTS-POSSIBLE PIC 9(14).
10 INPUT-POINTS-EARNED PIC 9(14).
WORKING-STORAGE SECTION.
77 GRADES-FILE-EOF PIC 9.
01 RECORD-COUNT PIC 9(8) VALUE 0.
01 TOTAL-EARNED-POINTS PIC 9(4) VALUE ZERO.
01 TOTAL-POSSIBLE-POINTS PIC 9(14) VALUE 5.
01 K PIC 9(14) VALUE 1.
01 TMP PIC 9(14).
01 CURRENT-CATEGORY PIC X(20).
01 CATEGORY-WEIGHT PIC X(3).
01 LAST-CATEGORY PIC X(20).
01 TOTAL-POINTS PIC 9(4).
01 GRADES.
05 GRADE OCCURS 1 TO 100 TIMES DEPENDING ON RECORD-COUNT INDEXED BY J.
10 ASSIGNMENT-NAME PIC X(20).
10 CATEGORY PIC X(20).
10 POINTS-POSSIBLE PIC 9(14).
10 POINTS-EARNED PIC 9(14).
PROCEDURE DIVISION.
OPEN INPUT GRADES-FILE.
READ GRADES-FILE INTO TOTAL-POINTS.
DISPLAY TOTAL-EARNED-POINTS
PERFORM UNTIL GRADES-FILE-EOF = 1
READ GRADES-FILE
AT END SET
GRADES-FILE-EOF TO 1
NOT AT END
ADD 1 TO RECORD-COUNT
MOVE INPUT-GRADES TO GRADE(RECORD-COUNT)
SET TOTAL-EARNED-POINTS UP BY POINTS-EARNED(RECORD-COUNT)
DISPLAY TOTAL-EARNED-POINTS
END-READ
END-PERFORM.
CLOSE GRADES-FILE.
DISPLAY TOTAL-EARNED-POINTS.
SORT GRADE ASCENDING CATEGORY.
MOVE CATEGORY(1) TO LAST-CATEGORY.
PERFORM RECORD-COUNT TIMES
MOVE CATEGORY(K) TO CURRENT-CATEGORY
IF CURRENT-CATEGORY = LAST-CATEGORY THEN
DISPLAY "SAME CATEGORY"
ELSE
DISPLAY "NEW CATEGORY"
MOVE LAST-CATEGORY TO CURRENT-CATEGORY
END-IF
SET K UP BY 1
END-PERFORM
DISPLAY GRADES.
STOP RUN.
编辑 2:在实现给定答案以将输入从文件转换为数字形式后,表的第一行读取正常,但从那时起它都是空白值。这是 READ 块的新代码,我不确定是否有更有效的方法来读取和转换组字段中的特定字段,但这是我认为应该这样做的方式。
PERFORM UNTIL GRADES-FILE-EOF = 1
READ GRADES-FILE
AT END SET
GRADES-FILE-EOF TO 1
NOT AT END
ADD 1 TO RECORD-COUNT
MOVE INPUT-ASSIGNMENT-NAME(RECORD-COUNT) TO ASSIGNMENT-NAME(RECORD-COUNT)
DISPLAY INPUT-ASSIGNMENT-NAME(RECORD-COUNT)
DISPLAY ASSIGNMENT-NAME(RECORD-COUNT)
MOVE INPUT-CATEGORY(RECORD-COUNT) TO CATEGORY(RECORD-COUNT)
DISPLAY INPUT-CATEGORY(RECORD-COUNT)
DISPLAY CATEGORY(RECORD-COUNT)
MOVE FUNCTION NUMVAL (INPUT-POINTS-POSSIBLE(RECORD-COUNT)) TO POINTS-POSSIBLE(RECORD-COUNT)
DISPLAY INPUT-POINTS-POSSIBLE(RECORD-COUNT)
DISPLAY POINTS-POSSIBLE(RECORD-COUNT)
MOVE FUNCTION NUMVAL (INPUT-POINTS-EARNED(RECORD-COUNT)) TO POINTS-EARNED(RECORD-COUNT)
DISPLAY INPUT-POINTS-EARNED(RECORD-COUNT)
DISPLAY POINTS-EARNED(RECORD-COUNT)
COMPUTE TOTAL-EARNED-POINTS = TOTAL-EARNED-POINTS + POINTS-EARNED(RECORD-COUNT)
DISPLAY TOTAL-EARNED-POINTS
END-READ
END-PERFORM.
解决方案
它是数字 5 的有趣表示吗?
不,这是一个未经检查的致命异常:EC-DATA-INCOMPATIBLE
.
原因:
您的数据定义和记录定义不匹配:
获得 10 分的 PIC 9(14)。
将会
"00000000000005"
不是
"5 "
看起来更好的定义是
10 SOME-POSSIBILY-NUMERIC-DATA PIC X(14).
如果您按照标签的建议使用 GnuCOBOL,然后添加-debug
到编译命令,您将看到停止程序的致命异常(COBOL 标准定义默认情况下所有异常检查都是关闭的,在我看来:由于遗留和性能,但是至少对于开发和测试来说,激活它们是非常合理的[在大多数情况下,让程序异常结束而不是在测试结束时做错数学更为合理])。
与任何计算机语言一样,您应该非常确定拥有有效数据(永远不要相信外部数据,无论它是区块链的一部分还是您读入的文本文件)。
我怎样才能写这个,以便我可以用它做正确的数学来打印正确的成绩报告?
如果你想使用“坏数据被忽略”(这里可能合适),只需转换它:
MOVE FUNCTION NUMVAL (SOME-POSSIBILY-NUMERIC-DATA)
TO POINTS-EARNED(RECORD-COUNT)
否则,请进行显式检查(完全数字 [自己的检查],或在左/右可能有空格的数字FUNCTION TEST-NUMVAL
)并停止程序/使用 aDISPLAY ... UPON SYSERR
或适合您的任何内容跳过错误行。
推荐阅读
- c++ - 警告:ISO C++ 禁止将字符串常量转换为 'char*' [-Wwrite-strings]
- cookies - 如何在 yaml 脚本中使用 routeid cookie 来验证 sessionid,为理智编写脚本所以还需要验证 session id
- mongodb - 需要有关 mongodb 设计的建议
- angular - 在构造函数中给出值后未定义的变量?
- architecture - POCO 类和稳定抽象原则
- flutter - Flutter 获取“类型”列表
' 不是类型 'Map 的子类型 ' 在类型转换错误中 - c# - 如何从类列表中找到相应的项目和值
- point-cloud-library - PCL中点云的变换——变换值的理解
- node.js - 是否可以从 App Store Connect API 获取安装和崩溃数据?
- php - 将 phpunit 升级到最新版本