-1

对于西班牙语的代码,我很抱歉,但除此之外,您应该能够理解它的结构,我使用的是 Python 3.3.2 并在这里遇到问题。

leyendoestadisticas = open("listas\Estadisticas.txt", "r")
bufferestadisticas = leyendoestadisticas.read()
leyendoestadisticas.close()
if not '"'+user.name+'"' in bufferestadisticas: #If name is not found, do this
  escribiendoestadisticas = open("listas\Estadisticas.txt", 'a')
  escribiendoestadisticas.write(json.dumps([user.name, palabrasdelafrase, letrasdelafrase,
                                            "1", user.nameColor, user.fontColor, user.fontFace, user.fontSize, message.body, room.name])+"\n")
  escribiendoestadisticas.close()
else: #If name is found...
  data = []
  with open('listas\Estadisticas.txt', 'r+') as f:
    for line in f:
      data_line = json.loads(line)
      if data_line[0] == user.name: #if name matches...
        if data_line[9] == room.name: #And room also, then update info.
          data_line[1] = int(data_line[1])+int(palabrasdelafrase)
          data_line[2] = int(data_line[2])+int(letrasdelafrase)
          data_line[3] = int(data_line[3])+1
          data_line[4] = user.nameColor
          data_line[5] = user.fontColor
          data_line[6] = user.fontFace
          data_line[7] = user.fontSize
          data_line[8] = message.body
          data_line[9] = room.name
        else: #but if name is found and room doesn't matches. #PROBLEM HERE
          escribiendoestadisticas = open("listas\Estadisticas.txt", 'a')
          escribiendoestadisticas.write(json.dumps([user.name, palabrasdelafrase, letrasdelafrase,
                                                  "1", user.nameColor, user.fontColor, user.fontFace, user.fontSize, message.body, room.name])+"\n")
          escribiendoestadisticas.close()
      data.append(data_line)
      break
    f.seek(0)
    f.writelines(["%s\n" % json.dumps(i) for i in data])
    f.truncate()

它的目的是添加一个 user.name 并更新它的信息,只要该行的属性 9 匹配。如果不匹配,则使用相同的 user.name 但属性 9 的新值将新行添加到列表中。

如果属性 9 匹配,它会完美运行,但如果不匹配,它所做的是覆盖该 user.name 的现有值并从头开始计算 [1]、[2] 和 [3] 中的属性。

谢谢您的帮助。

编辑:使用答案中的信息进行更新:

else: 
  data = []
  with open('listas\Estadisticas.txt', 'r+') as f:
    lines = f.readlines()
  for line in lines:
    data_line = json.loads(line)
    if data_line[0] == user.name: 
      if data_line[9] == room.name:
        data_line[1] = int(data_line[1])+int(palabrasdelafrase)
        data_line[2] = int(data_line[2])+int(letrasdelafrase)
        data_line[3] = int(data_line[3])+1
        data_line[4] = user.nameColor
        data_line[5] = user.fontColor
        data_line[6] = user.fontFace
        data_line[7] = user.fontSize
        data_line[8] = message.body
        data_line[9] = room.name
      data.append(data_line)
  f.seek(0)
  f.writelines(["%s\n" % json.dumps(i) for i in data])
  f.truncate()
  for line in lines: 
    data_line = json.loads(line)
    if data_line[0] == user.name: 
      if not data_line[9] == room.name:
        escribiendoestadisticas = open("listas\Estadisticas.txt", 'a')
        escribiendoestadisticas.write(json.dumps([user.name, palabrasdelafrase, letrasdelafrase,
                                                "1", user.nameColor, user.fontColor, user.fontFace, user.fontSize, message.body, room.name])+"\n")
        escribiendoestadisticas.close()

但现在我收到错误:ValueError: I/O operation on closed file。完全迷失了。

4

1 回答 1

0

看起来您正在修改文件,同时仍在读取它。首先完全加载文件,然后让上下文处理程序关闭文件:

else: #Si está, suma datos
  data = []
  with open('listas\Estadisticas.txt', 'r+') as f:
      lines = f.readlines()

  for line in lines:
    data_line = json.loads(line)
    if data_line[0] == user.name: #if name is found...
      if data_line[9] == room.name: #And room also, then update info.
        data_line[1] = int(data_line[1])+int(palabrasdelafrase)
        [etc]

正如其他人在评论中指出的那样,您可能想要搜索整个文件user.name,而不仅仅是逐行搜索。

此外,正如@abarnert 指出的那样,只需data在循环中进行修改,然后在最后写出完整的信息集。


几个一般性意见:

  • data_line当包含少于 10 个项目时,您的代码不会处理这种情况。如果发生这种情况,您的程序将崩溃。

  • 您最好使用json将整个文件保存dict到文件中。这样,当您加载 时dict,您可以立即检查并访问user.name. 我建议你制作user.name字典的键。

它可能看起来像这样(编辑:看到你想要 user.name 和 room.name 的独特组合):

with open('listas\Estadisticas.json', 'rb') as f:
    data = json.load(f)

key = (user.name, room.name)

if key in data:
    data[key]['nameColor'] = user.nameColor
    [...etc...]
else:
    # I am not sure what details are different
    data[key] = {} # New sub-dict
    data[key]['nameColor'] = user.nameColor
    [...etc...]

with open('listas\Estadisticas.json', 'wb') as f:
    json.dump(f, data)
  • 名称可以更改。这就是为什么数据库通常使用不是人名或房间名称的键。我建议您将事物的名称分配给其他抽象的东西,例如数字,然后在需要显示时查找名称。如果一个用户的名字最初输入错误,后来他们想更正他们名字的拼写,你将很难向他们解释你不能,因为这意味着他们所有的数据都将丢失......
于 2013-09-17T21:58:49.017 回答