我从 Pro*C 开始,有一个程序可以读取记录并打印出按标识值( guest )分组的记录。为了打印所有信息,我在 for 循环中使用了一个 break 来控制什么去哪里。它编译但不遵循打印方案,实际上进入了无限循环。最后的最后一个中断是停止无限循环,并且由于某种原因它没有达到任何条件(即使是默认的 else)。它正在执行的远程数据库服务器上的调试有限(dbx 不可用)。
虽然不是游标的结尾(即 sqlcode = 0) 如果 PrevSaleItem Not = Cursor.Item_ID 写下项目的小计 将 ItemQty 和 ItemTotal 初始化为 0 将 PrevSaleItem 设置为 Cursor.Item_ID 万一 打印明细行 添加 Cursor.Quantity 到 ItemQty 将 Cursor.calc_tax 添加到 ItemTotal 和 GuestTotal 获取游标中的下一条记录 结束时 打印最后一项的小计 打印总计
exec sql open dbGuest;
exec sql fetch dbGuest into :sLastName, :sFirstName, :nItemID, :sItemName, :nQuantity, :nUnitPrice, :sTransDate, :sHotelName, :nHotelID, :sTaxable, :nCalcTax;
/* Lets check the status of the OPEN statement before proceeding , else exceptions would be suppressed */
if(sqlca.sqlcode !=0) // If anything went wrong or we read past eof, stop the loop
{
printf("Error while opening Cursor <%d><%s>\n",sqlca.sqlcode,sqlca.sqlerrm.sqlerrmc);
break;
}
int PrevSaleItem = 0;
int ItemQty = 0;
double ItemTotal = 0.0;
double GuestTotal = 0.0;
for(;;)
{
printf("%s %s %s %s %d\n","Charge Summary for:", sFirstName.arr, sLastName.arr, " Guest_ID:", nGuest_ID);
printf("%s %d %s %s \n", "Sales_Item: ", nItemID, " - ", sItemName.arr);
// Do the crazy stuff to end the C-Strings
sLastName.arr[sLastName.len] = 0;
sFirstName.arr[sFirstName.len] = 0;
sItemName.arr[sItemName.len] = 0;
sTransDate.arr[sTransDate.len] = 0;
sHotelName.arr[sHotelName.len] = 0;
sTaxable.arr[sTaxable.len] = 0;
// initialize
PrevSaleItem = nItemID;
ItemQty = 0;
ItemTotal = 0.0;
GuestTotal = 0.0;
//While not end of cursor (ie. sqlcode = 0)
/* Check for No DATA FOUND */
if (sqlca.sqlcode == 0)
{
if(PrevSaleItem != nItemID)
{
printf("ItemQty: %f \t ItemTotal: %f",ItemQty,ItemTotal);
ItemTotal = 0.0;
GuestTotal = 0.0;
PrevSaleItem = nItemID;
}
printf("GuestTotal \t\t %f",GuestTotal);
ItemQty += nQuantity;
ItemTotal += nCalcTax;
GuestTotal += nCalcTax;
}
else if(sqlca.sqlcode == 100 || sqlca.sqlcode == 1403) // If anything went wrong or we read past eof, stop the loop
{
printf("CURSOR is empty after all fetch");
PrevSaleItem = -1;
break;
}
/* Check for other errors */
else if(sqlca.sqlcode != 0)
{
printf("Error while fetching from Cursor <%d><%s>\n",sqlca.sqlcode,sqlca.sqlerrm.sqlerrmc);
break;
}
else {printf("HIT ELSE"); break;}
break;
}
// close the cursor and end the program
exec sql close dbGuest ;
应该发生的逻辑如下
:
假设中断机制在外部 for 循环(未显示)内。它应该打开游标,获取第一条记录,然后打印客人和销售项目的标题。然后它通过设置值来初始化总计和中断变量。 While 不是游标的结尾(sqlcode 为 0,而不是 while 仅用于带 break 的循环)如果前一项 != 项目 ID,则它打印小计,将项目数量和总计初始化为 0,并将前一项设置为项目id(然后打破 if 条件)。项目数量增加了表中的数量,calc_tax
被添加到项目和客人总数中。获取下一条记录,打印最后一项的总计,然后打印该客人的总计。
它没有在应有的位置打印数量,并且处于无限循环中,在我尝试中断所有条件后,我无法解释它是如何发生的。我敢肯定这是一个初学者的错误,所以也许初级 Oracle 开发人员(但专业人士会很棒)有时间让我回到正轨。
最佳答案
您的fetch
在for
循环之外。您在第一次 fetch
之后进入循环。如果 sqlca.sqlcode
在第一次获取后为零,则由于嵌套的 if
和 else
而没有达到任何中断,因此它再次循环。因为没有第二次获取,你仍然有一个很好的 sqlcode
,所以它会继续。我认为....无论如何,将您的 fetch
移动到循环中看起来就像您想要发生的那样?
看起来最后的 break
应该开始了,但是昨天你发布了你修改过的代码,所以我不确定这里是否也是这种情况。
如果您重新格式化代码并为 else
条件引入大括号和嵌套,您会发现它们没有对齐。您显示的最终 {
不是针对 for
,而是针对其中一个分支。
for(;;)
{
if (sqlca.sqlcode == 0)
{
if(PrevSaleItem != nItemID)
{
...
}
....
}
else
{
if(sqlca.sqlcode == 100 || sqlca.sqlcode == 1403)
{
break;
}
else
{
if(sqlca.sqlcode != 0)
{
break;
}
else
{
printf("HIT ELSE");
break;
}
break;
}
// close the cursor and end the program
exec sql close dbGuest ;
else if(sqlca.sqlcode != 0)
是多余的;此时 sqlca.sqlcode
必须是 0、100 或 1403 以外的值。因此它也无法到达下一个 else
。
关于sql - Pro*C Oracle——循环获取 sql,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21959839/