在python中处理错误时,在执行完except语句下的内容后是否有办法回跳到代码出错的地方继续执行

python2.x

跑一段代码的时候,可能会报错某个错误,如果报错了,只需要进行一些处理,然后再继续执行原来的代码就好。

我目前是用try .. except来处理的,但是像这样只要报错就只会执行except下的代码然后程序结束

如果我加上finally的话,我又不能知道具体出错的是哪句代码(每一句都有可能报错),所以也没办法在finally下面加上后续的代码。

阅读 13.7k
3 个回答

我懂你的意思,我在做一个爬虫(结构比较简单的网页),将数据保存到csv,以下是代码,楼主可以参考。

    import requests, csv, time
    from lxml import etree
    
    head = {'Cookie': '·······',
            'User-Agent': '·······'}
    url = 'http://www.···.···.cn/·····/min_1'
    path = 'D:/test/Q**_data.csv' #保存路径
    pages = 122422 #总共的页面数量
    
    def restarting(dot):
        try:
            for i in range(dot, pages + 1):
                print("Page " + str(i) + " collecting......")
                urls = url + '?orderParam=&order···&pageNo=' + str(i)
                text = requests.get(urls, headers=head).text
                html = etree.HTML(text)
                with open(path, "a", newline='', encoding='GB2312') as file:
                    tbhead = ['序号', '日期', '气温', '湿度', '水气压', '风速', '风向', '辐射', '气压', '降雨']
                    csv_file = csv.DictWriter(file, fieldnames=tbhead)
                    if i == 1:
                        csv_file.writeheader()  #只在page1加表头
                    for n in range(1, 6):       #每页只有5条记录
                        data = html.xpath('//tbody/tr['+str(n)+']/td/@data-value')
                        csv_file.writerow({'序号':(i-1)*5+n,'日期':data[0],'气温':data[1],'湿度':data[2],'水气压':data[3],'风速':data[4],'风向':data[5],'辐射':data[6],'气压':data[7],'降雨':data[8]})
                        #这里我不会用writerows()一次性写5行数据(好像需要pandas的DataFrame)
        except Exception:        
            print("ERROR: Current page "+str(i)+"failed, retry in 10s...\n")
            time.sleep(10)
            return i
            
            
    start_point = 1
    while start_point < pages:
        re = restarting(start_point)
        start_point = re

以上就是完整的代码了,如图在第10562页出错会打印一个错误,但我记录下了这个error_point并赋值给了start_point,于是程序可以重新运行一次10562。
没有这么处理之前,如果10562页出错,(其他方式修改代码的话),要么退出程序(Process finished with exit code 1),要么虽然能继续但是是从10563开始而非10562。
图片

减小try-catch的粒度呢?或者贴上你的代码来分析吧,你说的exception后继续执行原来的代码有点模糊。

【修改】
看了下另外两个回答,基本上是我在下面评论中提出的两个思路,应该说大家的思路是基本一致的。这里修改下,给一个代码方案吧:

# introduce a new function
# @return: succeed, new_ip
def exec(ip, url):
    try:
        # do what you want to do 
        return True, ip
    except:
        # change your id
        return False, new_ip
        
# your original logics
urls = [url1, url2, ..., url10]
i = 0
ip = ip1
while i <= len(urls):
    succeed, ip = exec(ip, urls[i])
    if succeed:
        i +=1 

我刚刚写了一个,是生成Access Token的,Access Token数据库中是唯一的,但是生成Access Token的函数不能保证每次生成的Access Token都是不一样的。所以我用了一个很恶心的while循环。大致如下:

while True:
    try:
        access_token = make_token()
        access_token.add_to_db
        break
    except IntegrityError as e:
        if e.orig.args[0] == 1062:
            # 这种情况下,发生了mysql 1062异常,说明数据库中有重复的access token,需要反复生成新的,直至不同
            pass
        else:
            raise e

不高明,但可以用。

或者讲下你的业务,也许不用这么恶心。

题主原文:我这个还要更麻烦一些,比如我要依次请求10个url,假设访问到第5个断网了,就报错,然后去执行except下边换IP的代码,不可能每次请求都写一个while吧,这简直没法看了

我的方案:

def r(uri, data):
    while True:
        try:
            res = requests(uri, data)
            return res
        except:
            change_ip()

这种方案在单进程、单线程同步请求是应该可以的。

撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
推荐问题
宣传栏