如何在Python中使用C风格的for循环?

时间:2012-02-26 04:06:24

标签: python

我想在Python中使用传统的C-style for loop。我想循环遍历字符串的字符,但也知道它是什么,并且能够跳过字符(例如代码中的i = 5)。

带有range

for并没有给我实际for循环的灵活性。

7 个答案:

答案 0 :(得分:41)

C

for(int i=0; i<9; i+=2)
{
    dosomething(i);
}

python3

for i in range(0, 9, 2):
    dosomething(i)

你只是用不同的语言表达同样的想法。

答案 1 :(得分:35)

简单的答案是,在Python中没有简单,精确的C语言for语句。使用带有范围的Python for语句涵盖了其他答案。如果您希望能够在循环中修改循环变量(并使其影响后续迭代),则必须使用while循环:

i = 0
while i < 7:
    if someCondition(i):
        i = 5
    i += 1

但是在该循环中,continue语句与continue语句在C for循环中的效果不同。如果您希望continue按照C语言的方式工作,则必须输入try / finally语句:

i = 0
while i < 7:
    try:
        if someCondition(i):
            i = 5
        elif otherCondition(i):
            continue
        print 'i = %d' % i
    finally:
        i += 1

如你所见,这非常难看。你应该寻找一种更加Pythonic的方式来编写你的循环。

更新

这刚刚发生在我身上......有一个复杂的答案让你可以像C风格的循环一样使用普通的Python for循环,并允许通过编写自定义迭代器来更新循环变量。我不会为任何真正的程序推荐这个解决方案,但这是一个有趣的练习。

循环示例“C风格”:

for i in forrange(10):
    print(i)
    if i == 3:
        i.update(7)

输出:

0
1
2
3
8
9

技巧forrange使用int的子类添加update方法。 forrange的实施:

class forrange:

    def __init__(self, startOrStop, stop=None, step=1):
        if step == 0:
            raise ValueError('forrange step argument must not be zero')
        if not isinstance(startOrStop, int):
            raise TypeError('forrange startOrStop argument must be an int')
        if stop is not None and not isinstance(stop, int):
            raise TypeError('forrange stop argument must be an int')

        if stop is None:
            self.start = 0
            self.stop = startOrStop
            self.step = step
        else:
            self.start = startOrStop
            self.stop = stop
            self.step = step

    def __iter__(self):
        return self.foriterator(self.start, self.stop, self.step)

    class foriterator:

        def __init__(self, start, stop, step):
            self.currentValue = None
            self.nextValue = start
            self.stop = stop
            self.step = step

        def __iter__(self): return self

        def next(self):
            if self.step > 0 and self.nextValue >= self.stop:
                raise StopIteration
            if self.step < 0 and self.nextValue <= self.stop:
                raise StopIteration
            self.currentValue = forrange.forvalue(self.nextValue, self)
            self.nextValue += self.step
            return self.currentValue

    class forvalue(int):
        def __new__(cls, value, iterator):
            value = super(forrange.forvalue, cls).__new__(cls, value)
            value.iterator = iterator
            return value

        def update(self, value):
            if not isinstance(self, int):
                raise TypeError('forvalue.update value must be an int')
            if self == self.iterator.currentValue:
                self.iterator.nextValue = value + self.iterator.step

答案 2 :(得分:12)

for i in range(n):

...是C语言的Python等价物

for (i = 0; i < n; i++){

或者,您可以使用:

for i in range(a, n, s):

......相当于......

for (i = a; i < n; i+=s){

答案 3 :(得分:11)

我通过抗议提供以下完全滑稽的解决方案。请注意,'break'和'continue'不起作用。另请注意,循环体不能缩进。

class For:
    def __init__(self, **loop_vars):
        self.loop_vars = loop_vars
    def __call__(self, arg):
        if not hasattr(self, 'condition'):
            self.condition = arg
            return self
        if not hasattr(self, 'update'):
            self.update = arg
            return self
        while eval(self.condition, self.loop_vars, self.loop_vars):
            exec arg in self.loop_vars
            exec self.update in self.loop_vars


For(i = 1, j = 1)('i * j < 50')('i += 1; j += 1')('''
print i, j
''')

答案 4 :(得分:6)

给定数组a:

,您可以执行以下操作
for i in range(len(a)):
  a[i] = i

这是最接近C语言循环的Python。

您还可以为range命令提供更多参数;例如,

for i in range(2, len(a), 3)

将从i = 2开始,只要结果小于len(a),就将其递增3。

答案 5 :(得分:4)

Python for循环始终具有foreach语义。但是,你可以这样做:

for i in xrange(10):
    print i

这非常像C for循环。 xrange(或range,因为它在Python 3中重命名)是Python对象的构造函数,它遍历一系列数字。有关详细信息,请参阅文档。

答案 6 :(得分:0)

对于所有为什么“为什么?”的示例,下面的示例中C风格的循环会有所帮助:

remaining_retries = 3
for i in range(num_runs):
  success = do_run()
  if not success and remaining_retries > 0:
    i = i - 1
    remaining_retries = remaning_retries - 1