[안드로이드] 공공데이터 기차역 정보 조회 서비스 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; ArrayListmResult; // 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 ArrayListmakeResult() { 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이렇게 되면, Message로 받은 obj에 데이터가 들어가게 되고, 전송 할 수 있습니다.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보냄 } } }
ProgressDialog는 수행시간이 길지 않아서 금방 꺼지더군요 아래는 결과 화면입니다.
서울역 -> 대구역 으로의 조회를 한 것이고,
코레일 홈페이지에서 조회한 목록입니다.
무슨 오류인지는 모르겠지만 KTX는 가격이 0원으로 뜨더군요
그것을 제외하면 모두 일치하는 것 같습니다.
다음에는 역 ListViewActivity구성과 초성 검색 등을 포스팅 하겠습니다.
추가로 읽으면 좋을 것
이 블로그의 인기 게시물
윤석열 계엄령 선포! 방산주 대폭발? 관련주 투자 전략 완벽 분석
## 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조 원] - **주요 산업**: 식품 제조 및 발효제품 - **관련주 근거**: 계엄...
한국 핵무장 논의와 방위산업 관련주: 핵무기 개발 과정과 유망 종목 분석
한국의 독자적 핵무장 논의가 주요 이슈로 떠오르며 방위산업 관련 주식들이 주목받고 있습니다. 특히, 핵무기 및 방어 관련 기술력을 보유한 기업들이 관심을 끌고 있어 투자자들에게 큰 잠재적 수혜가 예상됩니다. 트럼프 전 미국 대통령의 재집권 가능성 등 외교적 변화는 이러한 방위산업 관련주를 더욱 부각시키고 있습니다. --- ### 핵무기 생산과정 요약 #### **핵연료 확보** : 고농축 우라늄-235 또는 플루토늄-239와 같은 핵분열 물질을 확보하는 과정입니다. - **우라늄 농축**: 우라늄-235의 비율을 약 90% 이상으로 높이는 과정입니다. - **플루토늄 생산**: 원자로에서 우라늄-238을 중성자로 포획하여 플루토늄을 생성하고 이를 화학적으로 분리합니다. #### **폭발 장치 개발** : 확보한 핵연료를 폭발할 수 있도록 설계된 장치입니다. - **충돌 방식 (Gun-type)**: 고농축 우라늄을 이용해 두 덩어리를 빠르게 결합시켜 핵분열을 유도합니다. - **내부 압축 방식 (Implosion-type)**: 고폭압력으로 플루토늄을 압축하여 임계 질량을 초과하도록 합니다. ####. **무기화 및 배치** - 폭발 장치를 무기 형태로 조립하여 배치 가능한 상태로 만드는 과정입니다. 미사일, 폭격기 등에 탑재될 수 있도록 설계합니다. --- ### 핵심적인 부분 가장 중요한 부분은 **핵연료 확보**와 **폭발 장치 개발**입니다. - **핵연료 확보**: 핵분열 물질 확보가 핵무기 개발의 필수 조건입니다. 우라늄 농축과 플루토늄 생산은 고도의 기술력을 요구하며, 보안과 국제적인 감시가 강화된 부분입니다. - **폭발 장치 개발**: 핵연료가 있어도 이를 효과적으로 폭발시키는 장치가 없다면 무기화가 불가능합니다. 압축 방식 등 폭발 장치 개발 기술이 핵무기의 폭발력과 신뢰성을 좌우하는 중요한 요소입니다. --- ### 핵연료 확보 관련 기업 - **한전원자력연료 (KEPCO NF)** :...
[로스트아크] 제작 효율 최적화 위한 영지 세팅
### 1. 대성공 확률 증가 vs. 제작 수수료 절감 - **대성공 확률 증가**: 대성공 확률이 2% 증가해도 실제 효과는 크지 않습니다. 예를 들어, 기본 대성공 확률 5%에 2% 증가를 적용해도 실질적인 효과는 0.1% 증가에 불과합니다. - **제작 수수료 절감**: 제작 수수료를 2% 절감할 경우, 제작할 때마다 발생하는 골드 비용을 직접적으로 줄일 수 있어 비용 절약 효과가 훨씬 큽니다. - 결과적으로, 제작 수수료 절감이 대성공 확률 증가보다 약 10배 더 많은 이득을 제공합니다. 따라서 대성공 확률보다는 수수료 절감에 집중하는 것이 권장됩니다. --- ### 2. 효율적인 영지 세팅을 위한 이득 극대화 세팅 - 영지 내 필수 세팅 아이템으로 "곡예사의 대기실," "찬란한 소원 나무," "여신의 가호"가 추천됩니다. - **곡예사의 대기실**: 마리샵에서 블루 크리스탈로 구매할 수 있으며, 기본적인 제작 효율을 높이는 데 필수 아이템입니다. - **찬란한 소원 나무**: 수수료 절감을 제공하여 제작 비용을 절감하는 효과가 있어 이득 극대화에 도움이 됩니다. - **여신의 가호**: 미술품 42개를 모아 획득할 수 있으며, 추가적인 제작 효율을 제공합니다. 여유가 있다면 필수로 장착하는 것이 좋습니다. - 여신의 가호 대신, **곡예사의 무기 진열대**를 구매해 사용할 수도 있으며, 경제적인 선택지로 활용할 수 있습니다. --- ### 3. 의상 세팅 (선택적 적용) - 특정 의상을 착용하면 제작 효율이 약간 증가하지만, 최적의 의상 옵션은 없기 때문에 필수는 아닙니다. 크리스탈 비용이 부담스러울 경우 생략 가능하며, 다른 세팅을 우선적으로 강화하는 것이 좋습니다. - **드레스룸 이용**: 크리스탈을 사용하여 드레스룸에서 특정 NPC와의 호감도로 얻을 수 있는 의상을 구매할 수 있습니다. - **추천 의상**: 페...
댓글
댓글 쓰기