3

我制作了一个小型 MVC4 应用程序,它以 API 的形式从 SQL 服务器返回数据,本地 url 是http://10.0.0.116:35577/api/articoli(我已经启用了对本地 IIS 实例的网络访问,这实际上阻止了我一整天,直到我弄清楚了),它返回的格式如下:

[{"id":0,"desc":"Pizza"},{"id":1,"desc":"Pasta"},{"id":2,"desc":"Science"},{"id":3,"desc":"spaghetti"},{"id":4,"desc":"sfas"},{"id":5,"desc":"test"}]

调用http://10.0.0.116:35577/api/articoli/1返回单个元素:

{"id":1,"desc":"Pasta"}

我一直无法从我的 android 应用程序中解析它,从我收集到的数组应该有一个标题名称或其他东西,这里完全不存在,我真的不知道如何在这里添加它,或者如果我真的需要一个。

从我可以收集到的从教程中获得的解析器类不适合这种数据,我真的不知道如何使它与这样的结果一起工作。

解析器如下:

public class JSONParser {

static InputStream is = null;
static JSONObject jObj = null;
static String json = "";

// constructor
public JSONParser() {

}

public JSONObject getJSONFromUrl(String url) {

    // Making HTTP request
    try {

        URL aaa = new URL(url);
        URLConnection uConnection = aaa.openConnection();
        uConnection.connect();


        //is = httpEntity.getContent();
        is = aaa.openStream();

    } catch (UnsupportedEncodingException e) {
        e.printStackTrace();
    } catch (ClientProtocolException e) {
        e.printStackTrace();
    } catch (IOException e) {
        e.printStackTrace();
    }

    try {
        BufferedReader reader = new BufferedReader(new InputStreamReader(
                is, /*"UTF-8"*/ "iso-8859-1"), 8);
        StringBuilder sb = new StringBuilder();
        String line = null;
        while ((line = reader.readLine()) != null) {
            sb.append(line + "\n");
        }
        is.close();
        json = sb.toString();
    } catch (Exception e) {
        Log.e("Buffer Error", "Error converting result " + e.toString());
    }

    // try parse the string to a JSON object
    try {
        jObj = new JSONObject(json);
    } catch (JSONException e) {
        Log.e("JSON Parser", "Error parsing data " + e.toString());
    }

    // return JSON String
    return jObj;

}
}

我不得不编辑解析器的 url 检索部分,因为它试图通过使用 POST 来获取数据,服务器应用程序将其解释为尝试插入数据并调用了错误的方法,

这是一段调用解析器的代码,它应该检索每个元素,然后将每个元素作为一行放在列表视图中,然后单击它应该调用一个读取单个元素的活动:

private void fillDataJSON() {
        JSONParser jParser = new JSONParser();

        JSONObject json = jParser.getJSONFromUrl(URL);
        JSONArray articoli;
        try {
            articoli = json.getJSONArray("Articoli");

            if (articoli == null) {
                throw new NullPointerException();
            }

            for(int i = 0; i< articoli.length(); i++) {
                JSONObject c = articoli.getJSONObject(i);

                String id = c.getString(KEY_ID);
                String desc = c.getString(KEY_DESC);

                HashMap<String, String> map = new HashMap<String, String>();

                map.put(KEY_ID, id);
                map.put(KEY_DESC, desc);

                menuItems.add(map);

                adapter = new SimpleAdapter(ApiTest.this, menuItems,
                        R.layout.activity_api_test_row,
                        new String[] { KEY_DESC },
                        new int[] { R.id.TextView01 });
            }
        } catch (JSONException e) {
            e.printStackTrace();
        }
    }

我得到的错误如下:

03-13 10:05:33.291: E/JSON Parser(11811): Error parsing data org.json.JSONException: Value [{"id":0,"desc":"Pizza"},{"id":1,"desc":"Pasta"},{"id":2,"desc":"Science"},{"id":3,"desc":"spaghetti"},{"id":4,"desc":"sfas"},{"id":5,"desc":"test"}] of type org.json.JSONArray cannot be converted to JSONObject

这可以在android端修复还是我必须改变我的服务器的工作方式?

服务器只是通过控制器返回数据,其他一切都留给框架,例如:

 public IEnumerable<Articoli> GetAllArticoli()
    {
        return repository.GetAll();
    }

除此以外不做任何事情

4

2 回答 2

3

JSON 文档的根元素可以是 JSONObject 或 JSONArray(或原语之一)。您总是将其解析为对象:

// try parse the string to a JSON object
try {
    jObj = new JSONObject(json);
} catch (JSONException e) {
    Log.e("JSON Parser", "Error parsing data " + e.toString());
}

new JSONArray(json)当文档的根元素是一个数组时,您应该将其解析为 a 。您可以使用以下方法安全地执行此操作JSONTokener

// try parse the string to a JSON object
try {
    JSONTokener jt = new JSONTokener(json);
    Object rootElement = jt.nextValue();
    if (rootElement instanceof JSONObject) {
       // You got an object from the response
    } else if (rootElement instanceof JSONArray) {
       // You got a JSON array
    }
    // else... -> it can also be String, Boolean, Integer, Long or Double
} catch (JSONException e) {
    Log.e("JSON Parser", "Error parsing data " + e.toString());
}

如果您知道自己的要求,那么使用正确的响应类型会更容易。

于 2013-03-13T09:53:19.330 回答
1

您可以简单地在里面更改代码。只需fillDataJSON()将类型从更改JSONObjectJSONArray

JSONArray json = jParser.getJSONFromUrl(URL);
        JSONArray articoli;
        try {
            articoli = json;

并在您JSONParser更改为:

// try parse the string to a JSON object
    try {
        jObj = new JSONArray(json);
    } catch (JSONException e) {
        Log.e("JSON Parser", "Error parsing data " + e.toString());
    }

并将 jObj 的类型更改为 JSONArray

static JSONArray jObj = null;

和类的返回类型 from JSONObjecttoJSONArray

public JSONObject getJSONFromUrl(String url) {
于 2013-03-13T10:04:58.997 回答