Python получает последнюю строчку
Работая над проектом мониторинга, мне нужно считать импульсы от измерителей пульса. Я уже нашел несколько решений, которые я пытался адаптировать к своим потребностям.
Вот скрипт Python, работающий на Raspberry Pi:
#!/usr/bin/env python
import RPi.GPIO as GPIO
import datetime
import sys
import signal
from subprocess import check_output
#verbose = True # global variable
############################################################################################################
############################################################################################################
def printusage(progname):
print progname + ' <gpio-pin-number> <filename> [debug]'
print 'Example usage: '
print progname + ' 23 /path/to/mylogfile'
print progname + ' 23 /path/to/mylogfile debug'
sys.exit(-1)
def signal_handler(signal, frame):
if verbose:
print('You pressed Ctrl+C, so exiting')
GPIO.cleanup()
sys.exit(0)
def readvalue(myworkfile):
try:
f = open(myworkfile, 'ab+') # open for reading. If it does not exist, create it
line=subprocess.check_output(['tail','-f','-1',f])
elmts=line.split(" ")
value = int(elmts[2])
except:
value = 0 # if something went wrong, reset to 0
#print "old value is", value
f.close() # close for reading
return value
def writevalue(myworkfile,value):
f = open(myworkfile, 'a')
f.write((str(datetime.datetime.now())+' '+str(value)+'\r\n')) # timestamp
f.close()
############################################################################################################
############################################################################################################
######### Initialization
#### get input parameters:
try:
mygpiopin = int(sys.argv[1])
logfile = sys.argv[2]
except:
printusage(sys.argv[0])
verbose = False
try:
if sys.argv[3] == 'debug':
verbose = True
print "Verbose is On"
else:
printusage(sys.argv[0])
except:
pass
#### if verbose, print some info to stdout
if verbose:
print "GPIO is " + str(mygpiopin)
print "Logfile is " + logfile
print "Current value is " + str(readvalue(logfile))
#### setup
GPIO.setmode(GPIO.BCM)
GPIO.setup(mygpiopin, GPIO.IN, pull_up_down = GPIO.PUD_DOWN)
signal.signal(signal.SIGINT, signal_handler) # SIGINT = interrupt by CTRL-C
########## Main Loop
while True:
# wait for pin going up
GPIO.wait_for_edge(mygpiopin, GPIO.RISING)
# read value from file
counter=readvalue(logfile) + 1
if verbose:
print "New value is", counter
# write value to file
writevalue(logfile,counter)
# and wait for pin going down
GPIO.wait_for_edge(mygpiopin, GPIO.FALLING)
############################################################################################################
############################################################################################################
Я хочу зарегистрировать последний индекс и увеличить его, но все, что я до сих пор тестировал, оставляет цикл застрявшим с индексом 1.
Я не могу использовать "более тяжелый" метод для поиска последней строки, например, просматривая весь файл, потому что он будет становиться все тяжелее и тяжелее с течением времени, и я не могу пропустить импульс.
Я довольно новичок в программировании, так что спасибо за вашу помощь!
РЕДАКТИРОВАТЬ: файл результатов выглядит следующим образом:
2016-10-08 16:54:23.072469 1
2016-10-08 16:54:23.462465 1
2016-10-08 16:54:23.777977 1
2016-10-08 16:54:24.010045 1
2016-10-08 16:54:24.194032 1
2016-10-08 16:54:24.388120 1
2016-10-08 16:54:24.549389 1
2016-10-08 16:54:24.737994 1
2016-10-08 16:54:24.959462 1
2016-10-08 16:54:25.164638 1
2016-10-08 16:54:25.351850 1
2016-10-08 16:54:25.536655 1
2016-10-08 16:54:25.716214 1
2016-10-08 16:54:25.794152 1
2016-10-08 17:06:13.506531 1
2016-10-08 17:06:14.097642 1
2016-10-08 17:06:14.211579 1
2016-10-08 17:06:15.237852 1
2016-10-08 17:06:15.752239 1
2016-10-08 17:06:16.320419 1
2016-10-08 17:06:16.842906 1
2016-10-08 17:06:17.391121 1
2016-10-08 17:06:17.851521 1
2016-10-08 17:06:18.444486 1
2016-10-08 17:06:18.858358 1
3 ответа
Чтобы прочитать последнюю строку большого файла, выберите позицию в конце файла и прочитайте. Если там нет новых строк, вернитесь немного назад и прочитайте снова.
Затем найдите последнюю новую строку, а остальное ваша последняя строка.
EXPECTED_MAX_LINE_SIZE = 500 # this should be both small enough and big enough
def get_last_line(f):
end = f.seek(0, os.SEEK_END) # detect file size
pos = end-EXPECTED_MAX_LINE_SIZE
if pos < 0: pos = 0
f.seek(pos, os.SEEK_SET)
end_of_file = f.read(EXPECTED_MAX_LINE_SIZE)
# TODO: check if '\n' is not there and read some more
return end_of_file.splitlines()[-1]
Ваша логика слишком сложна, если вы будете запускать скрипт разное время, просто сохраните последний индекс в отдельном файле, постоянно перезаписывая:
def next_index(index):
with open(index, 'a+') as f: # open for reading. If it does not exist, create it
val = int(next(f, -1)) + 1 # first run value will be 0
open(index,"w").write(str(val)) # overwrite
return val
если значение создано в другом месте, просто сделайте запись там, просто убедитесь, что вы перезаписали предыдущее значение, открывая w
,
Я не думаю, что именно это имел в виду Падрейк Каннингем, но мне удалось решить мою проблему, используя два файла: один для записи обновленного значения индекса, а другой для хранения значений:
#!/usr/bin/env python
import RPi.GPIO as GPIO
import datetime
import sys
import signal
#verbose = True # global debug variable
############################################################################################################
def printusage(progname): #show how to use the script
print progname + ' <gpio-pin-number> <index file> <store file> [debug]'
print 'Example usage: '
print progname + ' 23 /path/to/mylogfile /path/to/storefile'
print progname + ' 23 /path/to/mylogfile /path/to/storefile debug'
sys.exit(-1)
def signal_handler(signal, frame): #exiting the script
if verbose:
print('You pressed Ctrl+C, so exiting')
GPIO.cleanup()
sys.exit(0)
def readvalue(myworkfile):
try:
f = open(myworkfile, 'ab+') # open for reading. If it does not exist, create it
line=f.readline() #read the first and only line of the file
elmts=line.split(" ") # the format is <yyyy-mm-dd hh:mm:ss.ssssss indexvalue>
value = int(elmts[2]) #get the index value
except:
value = 0 # if something went wrong, reset to 0
#print "old value is", value
f.close() # close for reading
return value
def writevalue(myworkfile,value):
f = open(myworkfile, 'w') #overwrite the value
f.write((str(datetime.datetime.now())+' '+str(value))) # timestamp + data
f.close()
def store_value(index_file,index_value):
f=open(index_file, 'a+')
f.write(str(datetime.datetime.now())+' '+str(index_value)+'\r\n') #store timestamp + data
f.close()
############################################################################################################
######### Initialization
#### get input parameters
try:
mygpiopin = int(sys.argv[1])
logfile = sys.argv[2]
storefile=sys.argv[3]
except:
printusage(sys.argv[0])
verbose = False
try:
if sys.argv[4] == 'debug':
verbose = True
print "Verbose is On"
else:
printusage(sys.argv[0])
except:
pass
if verbose: #### if verbose, print some info to stdout
print "GPIO is " + str(mygpiopin)
print "Logfile is " + logfile
print "Storefile is " + storefile
print "Current value is " + str(readvalue(logfile))
#### SETUP
GPIO.setmode(GPIO.BCM)
GPIO.setup(mygpiopin, GPIO.IN, pull_up_down = GPIO.PUD_DOWN)
signal.signal(signal.SIGINT, signal_handler) # SIGINT = interrupt by CTRL-C
########## Main Loop #####
while True:
GPIO.wait_for_edge(mygpiopin, GPIO.RISING) # wait for pin going up
counter=readvalue(logfile) + 1 # read value from file and increment index
if verbose:
print "New value is", counter
writevalue(logfile,counter) # write value to logfile
store_value(storefile,counter) # store value in storefile
GPIO.wait_for_edge(mygpiopin, GPIO.FALLING) # and wait for pin going down
############################################################################################################
Спасибо за вашу помощь!