我正在尝试解析Android中的天气API中的Json数据。 但我得到错误。以下是logcat错误列表:
10-17 21:22:36.395: E/log_tag(304): Error in http connection java.net.UnknownHostException: api.openweathermap.org
10-17 21:22:36.395: E/log_tag(304): Error converting result java.lang.NullPointerException
10-17 21:22:36.425: E/log_tag(304): Error parsing data org.json.JSONException: End of input at character 0 of
10-17 21:22:36.447: E/AndroidRuntime(304): FATAL EXCEPTION: AsyncTask #1
10-17 21:22:36.447: E/AndroidRuntime(304): java.lang.RuntimeException: An error occured while executing doInBackground()
10-17 21:22:36.447: E/AndroidRuntime(304): at android.os.AsyncTask$3.done(AsyncTask.java:200)
10-17 21:22:36.447: E/AndroidRuntime(304): at java.util.concurrent.FutureTask$Sync.innerSetException(FutureTask.java:273)
10-17 21:22:36.447: E/AndroidRuntime(304): at java.util.concurrent.FutureTask.setException(FutureTask.java:124)
10-17 21:22:36.447: E/AndroidRuntime(304): at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask.java:307)
10-17 21:22:36.447: E/AndroidRuntime(304): at java.util.concurrent.FutureTask.run(FutureTask.java:137)
10-17 21:22:36.447: E/AndroidRuntime(304): at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1068)
10-17 21:22:36.447: E/AndroidRuntime(304): at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:561)
10-17 21:22:36.447: E/AndroidRuntime(304): at java.lang.Thread.run(Thread.java:1096)
10-17 21:22:36.447: E/AndroidRuntime(304): Caused by: java.lang.NullPointerException
10-17 21:22:36.447: E/AndroidRuntime(304): at com.example.work.MainActivity$DownloadJSON.doInBackground(MainActivity.java:63)
10-17 21:22:36.447: E/AndroidRuntime(304): at com.example.work.MainActivity$DownloadJSON.doInBackground(MainActivity.java:1)
10-17 21:22:36.447: E/AndroidRuntime(304): at android.os.AsyncTask$2.call(AsyncTask.java:185)
10-17 21:22:36.447: E/AndroidRuntime(304): at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask.java:305)
10-17 21:22:36.447: E/AndroidRuntime(304): ... 4 more
这是我的代码: MainActivity.java:
package com.example.work;
import java.util.ArrayList;
import java.util.HashMap;
import org.json.JSONArray;
import org.json.JSONException;
import org.json.JSONObject;
import android.app.Activity;
import android.app.ProgressDialog;
import android.os.AsyncTask;
import android.os.Bundle;
import android.util.Log;
import android.widget.ListView;
public class MainActivity extends Activity {
// Declare Variables
JSONObject jsonobject;
JSONArray jsonarray;
ListView listview;
ListViewAdapter adapter;
ProgressDialog mProgressDialog;
ArrayList<HashMap<String, String>> arraylist;
static String NAME = "rank";
static String TYPE = "country";
static String FLAG = "flag";
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
// Get the view from listview_main.xml
setContentView(R.layout.activity_main);
// Execute DownloadJSON AsyncTask
new DownloadJSON().execute();
}
// DownloadJSON AsyncTask
private class DownloadJSON extends AsyncTask<Void, Void, Void> {
//@Override
//protected void onPreExecute() {
// super.onPreExecute();
// Create a progressdialog
//mProgressDialog = new ProgressDialog(MainActivity.this);
// Set progressdialog message
//mProgressDialog.setMessage("Loading...");
//mProgressDialog.setIndeterminate(false);
// Show progressdialog
//mProgressDialog.show();
// }
@Override
protected Void doInBackground(Void... params) {
// Create an array
arraylist = new ArrayList<HashMap<String, String>>();
// Retrieve JSON Objects from the given URL address
jsonobject = JSONParser
.getJSONfromURL("http://api.openweathermap.org/data/2.5/weather?id=2172797");
try {
// Locate the array name in JSON
jsonarray = jsonobject.getJSONArray("id");
for (int i = 0; i < jsonarray.length(); i++) {
HashMap<String, String> map = new HashMap<String, String>();
jsonobject = jsonarray.getJSONObject(i);
// Retrive JSON Objects
map.put(MainActivity.NAME, jsonobject.getString("main"));
map.put(MainActivity.TYPE, jsonobject.getString("description"));
//map.put(MainActivity.FLAG, jsonobject.getString("restaurantIMAGE"));
// Set the JSON Objects into the array
arraylist.add(map);
}
} catch (JSONException e) {
Log.e("Error", e.getMessage());
e.printStackTrace();
}
return null;
}
@Override
protected void onPostExecute(Void args) {
// Locate the listview in listview_main.xml
listview = (ListView) findViewById(R.id.listview);
// Pass the results into ListViewAdapter.java
adapter = new ListViewAdapter(MainActivity.this, arraylist);
// Set the adapter to the ListView
listview.setAdapter(adapter);
// Close the progressdialog
mProgressDialog.dismiss();
}
}
}
JSONParser.java:
public class JSONParser {
public static JSONObject getJSONfromURL(String url) {
InputStream is = null;
String result = "";
JSONObject jArray = null;
// Download JSON data from URL
try {
HttpClient httpclient = new DefaultHttpClient();
HttpGet httpget = new HttpGet(url);
HttpResponse response = httpclient.execute(httpget);
HttpEntity entity = response.getEntity();
is = entity.getContent();
} catch (Exception e) {
Log.e("log_tag", "Error in http connection " + e.toString());
}
// Convert response to string
try {
BufferedReader reader = new BufferedReader(new InputStreamReader(
is, "iso-8859-1"), 8);
StringBuilder sb = new StringBuilder();
String line = null;
while ((line = reader.readLine()) != null) {
sb.append(line + "\n");
}
is.close();
result = sb.toString();
} catch (Exception e) {
Log.e("log_tag", "Error converting result " + e.toString());
}
try {
jArray = new JSONObject(result);
} catch (JSONException e) {
Log.e("log_tag", "Error parsing data " + e.toString());
}
return jArray;
}
}
ListViewAdapter.java:
public class ListViewAdapter extends BaseAdapter {
// Declare Variables
Context context;
LayoutInflater inflater;
ArrayList<HashMap<String, String>> data;
//ImageLoader imageLoader;
HashMap<String, String> resultp = new HashMap<String, String>();
public ListViewAdapter(Context context,
ArrayList<HashMap<String, String>> arraylist) {
this.context = context;
data = arraylist;
// imageLoader = new ImageLoader(context);
}
@Override
public int getCount() {
return data.size();
}
@Override
public Object getItem(int position) {
return null;
}
@Override
public long getItemId(int position) {
return 0;
}
public View getView(final int position, View convertView, ViewGroup parent) {
// Declare Variables
TextView main;
TextView description;
//ImageView flag;
inflater = (LayoutInflater) context
.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
View itemView = inflater.inflate(R.layout.listview_item, parent, false);
// Get the position
resultp = data.get(position);
// Locate the TextViews in listview_item.xml
main = (TextView) itemView.findViewById(R.id.main);
description = (TextView) itemView.findViewById(R.id.description);
// Locate the ImageView in listview_item.xml
//flag = (ImageView) itemView.findViewById(R.id.flag);
// Capture position and set results to the TextViews
main.setText(resultp.get(MainActivity.NAME));
description.setText(resultp.get(MainActivity.TYPE));
// Capture position and set results to the ImageView
// Passes flag images URL into ImageLoader.class
// imageLoader.DisplayImage(resultp.get(MainActivity.FLAG), flag);
// Capture ListView item click
itemView.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View arg0) {
// Get the position
resultp = data.get(position);
Intent intent = new Intent(context, SingleItemView.class);
// Pass all data rank
intent.putExtra("name", resultp.get(MainActivity.NAME));
// Pass all data country
intent.putExtra("type", resultp.get(MainActivity.TYPE));
// Pass all data flag
// intent.putExtra("flag", resultp.get(MainActivity.FLAG));
// Start SingleItemView Class
context.startActivity(intent);
}
});
return itemView;
}
}
SingleItemView.java:
public class SingleItemView {
Intent intent = getIntent();
String id = intent.getStringExtra("name");
String name = intent.getStringExtra("type");
private Intent getIntent() {
// TODO Auto-generated method stub
return null;
}
}
布局中的listview_item.xml:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical" >
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:id="@+id/main"/>
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:id="@+id/description"/>
</LinearLayout>
我正在按城市ID搜索天气数据。但是收到这些错误。
答案 0 :(得分:2)
使用 JSONLint 查看您的Json结构
我使用了您的网址“http://api.openweathermap.org/data/2.5/weather?id=2172797”
并获得Json如下
{
"coord": {
"lon": 145.766663,
"lat": -16.91667
},
"sys": {
"country": "AU",
"sunrise": 1381952820,
"sunset": 1381997847
},
"weather": [
{
"id": 803,
"main": "Clouds",
"description": "broken clouds",
"icon": "04n"
}
],
"base": "global stations",
"main": {
"temp": 297.15,
"pressure": 1015,
"humidity": 88,
"temp_min": 297.15,
"temp_max": 297.15
},
"wind": {
"speed": 2.1,
"deg": 350
},
"rain": {
"3h": 0
},
"clouds": {
"all": 75
},
"dt": 1382022000,
"id": 2172797,
"name": "Cairns",
"cod": 200
}
在这一行
jsonarray = jsonobject.getJSONArray("id");//Wrong Key for JsonArray
你弄错了
应该是
jsonarray = jsonobject.getJSONArray("weather");// weather is key for Array
编辑:
@Override
protected Void doInBackground(Void... arg0) {
String url="http://api.openweathermap.org/data/2.5/weather?id=2172797";
JSONObject json=parser.getJSONFromUrl(url);
try {
JSONObject child=json.getJSONObject("coord");
System.out.println("Child"+child.getString("lon"));
JSONArray jsonArray=json.getJSONArray("weather");
for(int i=0;i<jsonArray.length();i++)
{
JSONObject wether=jsonArray.getJSONObject(i);
System.out.println("Wether"+wether.getString("id")+" "+wether.getString("main"));
}
} catch (JSONException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
return null;
}
<强>输出: - 强>
10-17 16:41:58.582: I/System.out(1312): Child 145.76666
10-17 16:41:58.582: I/System.out(1312): Wether 803 Clouds
现在按照相同的步骤获取所需的所有值。
希望这会对你有所帮助。
答案 1 :(得分:0)
您收到UnknownHostException,表示您遇到连接问题。检查您的互联网连接,并检查您是否在应用清单中添加了INTERNET权限。
在JSONParser类中,您将从getJSONFromURL方法返回JSONObject。出于某种原因,最可能由于连接问题或服务器问题响应无效。您已尝试将该响应读取为JSONObject,但无法解析。因此,您的方法在AsyncTask中将null返回到doInBackground。因此,发生空指针异常。
因此,您应该检查为什么要获取unknownHostException并修复getJSONFromURL方法中的null问题。而且你必须正确解析你的JSON,因为你的解析中也有错误,Amit Gupta提到了这一点。希望它有所帮助。
答案 2 :(得分:0)
你的logcat说:
http连接java.net.UnknownHostException错误: api.openweathermap.org
您需要在清单xml文件中添加网络权限
已经存在
然后检查您的设备:
我在不在设备上的模拟器上运行它
来自SO answer:
答案非常简单:删除,然后重新创建您的AVD Eclipse中的(虚拟设备/模拟器)。它对我有用 - 第一次。