如何在python中检查连接是否处于活动状态?

如何在python中检查连接是否处于活动状态?,python,python-3.x,Python,Python 3.x,我使用raspberry pi上的python和读卡器RDM880使项目读取RFID标签 我的想法是花时间进进出出与员工核实是否按时工作 我尝试使用python向本地mysql和远程mysql(IP:192.168.137.1)添加card\u ID、time\u in、time\u out 它在远程和本地mysql中具有相同的表 如果mysql remote坏了,我只想添加到本地mysql 这是我的密码: import serial import time import RPi.GPIO as

我使用raspberry pi上的python和读卡器RDM880使项目读取RFID标签

我的想法是花时间进进出出与员工核实是否按时工作

我尝试使用python向本地mysql和远程mysql(IP:192.168.137.1)添加card\u ID、time\u in、time\u out

它在远程和本地mysql中具有相同的表

如果mysql remote坏了,我只想添加到本地mysql

这是我的密码:

import serial
import time
import RPi.GPIO as GPIO
import MySQLdb
from datetime import datetime
from binascii import hexlify
serial=serial.Serial("/dev/ttyAMA0",                    
                  baudrate=9600,                     
                  parity=serial.PARITY_NONE,                    
                  stopbits=serial.STOPBITS_ONE,                    
                  bytesize=serial.EIGHTBITS,
                  timeout=0.1) 
db_local = MySQLdb.connect("localhost","root","root","luan_van") #connect local
db = MySQLdb.connect("192.168.137.1", "root_a","","luan_van") #connect remote
ID_rong = 128187 # reader respone if no card
chuoi= "\xAA\x00\x03\x25\x26\x00\x00\xBB"
def RFID(str): #function read RFID via uart 
      serial.write(chuoi)
      data = serial.readline()
      tach_5 = data[5]
      tach_6 = data[6]
      hex_5 = hexlify(tach_5)
      hex_6= hexlify(tach_6)
      num_5 = int(hex_5,16)
      num_6 = int(hex_6,16)
      num_a = num_5 * 1000 + num_6
      if(num_a != ID_rong):
            tach_7 = data[7]
            tach_8 = data[7]
            hex_7 = hexlify(tach_7)
            hex_8= hexlify(tach_8)
            num_7 = int(hex_7,16)
            num_8 = int(hex_8,16)
            num = num_8 + num_7 * 1000 + num_6 * 1000000 + num_5 * 1000000000
      else:
            num = num_5 * 1000 + num_6
      return num
def add_database(): # add card_ID and time_in to remote mysql
      with db:
           cur = db.cursor()
           cur.execure("INSERT INTO tt_control(Card_ID,Time_in) VALUES ('%d',NOW()) " %num)
           return
def add_database_local(): # add card_ID and time_in to remote mysql
      with db_local:
           cur = db_local.cursor()
           cur.execure("INSERT INTO tt_control(Card_ID,Time_in) VALUES ('%d',NOW()) " %num)
           return
def have_ID(int): #check ID in table tt_control
       with db_local:
           cur = db_local.cursor(MySQLdb.cursors.DictCursor)
           cur.execute("SELECT * FROM tt_control WHERE Card_ID = '%d'" %num)
           rows = cur.fetchall()
           ID=""
           for row in rows:
               ID = row['Card_ID']
       return ID
def add_time_out(): #add time out to remote mysql
       with db:
           cur = db.cursor(MySQLdb.cursors.DictCursor)
           cur.execute("UPDATE tt_control SET Time_out = NOW() WHERE Card_ID = '%d'" %num)
       return
def add_time_out_local(): #add time out to local mysql
       with db_local:
           cur = db_local.cursor(MySQLdb.cursors.DictCursor)
           cur.execute("UPDATE tt_control SET Time_out = NOW() WHERE Card_ID = '%d'" %num)
       return

  def add_OUT(): #increase Card_ID to distinguish second check
     with db:
           cur = db.cursor(MySQLdb.cursors.DictCursor) 
           cur.execute("UPDATE tt_control SET Card_ID = Card_ID + 1 WHERE Card_ID = '%d'" %num)
     return
  def add_OUT_local(): #increase Card_ID to distinguish second check
     with db_local:
           cur = db_local.cursor(MySQLdb.cursors.DictCursor) 
           cur.execute("UPDATE tt_control SET Card_ID = Card_ID + 1 WHERE Card_ID = '%d'" %num)
     return
  while 1:
     num = RFID(chuoi)
     time.sleep(1)
     Have_ID =have_ID(num)
     if(num != ID_rong):
            if(Have_ID ==""):
                 add_database() #---> it will error if remote broken, how can i fix it?
                 add_database_local()
            else:
                 add_time_out() #---> it will error if remote broken, how can i fix it? I think connection alive can fix, but I don't know
                 add_time_out_local()
                 add_OUT()
                 add_OUT_local() #---> it will error if remote broken, how can i fix it?

您有两个选择:

(不太好)定期Ping服务器以保持连接活动

(最好)在调用cur.execute时处理MySQLdb异常,方法是重新建立连接并重试调用。从那篇文章中,您可以自己处理异常:

def __execute_sql(self,sql,cursor):
    try:
        cursor.execute(sql)
        return 1
    except MySQLdb.OperationalError, e:            
        if e[0] == 2006:
            self.logger.do_logging('info','DB', "%s : Restarting db" %(e))
            self.start_database()
            return 0

(最后)在实际调用数据库条目之前建立一个新的数据库连接。在这种情况下,将
db
db_local
定义移动到
光标前调用的函数中。如果您要进行数千次查询,这不是最好的。但是,如果只是几个数据库查询,则可能没问题。

我使用以下方法:

def checkConn(self):
    sq = "SELECT NOW()"
    try:
        self.cur.execute( sq )
    except pymysql.Error as e:
        if e.errno == 2006:
            return self.connect()
        else:
            print ( "No connection with database." )
            return False

我用了一个简单的技巧。最初,我使用以下方式连接到DB:

conect = mysql.connector.connect(host=DB_HOST, user=DB_USER, password=DB_PASS, database=DB_NAME)
每当我需要检查DB是否仍然连接时,我都会使用一条线路:

conect.ping(reconnect=True, attempts=3, delay=2)

这将检查数据库连接是否仍处于活动状态。如果没有,它将重新启动连接,从而解决问题。

您能否更具体一点,准确地描述您想要的是什么,并仅描述该部分?如果数据库的运行方式不符合您的要求,您需要的是异常处理,那么请先阅读异常处理(,),谢谢。我想制作客户端和服务器系统。客户端为本地mysql,服务器为远程mysql。在我的代码中,若服务器并没有坏掉,它将向服务器和客户端添加卡ID、输入时间和输出时间。但若服务器坏了,它就会出错并停止。我想当服务器坏了,代码仍然工作,并跳过添加数据到服务器。它只会添加到客户端。(对不起,我的英语不太好)ModuleNotFoundError:没有名为“pymysql”的模块是最糟糕的做法。如果数据库一开始就没有连接怎么办?@lone_coder我们应该在连接后检查我们的连接。^仅仅因为他们没有在回答中添加这一点,并不意味着他建议人们不要这样做。这超出了这个问题的范围这比公认的答案要好,因为它是内置选项。