[안드로이드] 공공데이터 기차역 정보 조회 서비스 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조 원] - **주요 산업**: 식품 제조 및 발효제품 - **관련주 근거**: 계엄...
대통령 퇴진운동 관련주: 방송·통신·촛불수혜주 완벽 분석
--- ## 1. 배경 2024년 12월 3일, 윤석열 대통령이 비상계엄령을 선포했으나, 짧은 시간 내에 이를 해제하면서 정치적 긴장감이 커졌습니다. 이에 따라 대규모 촛불시위와 같은 사회적 움직임이 예상되며, 통신과 관련된 기업 및 촛불 제조와 연관된 산업에 관심이 모이고 있습니다. --- ## 2. 기업 및 관련주 대규모 시위 및 관련 활동으로 인해 통신, 미디어, 그리고 촛불 제조와 관련된 기업들이 단기적인 수혜를 볼 것으로 예상됩니다. ### 2-1. 통신 관련주 #### 1. **KT (030200) [약 12조 원]** - **주요 산업:** 통신 - **관련주 근거:** 시위 생중계 및 대규모 통신 트래픽 증가로 매출 증대 가능성 - **주가정보:** [네이버 차트](https://finance.naver.com/item/main.nhn?code=030200) #### 2. **SK텔레콤 (017670) [약 12조 원]** - **주요 산업:** 통신 - **관련주 근거:** 대규모 데이터 사용 증가로 인한 수익 상승 - **주가정보:** [네이버 차트](https://finance.naver.com/item/main.nhn?code=017670) #### 3. **LG유플러스 (KOSPI, 032640) [약 4.9조 원]** - **주요 산업:** 통신 - **관련주 근거:** 촛불시위로 인한 데이터 및 음성 서비스 사용 증가 예상 - **주가정보:** [네이버 차트](https://finance.naver.com/item/main.nhn?code=032640) --- ### 2-2. 방송 관련주 #### 1. **SBS (034120) [약 2,924억 원]** - **주요 산업:** 방송 및 미디어 콘텐츠 제작 - **관련주 근거:** 시위 관련 특집 방송 및 실시간 보도에 따른 광고 수익 증가 - **주가정보:** [네이버 차트](https://finance.naver.com/item/main.nhn?code...
키움 OPEN API MFC 개발 (1)
* 키움 API 개발 - visual studio 2019 , MFC * Visual Studio Set - 새 프로젝트 만들기 / MFC 검색 - 다음 이후, MFC 설정에서 어플리케이션 종류 변경 (대화 상자 기반) * 기본 적용 Flow ( https://www.kiwoom.com/nkw.templateFrameSet.do?m=m1408000000 ) = 우선 생략하고, Step 2 설치 = Step 3 자료실/ KhOpenApiTest_2.71.zip 다운로드 * Step 2 설치 후, 설치 경로의 OpenAPI 디렉토리 찾기 1. 파일 찾기 2. KHOpenAPI.ocx 를 프로젝트 소스에 복사 * Step 3 자료실/다운로드 1. khOpenApiTest_2.71.zip 다운/압축 풀고, 2. KHOpenAPI.cpp/h KHOpenAPICtrl.cpp/h 프로젝트 소스에 복사 * 내부 소스에 다음추가 header에 class 생성 cpp에 다음 소스 추가 * 리소스 뷰 > IDD_TRADINGAPP_DIALOG 1. 확인 우클릭 > Active X 컨트롤 삽입 2. KHOpenAPI Control 적용 하면 위 화면처럼 적용 이후 실행 시 다음 화면 이후 매수/매도 적용
댓글
댓글 쓰기