API를 이용해 실시간 버스노선정보 XML을 Parsing하여 출력하는 프로그램
문제
AndroidManifest.xml
<?xml version="1.0" encoding="utf-8"?>
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="kr.ac.skuniv.a1210practice_4">
<uses-permission android:name="android.permission.INTERNET"/>
<application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:roundIcon="@mipmap/ic_launcher_round"
android:supportsRtl="true"
android:theme="@style/Theme.1210practice_4"
android:usesCleartextTraffic="true">
<activity android:name=".MainActivity" android:exported="true">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
</application>
</manifest>
activity_main.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
tools:context=".MainActivity">
<TextView
android:id="@+id/data"
android:layout_width="wrap_content"
android:layout_height="wrap_content"/>
</LinearLayout>
MainActivity.java
import androidx.appcompat.app.AppCompatActivity;
import android.os.AsyncTask;
import android.os.Bundle;
import android.widget.TextView;
import org.xmlpull.v1.XmlPullParser;
import org.xmlpull.v1.XmlPullParserFactory;
import java.io.BufferedInputStream;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.StringReader;
import java.net.HttpURLConnection;
import java.net.URL;
public class MainActivity extends AppCompatActivity {
TextView tv;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
tv = (TextView) findViewById(R.id.data);
// 노선ID 추출을 위한 공공 DB API의 URL
String serviceUrl = "http://ws.bus.go.kr/api/rest/busRouteInfo/getBusRouteList";
// 노선ID 추출을 위한 공공 DB API 키
String serviceKey = "노선ID 추출을 위한 공공 DB API 키";
// 노선버스의 노선번호
String strSrch = "1164";
// 공공 DB API 호추을 위한 URL
String strUrl = serviceUrl+"?ServiceKey="+serviceKey+"&strSrch="+strSrch;
new DownloadWebpageTask().execute(strUrl); // URL에 해당하는 문서 추출을 위한 객체 생성
}
// downloadUrl 메소드 호출하여 문서 추출
private class DownloadWebpageTask extends AsyncTask<String, Void, String> {
@Override
protected String doInBackground(String... urls) {
try {
return (String)downloadUrl((String)urls[0]);
} catch (IOException e) {
return "다운로드 실패";
}
}
//문서 추출 결과의 출력
protected void onPostExecute(String result) {
try {
// XmlPull Parser를 만들기 위한 XmlPullParserFactor 객체 생성
XmlPullParserFactory factory = XmlPullParserFactory.newInstance();
// XmlPullParserFactor에 의해 생성될 parser를 만들 때, XML name space지원 여부를 설정함
factory.setNamespaceAware(true);
// Xml Pull Parser 객체 생성
XmlPullParser xpp = factory.newPullParser();
xpp.setInput(new StringReader(result)); // 데이터 리소스(result)에 대한 Input Stream 설정
int eventType = xpp.getEventType(); // parser가 현재 가르키고 있는 이벤트 타입(START_TAG, END_TAG, TEXT 등)을 반환함
String headerCd = "";
String busRouteId = "";
String busRouteNm = "";
boolean bSet_headerCd = false;
boolean bSet_busRouteId = false;
boolean bSet_busRouteNm = false;
while (eventType != XmlPullParser.END_DOCUMENT) { // 현재 이벤트 타입이 END_DOCUMENT가 될 때까지 처리를 반복
if(eventType == XmlPullParser.START_DOCUMENT) {
;
} else if(eventType == XmlPullParser.START_TAG) { // 이벤트 타입이 START_TAG인 경우
String tag_name = xpp.getName(); // 태그 이름 추출
if (tag_name.equals("headerCd")) // 태그 이름이 (headerCD)인 경우
bSet_headerCd = true;
if (tag_name.equals("busRouteId"))
bSet_busRouteId = true;
if (tag_name.equals("busRouteNm"))
bSet_busRouteNm = true;
} else if(eventType == XmlPullParser.TEXT) { // 이벤트 타입이 TEXT인 경우
if (bSet_headerCd) {
headerCd = xpp.getText(); // 시작태그(<>)와 마침태그 (</>)사이에 있는 데이터 추출
tv.append("headerCd: " + headerCd + "\n");
bSet_headerCd = false;
}
if (headerCd.equals("0")) { // headerCd 태그의 값이 "0"인 경우
if (bSet_busRouteId) {
busRouteId = xpp.getText();
tv.append("busRouteId: " + busRouteId + "\n");
bSet_busRouteId = false;
}
if (bSet_busRouteNm) {
busRouteNm = xpp.getText();
tv.append("busRouteNm; " + busRouteNm + "\n");
bSet_busRouteNm = false;
}
}
} else if(eventType == XmlPullParser.END_TAG) { // 이벤트 타입이 END_TAG인 경우
;
}
eventType = xpp.next(); // parser를 다음 이벤트 타입으로 이동 후, 인식한 타입을 변환
}
} catch (Exception e) {
tv.setText(e.getMessage()); // 데이터 추출 과정에서 에러 발생 시, 에러 내용 출력
}
}
private String downloadUrl(String myurl) throws IOException {
HttpURLConnection conn = null; // HTTP로 URL 접속을 위한 객체 생성
try {
URL url = new URL(myurl); // URL 객체 생성
conn = (HttpURLConnection) url.openConnection(); // URL 객체를 이용한 HTTP 연결
BufferedInputStream buf = new BufferedInputStream(conn.getInputStream()); // 버퍼에 문서 다운로드
BufferedReader bufreader = new BufferedReader(new InputStreamReader(buf, "utf-8")); // 버퍼 내용을 UTF-8 문서 형식으로 변환
String line = null;
String page = "";
// 버퍼 내용을 형 단위로 읽어 문자 변수에 저장
while((line = bufreader.readLine()) != null) {
page += line;
}
return page; // 추출한 웹문서의 문서 내용을 변환
} finally {
conn.disconnect(); // HTTP 연결 해제
}
}
}
}