C语言关键字之voliate

C语言关键字之voliate

voliate的作用是作为指令关键字,确保本条指令不会因为编译器的优化而省略,而且要求每次从内存中直接读取值

当使用voliate 声明变量值时,系统总是重新从它所在的内存读取数据,直接访问变量地址,而编译器对于访问该变量时也不再进行优化

voliate关键字影响编译器的结果,用voliate 声明的变量表示该变量随时可能发生变化(因为编译器优化时可能将其放入寄存器中),与该变量有关的运算,不要再进行编译优化以免出错。

寄存器常用的优化方法:1.将内存变量缓存到寄存器中
                                           2.调整指令顺序,充分利用CPU指令流水线,进行指令重新排序读写指令。

下面来看一个程序

int main(){

int i = 0;
int arr[] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };

for (i = 0; i < 12; i++)
{
    arr[i] = i;
    printf("hehe\n");
}
return 0;

}

结果会是什么?出现了数组越界访问,程序还对吗?

但是在vs2013运行的结果却是死循环,为什么?

而且当我们将调试器换为release时竟然运行出来了12个hehe?!!

  1. 首先我们看看为什么会是循环
    内存的存储中的存储是由高到低的
    而数组则是由低到高的

    当到arr[12]时arr[12]的地址又与i的地址相同了,i又被初始为零

    在vs2013中有两个预留的间隔,而在vc6中只有一个预留的间隔所以编译器不一样结果也不一样

  2. 其次我们看看编译器是怎样优化

         

    编译器  更改了指令的顺序,将i的初始时序放到了数组下面,避免了循环,这个可以在反汇编中查看

所以说编译器的优化结果是可能对程序结果产生影响的
在Java中,编写比较方法时需要遵守“通用契约”,这是为了保证排序算法的正确性。然而,有时候会出现“java时间排序comparison method violate lts general contract”的错误。这是因为在比较方法中违反了通用契约的规定。 通用契约中指定了三个规则: 1. 反对称性(Antisymmetry):如果两个对象进行比较,其中一个大于另一个,那么另一个一定小于前者。如果两个对象相等,它们之间的比较结果应为0。 2. 传递性(Transitivity):如果对象A大于对象B,B大于对象C,那么A一定大于C。同样地,如果A等于B,B等于C,那么A一定等于C。 3. 一致性(Consistency):在比较过程中,如果两个对象发生变化,它们之间的比较结果应该随之改变。 当我们在比较Java中的时间时,经常使用的是Date类。要满足通用契约,我们需要按照时间顺序进行比较。然而,有时候我们在编写比较方法时可能没有正确地实现这些规则,导致了“java时间排序comparison method violate lts general contract”错误。 要解决这个问题,我们需要回顾一下比较方法的实现,并确保满足通用契约的规定。首先,我们应该确保在比较中使用的属性是可比较的,并且没有为null的情况。然后,我们应该检查比较方法是否正确地比较了两个对象的时间顺序,并根据比较结果返回正确的值。 此外,还有一种可能导致此错误的情况是,排序算法中使用了不支持通用契约的比较方法。在这种情况下,我们应该更改排序算法或者使用其他可靠的比较方法。 总之,解决“java时间排序comparison method violate lts general contract”错误,首先我们需要检查并修复比较方法中的错误。确保按照通用契约的规定进行比较,并且没有使用不支持通用契约的比较方法。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值