3

SQLサーバーからAPIとしてデータを返す小さなMVC4アプリを作成しました。ローカルURLはhttp://10.0.0.116:35577/api/articoli(ローカルIISインスタンスへのネットワークアクセスを既に有効にしています。これにより、理解するまで1日中停止しました)、返される形式は次のとおりです。

[{"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;

}
}

サーバーアプリケーションがデータ挿入の試みとして解釈し、間違ったメソッドを呼び出したPOSTを使用してデータを取得しようとしたため、パーサーのURL取得部分を編集する必要がありました。

これはパーサーが呼び出されるコードの一部です。各要素を取得し、各要素をリストビューの行として配置します。次にクリックすると、単一の要素を読み取るアクティビティが呼び出されます。

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(またはプリミティブの1つ)のいずれかです。あなたは常にそれをオブジェクトとして解析しています:

// 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)ドキュメントのルート要素が配列の場合は、として解析する必要があります。これは、次を使用して安全に行うことができます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()。タイプをからJSONObjectに変更するだけです。JSONArray

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;

JSONObjectおよびからへのクラスの戻りタイプJSONArray

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