[안드로이드] 공공데이터 기차역 정보 조회 서비스 API 사용 [2]
- 공유 링크 만들기
- X
- 이메일
- 기타 앱
일단 코레일 티켓창과 비슷하게 만들기 위해,
출발역과 도착역을 설정할 수 있는 Activity 와
출력할 수 있는 것을 만들어 보겠습니다.
주요 기능으로는 URL 요청, XML 파싱, 리스트 뷰, 초성 검색 , 한글 정렬 , 텍스트 Read
ProgressDialog 기능들로 구성되어 있습니다.
대략적인 구조를 설명하자면
이런식으로 동작하게 됩니다.
AndroidManifest 에 권한 설정을 해주셔야됩니다.
<uses-permission android:name="android.permission.INTERNET" /> <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
를 해주시고...
MainActivity 에서 URLRequest.java에 데이터 요청 후, 받는 과정을 설명하겠습니다.
activity_main.xml
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" android:paddingBottom="@dimen/activity_vertical_margin" android:paddingLeft="@dimen/activity_horizontal_margin" android:paddingRight="@dimen/activity_horizontal_margin" android:paddingTop="@dimen/activity_vertical_margin" android:orientation="vertical" tools:context="example.korailproject.MainActivity"> <LinearLayout android:orientation="horizontal" android:layout_width="match_parent" android:layout_height="wrap_content"> <Button android:id = "@+id/btn_startStation" android:text = "출발 역 설정" android:layout_width="wrap_content" android:layout_height="wrap_content" /> <TextView android:id = "@+id/tv_startStation" android:text = "서울역" android:layout_width="wrap_content" android:layout_height="wrap_content" /> </LinearLayout> <LinearLayout android:orientation="horizontal" android:layout_width="match_parent" android:layout_height="wrap_content"> <Button android:id = "@+id/btn_endStation" android:text = "도착 역 설정" android:layout_width="wrap_content" android:layout_height="wrap_content" /> <TextView android:id = "@+id/tv_endStation" android:text = "부산역" android:layout_width="wrap_content" android:layout_height="wrap_content" /> </LinearLayout> <Button android:id = "@+id/btn_SearchTicket" android:text = "조회" android:layout_width="match_parent" android:layout_height="wrap_content" /> <ScrollView android:layout_width="match_parent" android:layout_height="match_parent"> <TextView android:layout_gravity="center" android:text = "No Data" android:id = "@+id/tv_result" android:layout_width="wrap_content" android:layout_height="wrap_content" /> </ScrollView> </LinearLayout>
MainActivity.java
public class MainActivity extends AppCompatActivity implements View.OnClickListener{
Button btn_startStation; // 역 설정 Activity로 들어가기 위한 버튼
Button btn_endStation;
Button btn_search;
ArrayList mResult; // URLRequest로 받은 데이터 저장
ArrayList mTrainCategory; // 역 구분을 위함
MainActivity mThis;
URLRequest mRequest;
//NAT010000
//NAT011668
StationInfo mStartStation; //초기 설정값
StationInfo mEndStation;
TextView tv_result; // 조회 목록 출력
TextView tv_startStationName; //출발역 TextView
TextView tv_endStationName; // 도착역 TextView
Context mContext;
ProgressDialog dialog; // Loading 표시 위한 ProgressDialog
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mThis = this;
mContext = this.getApplicationContext();
Initialize();
TrainCategorySet();
}
void TrainCategorySet() // 기차 코드번호에 따른 기차 종류
{
mTrainCategory.add(new TrainType("00","KTX"));
mTrainCategory.add(new TrainType("01","새마을호"));
mTrainCategory.add(new TrainType("02","무궁화호"));
mTrainCategory.add(new TrainType("03","통근열차"));
mTrainCategory.add(new TrainType("04","누리로"));
mTrainCategory.add(new TrainType("06", "공항직통"));
mTrainCategory.add(new TrainType("07", "KTX-산천"));
mTrainCategory.add(new TrainType("08","ITX-새마을"));
mTrainCategory.add(new TrainType("09", "ITX-청춘"));
mTrainCategory.add(new TrainType("10", "KTX-산천"));
}
void Initialize()
{
mTrainCategory = new ArrayList<>();
mResult = new ArrayList<>();
tv_result = (TextView)findViewById(R.id.tv_result);
btn_endStation = (Button)findViewById(R.id.btn_endStation);
btn_startStation = (Button)findViewById(R.id.btn_startStation);
btn_search = (Button)findViewById(R.id.btn_SearchTicket);
btn_endStation.setOnClickListener(this);
btn_startStation.setOnClickListener(this);
btn_search.setOnClickListener(this);
tv_startStationName = (TextView)findViewById(R.id.tv_startStation);
tv_endStationName= (TextView)findViewById(R.id.tv_endStation);
mStartStation = new StationInfo("서울역","NAT010000"); // 기본 역과 코드
mEndStation = new StationInfo("부산역","NAT014445");
}
@Override
public void onClick(View v) {
Intent intent;
switch(v.getId())
{
case R.id.btn_endStation:
intent = new Intent(this,StationListActivity.class);
startActivityForResult(intent,HandlerMessage.END_SET);
break;
case R.id.btn_startStation:
intent = new Intent(this,StationListActivity.class);
startActivityForResult(intent,HandlerMessage.START_SET);
break;
case R.id.btn_SearchTicket: // 조회 버튼
mRequest = new URLRequest(mStartStation,mEndStation,mThis);
mRequest.run();
dialog = ProgressDialog.show(this, "",
"Loading", true); // ProgressDialog 시작
break;
}
}
public Handler handler = new Handler() { // Thread 작업이 완료되면 메세지를 받는다.
@Override
public void handleMessage(Message msg) {
super.handleMessage(msg);
switch (msg.what) {
case HandlerMessage.THREAD_HANDLER_SUCCESS_INFO:
ArrayList mTrainTicekt = (ArrayList)msg.obj;
ContentValueToArrayList(mTrainTicekt);
ArrayList result_Text = makeResult();
if(result_Text == null) {
Toast.makeText(mContext,"역 운행정보가 없습니다",Toast.LENGTH_SHORT);
break;
}
String tv_SetResult = "";
for(int i = 0; i < result_Text.size(); i++)
{
String temp = result_Text.get(i);
tv_SetResult += temp;
}
tv_result.setText(tv_SetResult);
dialog.dismiss(); // 프로그레스 다이얼로그 종료
break;
default:
break;
}
}
};
}
받은 데이터를 TextView하나로 출력하려 합니다. 그 TextView를 ScrollView로 감싸, 모든 데이터를 볼 수 있게 고칠 예정입니다.
makeDate() 함수로 받은 데이터를 SubString으로 쪼개고,
화살표 및 년,월,일을 넣어줍니다. 이 같은 방법이 가능한 것이, 데이터 포맷을 미리 확인하였기 때문입니다.(인터넷 주소창에 치면 나오게 됩니다. OpenWeather 활용[1] 에 있습니다.)
아래는 데이터 포맷을 알기 때문에, MainActivity에서
subString으로 데이터를 만드는 함수들입니다. 전부 MainActivity안에 있으며, 이해를 돕기위해 구분 지어 올렸습니다.
public String makeDate(String rawString)
{
String result;
String year = rawString.substring(0,4);
String month = rawString.substring(4,6);
String day = rawString.substring(6,8);
String hour = rawString.substring(8,10);
String min = rawString.substring(10,12);
result = year + "년 " + month + "월 " + day + "일 "
+ hour + "시 " + min + "분";
return result;
}
public String makeCity(String startStation,String endStation)
{
String result = startStation + " -> " + endStation;
return result;
}
public String TrainCategory(String train)
{
for(int i = 0; i < mTrainCategory.size(); i++)
{
if(mTrainCategory.get(i).id.equals(train))
return mTrainCategory.get(i).val;
}
return "화물열차";
}
public ArrayList makeResult()
{
ArrayList result = new ArrayList<>();
for(int i = 0; i < mResult.size(); i++)
{
String endDate = mResult.get(i).arrPlandTime;
String startDate = mResult.get(i).depPlandTime;
startDate = makeDate(startDate);
endDate = makeDate(endDate);
String city = makeCity(mResult.get(i).depPlaceName,mResult.get(i).arrPlaceName);
String pay = mResult.get(i).adultCharge;
String trainCategory =TrainCategory(mResult.get(i).trainGradeName);
result.add(startDate + " -> " + "\r\n" + endDate + "\r\n" + city + "\r\n" + trainCategory + " " + pay + "\r\n" + "\r\n");
}
if(result.size()!=0)
return result;
return null;
}
void ContentValueToArrayList(ArrayList mData)
{
for(int j = 0; j < mData.size(); j++)
{
String trainGradeName = String.valueOf(mData.get(j).get("traingradename")); // 차량 종류
String depPlandTime = String.valueOf(mData.get(j).get("depplandtime")); // 출발 시간
String arrPlandTime = String.valueOf(mData.get(j).get("arrplandtime")); // 도착 시간
String depPlaceName = String.valueOf(mData.get(j).get("depplacename")); // 출발지
String arrPlaceName = String.valueOf(mData.get(j).get("arrplacename")); // 도착지
String adultCharge = String.valueOf(mData.get(j).get("adultcharge")); // 비용
if(trainGradeName != null && depPlaceName != null && depPlandTime != null && arrPlandTime != null
&& arrPlaceName != null && adultCharge != null) {
TrainTicketInfo mTemp = new TrainTicketInfo(trainGradeName, depPlandTime, arrPlandTime, depPlaceName, arrPlaceName, adultCharge);
mResult.add(mTemp);
}
}
}
HandlerMessage.java
public class HandlerMessage {
public static final int THREAD_HANDLER_SUCCESS_INFO = 1;
public static final int START_SET = 2;
public static final int END_SET = 3;
}
요청과, 완료 메세지를 중복으로 쓰기위한 class입니다. 이것을 토대로 핸들러 메세지를 설정합니다.
아래는 티켓정보를 받는 클래스이며, 이 클래스를 토대로 ArrayList를 구성합니다.
TrainTicketInfo
public class TrainTicketInfo {
String trainGradeName; // 차량 종류
String depPlandTime; // 출발 시간
String arrPlandTime; // 도착 시간
String depPlaceName; // 출발지
String arrPlaceName; // 도착지
String adultCharge; // 비용
public TrainTicketInfo(String trainGradeName, String depPlandTime, String arrPlandTime, String depPlaceName, String arrPlaceName, String adultCharge) {
this.trainGradeName = trainGradeName;
this.depPlandTime = depPlandTime;
this.arrPlandTime = arrPlandTime;
this.depPlaceName = depPlaceName;
this.arrPlaceName = arrPlaceName;
this.adultCharge = adultCharge;
}
}
Thread로 URL요청, 데이터를 받는 class입니다.
URLRequest.java
public class URLRequest extends Thread {
StationInfo mStart;
StationInfo mEnd;
Calendar mCal;
String date = "";
ArrayList mResult;
String mXmlList[] = {"adultcharge", "arrplacename", "arrplandtime", "depplacename",
"depplandtime", "traingradename"};
MainActivity mHandler;
String makeString(int val)
{
String temp = String.valueOf(val);
if(val >= 1 && val < 10)
temp = "0"+val;
return temp;
}
public URLRequest(StationInfo mStart, StationInfo mEnd,MainActivity mHandler) {
this.mStart = mStart;
this.mEnd = mEnd;
mCal = Calendar.getInstance();
String year = String.valueOf(mCal.get(Calendar.YEAR));
String month = makeString(mCal.get(Calendar.MONTH) + 1);
String day = makeString(mCal.get(Calendar.DAY_OF_MONTH));
date = year + month + day;
mResult = new ArrayList<>();
this.mHandler = mHandler;
}
boolean isPossibleTagName(String tag)
{
if(tag == null) return false;
for(int i = 0; i < mXmlList.length; i++)
if(tag.equals(mXmlList[i])) return true;
return false;
}
boolean CheckEndItem(int event,String name)
{
if(event == XmlPullParser.END_TAG)
{
if(name != null) {
name.equals("items");
return true;
}
}
return false;
}
//getVhcleKndList
int RequestURL(int pageNo) //pageNo조정으로 모든 데이터 추가
{
if(mStart == null || mEnd == null ) return 0;
String st_key = "부여받은 키값 입력하시면 됩니다";
int idx = 0;
try {
URL url = new URL("http://openapi.tago.go.kr/openapi/service/TrainInfoService/getStrtpntAlocFndTrainInfo?ServiceKey=" + st_key +
"&numOfRows=100" +
"&pageNo=" + pageNo +
"&depPlaceId=" + mStart.st_stationCode +
"&arrPlaceId=" + mEnd.st_stationCode +
"&depPlandTime=" + date
);
XmlPullParserFactory factory = XmlPullParserFactory.newInstance();
// 위에서 생성된 URL을 통하여 서버에 요청하면 결과가 XML Resource로 전달됨
XmlPullParser parser = factory.newPullParser();
// XML Resource를 파싱할 parser를 factory로 생성
parser.setInput(url.openStream(), null);
// 파서를 통하여 각 요소들의 이벤트성 처리를 반복수행
int parserEvent = parser.getEventType();
String tagName;
while (parserEvent != XmlPullParser.END_DOCUMENT) {
// XML문이 끝날 때 까지 정보를 읽는다
if (parserEvent == XmlPullParser.START_TAG) {
// TrainInfo train = new TrainInfo();
//시작태그의 이름을 알아냄
ContentValues mContent = new ContentValues();
for( ; ; ) { // 하나의 item이 끝날 때, 들어감.
tagName = parser.getName();
if (!isPossibleTagName(tagName)) {
if(CheckEndItem(parserEvent,tagName)) break;
parserEvent = parser.next();
continue;
}
for (int index = 0; index < mXmlList.length; index++) {
if (parserEvent == parser.START_TAG && tagName.equals(mXmlList[index]) && mXmlList.length - 1 == index) {
// mContent.put(mXmlList[index], parser.getAttributeValue(null, mXmlList[index]));
mContent.put(mXmlList[index], parser.nextText());
mResult.add(mContent);
idx++;
break;
} else if (parserEvent == parser.START_TAG && tagName.equals(mXmlList[index])) {
// mContent.put(mXmlList[index], parser.getAttributeValue(null, mXmlList[index]));
mContent.put(mXmlList[index], parser.nextText());
break;
}
}
parserEvent = parser.next();
}
}
parserEvent = parser.next();
}
}catch(MalformedURLException e){
e.printStackTrace();
}catch(XmlPullParserException e){
e.printStackTrace();
}catch(IOException e){
e.printStackTrace();
}
return idx;
}
@Override
public void run() {
super.run();
if (android.os.Build.VERSION.SDK_INT > 9) { // AsyncTask를 지정하지 않고, 사용하기 위한 조건
StrictMode.ThreadPolicy policy = new StrictMode.ThreadPolicy.Builder().permitAll().build();
StrictMode.setThreadPolicy(policy);
}
for(int i = 1; ;i++) // 모든 열차 운행 정보를 받기위함
if(RequestURL(i) == 0) break;
Message msg= new Message();
msg.what = HandlerMessage.THREAD_HANDLER_SUCCESS_INFO;
msg.obj = mResult;
mHandler.handler.sendMessage(msg);
//Thread 작업 종료, UI 작업을 위해 MainHandler에 Message보냄 }
}
}
이렇게 되면, Message로 받은 obj에 데이터가 들어가게 되고, 전송 할 수 있습니다.ProgressDialog는 수행시간이 길지 않아서 금방 꺼지더군요 아래는 결과 화면입니다.

서울역 -> 대구역 으로의 조회를 한 것이고,
코레일 홈페이지에서 조회한 목록입니다.
무슨 오류인지는 모르겠지만 KTX는 가격이 0원으로 뜨더군요
그것을 제외하면 모두 일치하는 것 같습니다.
다음에는 역 ListViewActivity구성과 초성 검색 등을 포스팅 하겠습니다.
추가로 읽으면 좋을 것
이 블로그의 인기 게시물
한국 핵무장 논의와 방위산업 관련주: 핵무기 개발 과정과 유망 종목 분석
한국의 독자적 핵무장 논의가 주요 이슈로 떠오르며 방위산업 관련 주식들이 주목받고 있습니다. 특히, 핵무기 및 방어 관련 기술력을 보유한 기업들이 관심을 끌고 있어 투자자들에게 큰 잠재적 수혜가 예상됩니다. 트럼프 전 미국 대통령의 재집권 가능성 등 외교적 변화는 이러한 방위산업 관련주를 더욱 부각시키고 있습니다. ( 참조: https://gussconomy.tistory.com/entry/한국-핵무장-시나리오-관련주-투자-포인트-총정리 ) --- ### 핵무기 생산과정 요약 #### **핵연료 확보** : 고농축 우라늄-235 또는 플루토늄-239와 같은 핵분열 물질을 확보하는 과정입니다. - **우라늄 농축**: 우라늄-235의 비율을 약 90% 이상으로 높이는 과정입니다. - **플루토늄 생산**: 원자로에서 우라늄-238을 중성자로 포획하여 플루토늄을 생성하고 이를 화학적으로 분리합니다. #### **폭발 장치 개발** : 확보한 핵연료를 폭발할 수 있도록 설계된 장치입니다. - **충돌 방식 (Gun-type)**: 고농축 우라늄을 이용해 두 덩어리를 빠르게 결합시켜 핵분열을 유도합니다. - **내부 압축 방식 (Implosion-type)**: 고폭압력으로 플루토늄을 압축하여 임계 질량을 초과하도록 합니다. ####. **무기화 및 배치** - 폭발 장치를 무기 형태로 조립하여 배치 가능한 상태로 만드는 과정입니다. 미사일, 폭격기 등에 탑재될 수 있도록 설계합니다. --- ### 핵심적인 부분 가장 중요한 부분은 **핵연료 확보**와 **폭발 장치 개발**입니다. - **핵연료 확보**: 핵분열 물질 확보가 핵무기 개발의 필수 조건입니다. 우라늄 농축과 플루토늄 생산은 고도의 기술력을 요구하며, 보안과 국제적인 감시가 강화된 부분입니다. - **폭발 장치 개발**: 핵연료가 있어도 이를 효과적으로 폭발시키는 장치가 없다면 무기화가 불가능합니다. 압축 방식 등 폭발 장치 개발 기술이 핵무기의 폭발력...
[로스트아크] 제작 효율 최적화 위한 영지 세팅
### 1. 대성공 확률 증가 vs. 제작 수수료 절감 - **대성공 확률 증가**: 대성공 확률이 2% 증가해도 실제 효과는 크지 않습니다. 예를 들어, 기본 대성공 확률 5%에 2% 증가를 적용해도 실질적인 효과는 0.1% 증가에 불과합니다. - **제작 수수료 절감**: 제작 수수료를 2% 절감할 경우, 제작할 때마다 발생하는 골드 비용을 직접적으로 줄일 수 있어 비용 절약 효과가 훨씬 큽니다. - 결과적으로, 제작 수수료 절감이 대성공 확률 증가보다 약 10배 더 많은 이득을 제공합니다. 따라서 대성공 확률보다는 수수료 절감에 집중하는 것이 권장됩니다. --- ### 2. 효율적인 영지 세팅을 위한 이득 극대화 세팅 - 영지 내 필수 세팅 아이템으로 "곡예사의 대기실," "찬란한 소원 나무," "여신의 가호"가 추천됩니다. - **곡예사의 대기실**: 마리샵에서 블루 크리스탈로 구매할 수 있으며, 기본적인 제작 효율을 높이는 데 필수 아이템입니다. - **찬란한 소원 나무**: 수수료 절감을 제공하여 제작 비용을 절감하는 효과가 있어 이득 극대화에 도움이 됩니다. - **여신의 가호**: 미술품 42개를 모아 획득할 수 있으며, 추가적인 제작 효율을 제공합니다. 여유가 있다면 필수로 장착하는 것이 좋습니다. - 여신의 가호 대신, **곡예사의 무기 진열대**를 구매해 사용할 수도 있으며, 경제적인 선택지로 활용할 수 있습니다. --- ### 3. 의상 세팅 (선택적 적용) - 특정 의상을 착용하면 제작 효율이 약간 증가하지만, 최적의 의상 옵션은 없기 때문에 필수는 아닙니다. 크리스탈 비용이 부담스러울 경우 생략 가능하며, 다른 세팅을 우선적으로 강화하는 것이 좋습니다. - **드레스룸 이용**: 크리스탈을 사용하여 드레스룸에서 특정 NPC와의 호감도로 얻을 수 있는 의상을 구매할 수 있습니다. - **추천 의상**: 페...
윤석열 계엄령 선포! 방산주 대폭발? 관련주 투자 전략 완벽 분석
## 1. 배경 2024년 12월 3일, 윤석열 대통령이 국가 비상사태를 이유로 계엄령을 선포하였습니다. 계엄령은 전시나 사변 등 국가의 안녕과 공공질서가 심각하게 위협받을 때 대통령이 군사적 권한을 통해 이를 방어하고 유지하기 위해 발효하는 특별한 조치입니다. 이러한 조치는 국내 정치·경제 전반에 큰 영향을 미치며, 특히 주식시장에서는 관련 기업들의 주가 변동이 예상됩니다. 24.12.03 오전 5시 계엄 해제로 아래 관련주 추천 - [윤석열 계엄령 해제! 이재명 관련주 급등? 투자자 필독 전략](https://warguss.blogspot.com/2024/12/yoon-martial-law-lift-lee-jaemyung-stocks.html) --- ## 2. 기업 및 관련주 ### 2-1 식품 관련주 - 계엄령이 선포되면 사회적 불안정성이 증가할 수 있으며, 이에 따라 생필품 및 음식 관련 주식이 단기적으로 강세를 보일 가능성이 있습니다. #### 1. CJ제일제당 (KOSPI: 097950) [시가총액: 약 10조 원] - **주요 산업**: 식품 및 생필품 제조 - **관련주 근거**: 국가적 위기 상황에서 식료품 수요가 증가하며, 즉석밥, 가공식품 등의 판매가 확대될 가능성이 있습니다. - **주가정보**: [네이버 차트](https://finance.naver.com/item/main.nhn?code=097950) #### 2. 오뚜기 (KOSPI: 007310) [시가총액: 약 3조 원] - **주요 산업**: 식품 제조 및 유통 - **관련주 근거**: 라면, 즉석식품 등 비축 가능한 식품 수요가 증가하며, 매출 상승이 기대됩니다. - **주가정보**: [네이버 차트](https://finance.naver.com/item/main.nhn?code=007310) #### 3. 대상 (KOSPI: 001680) [시가총액: 약 2조 원] - **주요 산업**: 식품 제조 및 발효제품 - **관련주 근거**: 계엄...
댓글
댓글 쓰기