相比起我们一般用二层循环遍历二维数组,用一层循环不见得有效率上的优势。
但是在某些场合却能方便人们理解和使用。
今天我要使用这个的时候,却发现我有点忘了,然后想明白之后记录于此。
实质其实还是先行后列的遍历方式, 利用的是求余和整除两种运算。
假定我们要遍历的是20*20的整数型数组,一层循环的循环变量为n。
首先是在c语言这类以0作为数组一维的第一个下标的系统:
我们可以用一个从0到399的循环,那么在循环中要访问的一个数组元素的行标可为n/20,列标可为n%20。
这里主要需要考虑的是行列标的变动范围和边界值,n为0-19时,行标一直为0(整除!),列标分别为0-19(%20后的值范围就是0-19);
当n=20(第21次循环,轮到第二行第一列的元素了),n/20=1;n%20=0
如果你硬是要让循环从1开始到400,那么你就将上面提到的n变成(n-1)好了。
我们再来看以1作为数组一维的第一个下标的系统(例如:易语言):
这次我们先讨论循环从1到400,那么在循环中要访问的一个数组元素的行列标就不是上面那么简单了。
先来看行标,同样的,我们利用除法 也是n/20么?不是的,对了,1是起始下标,那么我们给他加个1,即n/20+1对么?
也不对,考虑n=20的时候(第20次循环,轮到第一行第二十个元素),n/20+1=2了,跳到第二行去了。
我们这样处理,应该为n/21+1,我们考虑跳行的边界值n=20、40、60……如果是除以20,那么这将提前跳行了,不是想要的结果。
因为刚刚好除出来整数了,而换成21之后就避开了这种情况。x*20/21=x-1 (整数运算)
再来看看列标,+1是必需的,但是答案不是n%20+1,而是(n-1)%20+1。
还是考虑跳行的边界值n=20、40、60……如果是前者,还是因为刚刚好除出来整数了,余数为零了,那么这将提前跳列了,不是想要的结果。
如果你硬是要让循环从0开始到399,那么你就将上面提到的n变成(n+1)好了。
这个问题不是很难的,怎么费了这么多唇舌还好像说的不是很清楚?