it-roy-ru.com

Лучший способ получить значения переменных из текстового файла - Python - Json

Ссылаясь на этот вопрос , у меня есть аналогичная -но не та же проблема ..

По пути у меня будет текстовый файл, структурированный как:

var_a: 'home'
var_b: 'car'
var_c: 15.5

И мне нужно, чтобы python прочитал файл, а затем создал переменную с именем var_a со значением 'home' и так далее.

Пример:

#python stuff over here
getVarFromFile(filename) #this is the function that im looking for
print var_b
#output: car, as string
print var_c
#output 15.5, as number.

Возможно ли, я имею в виду, даже сохранить тип var?

Обратите внимание, что у меня есть полная свобода в структуре текстового файла, я могу использовать формат, который мне нравится, если предложенный мной вариант не самый лучший.

EDIT: ConfigParser может быть решением, но мне это не очень нравится, потому что в моем скрипте мне нужно будет ссылаться на переменные в файле с помощью

config.get("set", "var_name")

Но что я буду любить, так это ссылаться на переменную напрямую, как я объявил в скрипте Python ...

Есть ли способ импортировать файл как словарь Python?

О, наконец, помните, что я точно не знаю, сколько переменных у меня будет в текстовом файле.

Edit 2: я очень заинтересован в решении JSON Стефана, потому что таким образом текстовый файл может быть просто прочитан на других языках (например, PHP, затем через AJAX JavaScript), но мне не удается в чем-то, действуя это решение:

#for the example, i dont load the file but create a var with the supposed file content
file_content = "'var_a': 4, 'var_b': 'a string'"
mydict = dict(file_content)
#Error: ValueError: dictionary update sequence element #0 has length 1; 2 is required
file_content_2 = "{'var_a': 4, 'var_b': 'a string'}"
mydict_2 = dict(json.dump(file_content_2, True))
#Error:
#Traceback (most recent call last):
#File "<pyshell#5>", line 1, in <module>
#mydict_2 = dict(json.dump(file_content_2, True))
#File "C:\Python26\lib\json\__init__.py", line 181, in dump
#fp.write(chunk)
#AttributeError: 'bool' object has no attribute 'write'

В какие проблемы я могу попасть с форматом JSON? И как я могу прочитать массив JSON в текстовом файле и преобразовать его в python?

П.С .: Мне не нравится решение с использованием файлов .py; Я предпочитаю .txt, .inc,. Что бы не ограничивало один язык.

34
Strae

Загрузите ваш файл с JSON или PyYAML в словарь the_dict (см. Документацию для JSON или PyYAML для этого шага, оба могут хранить тип данных) и добавьте словарь в свой глобальный словарь, например, используя globals().update(the_dict).

Если вы хотите вместо этого в локальном словаре (например, внутри функции), вы можете сделать это следующим образом:

for (n, v) in the_dict.items():
    exec('%s=%s' % (n, repr(v)))

до тех пор, пока это безопасно использовать exec. Если нет, вы можете использовать словарь напрямую.

14
stephan

Но то, что я буду любить, это ссылаться на переменную direclty, как я объявил это в скрипте Python ..

Предполагая, что вы готовы немного изменить свой синтаксис, просто используйте python и импортируйте модуль «config».

# myconfig.py:

var_a = 'home'
var_b = 'car'
var_c = 15.5

Тогда делай 

from myconfig import *

И вы можете ссылаться на них по имени в вашем текущем контексте.

59
hbn

Используйте ConfigParser.

Ваш конфиг:

[myvars]
var_a: 'home'
var_b: 'car'
var_c: 15.5

Ваш код Python:

import ConfigParser

config = ConfigParser.ConfigParser()
config.read("config.ini")
var_a = config.get("myvars", "var_a")
var_b = config.get("myvars", "var_b")
var_c = config.get("myvars", "var_c")
24
Igor Krivokon

Вы можете рассматривать ваш текстовый файл как модуль Python и загружать его динамически, используя imp.load_source :

import imp
imp.load_source( name, pathname[, file]) 

Пример:

// mydata.txt
var1 = 'hi'
var2 = 'how are you?'
var3 = { 1:'elem1', 2:'elem2' }
//...

// In your script file
def getVarFromFile(filename):
    import imp
    f = open(filename)
    global data
    data = imp.load_source('data', '', f)
    f.close()

# path to "config" file
getVarFromFile('c:/mydata.txt')
print data.var1
print data.var2
print data.var3
...
18
Nick Dandoulakis

Другие решения, размещенные здесь, не работают для меня, потому что: 

  • мне просто нужны параметры из файла для нормального скрипта
  • import * не работает для меня, так как мне нужен способ переопределить их, выбрав другой файл
  • Просто файл с надписью не был в порядке, так как мне нужны были комментарии.

В итоге я использовал Configparser и globals().update()

Тестовый файл:

#File parametertest.cfg:
[Settings]
#Comments are no Problem
test= True
bla= False    #Here neither

#that neither

И это мой демонстрационный скрипт:

import ConfigParser

cfg = ConfigParser.RawConfigParser()
cfg.read('parametertest.cfg')       # Read file

#print cfg.getboolean('Settings','bla') # Manual Way to acess them

par=dict(cfg.items("Settings"))
for p in par:
    par[p]=par[p].split("#",1)[0].strip() # To get rid of inline comments

globals().update(par)  #Make them availible globally

print bla

Это только для файла с одним разделом сейчас, но это будет легко принять.

Надеюсь, это будет полезно для кого-то :)

5
SolarSimon

Предположим, что у вас есть файл с именем «test.txt» с:

a=1.251
b=2.65415
c=3.54
d=549.5645
e=4684.65489

И вы хотите найти переменную (a, b, c, d или e):

ffile=open('test.txt','r').read()

variable=raw_input('Wich is the variable you are looking for?\n')

ini=ffile.find(variable)+(len(variable)+1)
rest=ffile[ini:]
search_enter=rest.find('\n')
number=float(rest[:search_enter])

print "value:",number
4
JorgeGarza

Насколько надежен ваш формат? Если разделитель всегда точно равен «:», работает следующее. Если нет, то сравнительно простое регулярное выражение должно делать эту работу.

Пока вы работаете с довольно простыми типами переменных, функция eval в Python делает сохранение переменных в файлах удивительно простым.

(Ниже приведен словарь, кстати, который вы упомянули как одно из ваших предпочтительных решений).

def read_config(filename):
    f = open(filename)
    config_dict = {}
    for lines in f:
        items = lines.split(': ', 1)
        config_dict[items[0]] = eval(items[1])
    return config_dict
3
mavnn

То, что вы хотите, кажется, хотите, это следующее, но это НЕ РЕКОМЕНДУЕТСЯ:

>>> for line in open('dangerous.txt'):
...     exec('%s = %s' % Tuple(line.split(':', 1)))
... 
>>> var_a
'home'

Это создает поведение, аналогичное PHP register_globals , и, следовательно, имеет те же проблемы безопасности. Кроме того, использование exec, которое я показал, позволяет выполнять произвольный код. Только используйте это, если вы абсолютно уверены, что содержимому текстового файла можно доверять при любых обстоятельствах.

Вы действительно должны рассмотреть возможность связывания переменных не с локальной областью, а с объектом, и использовать библиотеку, которая анализирует содержимое файла так, что код не выполняется. Итак: перейдите к любому из других решений, представленных здесь.

(Обратите внимание: я добавил этот ответ не как решение, а как явное не решение.)

1
Stephan202

ответ hbn не будет работать «из коробки», если файл для загрузки находится в подкаталоге или назван тире .

В таком случае вы можете рассмотреть эту альтернативу:

exec open(myconfig.py).read()

Или проще, но не рекомендуется в python3:

execfile(myconfig.py)

Я думаю, что предупреждение Stephan202 относится к обоим вариантам, и, возможно, цикл на линиях безопаснее.

0
Skippy le Grand Gourou