Проблемы с превращением данных json в переменную в Python

Я работал над образовательным проектом, небольшая его часть требует, чтобы я преобразовал одну строку данных json в переменную в python 3, которую я получаю от domoticz (внешнее программное обеспечение с открытым исходным кодом), однако из-за моего уровня навыков работы с json я возникли некоторые проблемы, и я не совсем уверен, что я делаю не так. Я каждый раз получал ответ 200, поэтому я предполагаю, что из того, что я понял, это означает, что проблема не в соединении, а в коде Python. (Я подвергал цензуре адреса, но они верны.)

The code im using:
import time
import re
import requests
from ctypes import c_int, c_char_p, byref, sizeof, c_uint16, c_int32, c_byte
from ctypes import c_void_p
from datetime import datetime
import os
import urllib.request
import json
import logging
import sys
from requests.exceptions import HTTPError
logger = logging.getLogger(__name__)
domoticzserver='ip'
switchid='3'
device='5'
tempbed=str(4)
def domoticzrequest (url):
  request = urllib.request.Request(url)
  response = urllib.request.urlopen(request)
  return response.read()
import urllib.request, json 
with urllib.request.urlopen("http://domoticzip/json.htm?type=devices&rid=4") as url:
data = json.loads(url.read().decode())
    print(data)

Я возвращаю json, который я могу увидеть, набрав, щелкнув URL-адрес в python:

{
    "ActTime" : 1606722346,
    "AstrTwilightEnd" : "18:37",
    "AstrTwilightStart" : "06:23",
    "CivTwilightEnd" : "17:14",
    "CivTwilightStart" : "07:47",
    "DayLength" : "08:08",
    "NautTwilightEnd" : "17:56",
    "NautTwilightStart" : "07:04",
    "ServerTime" : "2020-11-30 08:45:46",
    "SunAtSouth" : "12:30",
    "Sunrise" : "08:26",
    "Sunset" : "16:34",
    "app_version" : "2020.2",
    "result" : 
    [
        {
            "AddjMulti" : 1.0,
            "AddjMulti2" : 1.0,
            "AddjValue" : 0.0,
            "AddjValue2" : 0.0,
            "BatteryLevel" : 255,
            "CustomImage" : 0,
            "Data" : "Normal",
            "Description" : "",
            "Favorite" : 0,
            "HardwareID" : 1,
            "HardwareName" : "Domoticz Internal",
            "HardwareType" : "Domoticz Internal interface",
            "HardwareTypeVal" : 67,
            "HaveDimmer" : false,
            "HaveGroupCmd" : false,
            "HaveTimeout" : false,
            "ID" : "148702",
            "LastUpdate" : "2020-10-19 15:10:02",
            "MaxDimLevel" : 0,
            "Name" : "Domoticz Security Panel",
            "Notifications" : "false",
            "PlanID" : "0",
            "PlanIDs" : 
            [
                0
            ],
            "Protected" : false,
            "ShowNotifications" : true,
            "SignalLevel" : "-",
            "Status" : "Normal",
            "StrParam1" : "",
            "StrParam2" : "",
            "SubType" : "Security Panel",
            "SwitchType" : "Security",
            "SwitchTypeVal" : 0,
            "Timers" : "false",
            "Type" : "Security",
            "TypeImg" : "security",
            "Unit" : 0,
            "Used" : 0,
            "XOffset" : "0",
            "YOffset" : "0",
            "idx" : "2"
        },
        {
            "AddjMulti" : 1.0,
            "AddjMulti2" : 1.0,
            "AddjValue" : 0.0,
            "AddjValue2" : 0.0,
            "BatteryLevel" : 255,
            "CustomImage" : 0,
            "Data" : "-5.0 C",
            "Description" : "",
            "Favorite" : 1,
            "HardwareID" : 2,
            "HardwareName" : "Test sensor",
            "HardwareType" : "Dummy (Does nothing, use for virtual switches only)",
            "HardwareTypeVal" : 15,
            "HaveTimeout" : true,
            "ID" : "14053",
            "LastUpdate" : "2020-11-09 09:03:34",
            "Name" : "Temperatuur Kachel",
            "Notifications" : "false",
            "PlanID" : "0",
            "PlanIDs" : 
            [
                0
            ],
            "Protected" : false,
            "ShowNotifications" : true,
            "SignalLevel" : "-",
            "SubType" : "LaCrosse TX3",
            "Temp" : -5.0,
            "Timers" : "false",
            "Type" : "Temp",
            "TypeImg" : "temperature",
            "Unit" : 1,
            "Used" : 1,
            "XOffset" : "0",
            "YOffset" : "0",
            "idx" : "3",
            "trend" : 0
        },
        {
            "AddjMulti" : 1.0,
            "AddjMulti2" : 1.0,
            "AddjValue" : 0.0,
            "AddjValue2" : 0.0,
            "BatteryLevel" : 255,
            "CustomImage" : 0,
            "Data" : "17.5",
            "Description" : "",
            "Favorite" : 1,
            "HardwareID" : 3,
            "HardwareName" : "Test switch",
            "HardwareType" : "Dummy (Does nothing, use for virtual switches only)",
            "HardwareTypeVal" : 15,
            "HaveTimeout" : true,
            "ID" : "0014054",
            "LastUpdate" : "2020-11-06 11:51:09",
            "Name" : "Temperatuur gewenst",
            "Notifications" : "false",
            "PlanID" : "0",
            "PlanIDs" : 
            [
                0
            ],
            "Protected" : false,
            "SetPoint" : "17.5",
            "ShowNotifications" : true,
            "SignalLevel" : "-",
            "SubType" : "SetPoint",
            "Timers" : "false",
            "Type" : "Thermostat",
            "TypeImg" : "override_mini",
            "Unit" : 1,
            "Used" : 1,
            "XOffset" : "0",
            "YOffset" : "0",
            "idx" : "4"
        }
    ],
    "status" : "OK",
    "title" : "Devices"
}

В основном я хочу SetPoint(на последней вкладке текста также прямо над этим), который в этом экземпляре 17.5 как переменная в python, и я сделаю цикл кода python, чтобы он каждый раз получал этот URL-адрес json для обновления состояния заданного значения. Но у меня проблемы с захватом 17,5, чтобы превратить его в переменную. Я получаю весь json, как этот код. или я закончу тем, что получу все прошлое и включу заданное значение, если я что-то изменю. Кто-нибудь знает, что я делаю не так и, возможно, где я должен искать решение? Я немного неопытен с json-частью python, и я не знаю, с чего начать, поскольку код, который я нашел и пытался, похоже, не работает или дает мне ошибки. Большое спасибо за уделенное время!

2 ответа

Решение

Вы сохраняете свои данные в data ={"key": argument}как словарь. Если вы хотите получить доступ к определенному значению, вы должны его вызвать. в твоем случае:

SetPoint = float(data["result"][-1]["SetPoint"])

Чтобы разбить это:

data["result"] # gives you a list which elements are dictionaries.
the [-1] # calls for the last element in you list which contains the SetPoint
["SetPoint"] # then calls for the SetPoint Value which is a String "17.5"
float(...) converts the string to a float value

Json .loads возвращает словарь Python, поэтому, возможно, что-то вроде этого будет:

result = json['result']
set_point = 0.0 

for res in result:
   if 'SetPoint' in res:
        set_point = res['SetPoint']
Другие вопросы по тегам