Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 2개 있습니다.)
(시리즈 글이 6개 있습니다.)
.NET Framework: 884. eBEST XingAPI의 C# 래퍼 버전 - XingAPINet Nuget 패키지
; https://www.sysnet.pe.kr/2/0/12134

.NET Framework: 892. eBEST C# XingAPI 래퍼 - Sqlite 지원 추가
; https://www.sysnet.pe.kr/2/0/12145

.NET Framework: 893. eBEST C# XingAPI 래퍼 - 로그인 처리
; https://www.sysnet.pe.kr/2/0/12146

.NET Framework: 894. eBEST C# XingAPI 래퍼 - 연속 조회 처리 방법
; https://www.sysnet.pe.kr/2/0/12149

.NET Framework: 1996. C# XingAPI - 주식 종목에 따른 PBR, PER, ROE, ROA 구하는 방법(t3320, t8430 예제)
; https://www.sysnet.pe.kr/2/0/13034

.NET Framework: 2002. C# XingAPI - ACF 파일을 이용한 퀀트 종목 찾기(t1857)
; https://www.sysnet.pe.kr/2/0/13046




C# XingAPI - 주식 종목에 따른 PBR, PER, ROE, ROA 등의 재무정보 구하는 방법(t3320, t8430 예제)

^^ 먼지 쌓인(?) XingAPINet을 다시 꺼냈는데요,

eBEST XingAPI의 C# 래퍼 버전 - XingAPINet Nuget 패키지
; https://www.sysnet.pe.kr/2/0/12134

그새 뭐가 많이 바뀌었는지 기존 코드로는 로그인도 안 되는군요. ^^; 일단 XingAPINet을 업데이트했고 1.1.5 버전으로 nuget에 올려두었습니다.

XingAPINet
; https://www.nuget.org/packages/XingAPINet/

자, 그럼 이것을 참조 추가해서,

Install-Package XingAPINet

다음과 같이 로그인 과정까지 완료할 수 있습니다.

using System;
using XingAPINet;

internal class Program
{
    static void Main(string[] args)
    {
        bool useDemoServer = true;
        Program pg = new Program();

        if (args.Length == 1)
        {
            if (args[0] == "hts")
            {
                useDemoServer = false;
            }
        }

        pg.Main(useDemoServer);
    }

    void Main(bool useDemoServer)
    {
        LoginInfo user = LoginInfo.FromPlainText("...id...", "...password...",
                                                 "...인증서 password..."); // 만약 demo 서버 접속이라면 인증서 계정은 필요 없습니다.

        using (XingClient xing = new XingClient(useDemoServer))
        {
            if (xing.ConnectWithLogin(user) == false)
            {
                Console.WriteLine(xing.ErrorMessage);
                return;
            }
        }
    }
}

그다음, 해야 할 일은 KODAQ과 KOSPI에 상장된 주식 종목을 가져오면 되는데요, 이것은 t8430 API를 이용해,

BEGIN_FUNCTION_MAP
    .Func,주식종목조회(t8430),t8430,block,headtype=A;
    BEGIN_DATA_MAP
    t8430InBlock,기본입력,input;
    begin
        구분(0:전체1:코스피2:코스닥),gubun,gubun,char,1;
    end
    t8430OutBlock,출력1,output,occurs;
    begin
        종목명,hname,hname,char,20;
        단축코드,shcode,shcode,char,6;
        확장코드,expcode,expcode,char,12;
        ETF구분(1:ETF),etfgubun,etfgubun,char,1;
        상한가,uplmtprice,uplmtprice,long,8;
        하한가,dnlmtprice,dnlmtprice,long,8;
        전일가,jnilclose,jnilclose,long,8;
        주문수량단위,memedan,memedan,char,5;
        기준가,recprice,recprice,long,8;
        구분(1:코스피2:코스닥),gubun,gubun,char,1;
    end
    END_DATA_MAP
END_FUNCTION_MAP

다음과 같이 간단하게 해결할 수 있습니다.

private SHCodeInfo[] GetStockList()
{
    List<SHCodeInfo> list = new List<SHCodeInfo>();

    foreach (var block in XQt8430.Get(XQt8430Gubun.전체))
    {
        if (block.etfgubun != '0') // ETF/ETN 유형은 제거
        {
            continue;
        }

        if (block.shcode[5] != '0') // 보통주만 허용
        {
            continue;
        }

        SHCodeInfo code = new SHCodeInfo(block.hname, block.shcode, block.expcode, false, false);
        list.Add(code);
    }

    return list.ToArray();
}

위의 코드를 보면, PBR과 PER 등의 정보는 주식과 관련이 있기 때문에 ETF/ETN에 해당하는 종목은 걸러내고 있습니다. 또한 우선주의 경우에는 이후 정보를 가져올 t3320 API로는 값이 안 나오기 떄문에 이것 역시 보통주만 선택하도록 필터링합니다.

자, 그다음 마지막으로 이제 해당 주식을 발행하는 회사의 PBR과 PER 등의 정보를 가져와야 하는데요, 이것은 t3320 API를 이용해 가져올 수 있습니다.

BEGIN_FUNCTION_MAP
    .Func,FNG_요약(t3320),t3320,attr,block,headtype=A;
    BEGIN_DATA_MAP
    t3320InBlock,기본입력,input;
    begin
        종목코드,gicode,gicode,char,7;
    end
    t3320OutBlock,기업기본정보,output;
    begin
        업종구분명,upgubunnm,upgubunnm,char,20;
        시장구분,sijangcd,sijangcd,char,1;
        시장구분명,marketnm,marketnm,char,10;
        한글기업명,company,company,char,100;
        본사주소,baddress,baddress,char,100;
        본사전화번호,btelno,btelno,char,20;
        최근결산년도,gsyyyy,gsyyyy,char,4;
        결산월,gsmm,gsmm,char,2;
        최근결산년월,gsym,gsym,char,6;
        주당액면가,lstprice,lstprice,long,12;
        주식수,gstock,gstock,long,12;
        Homepage,homeurl,homeurl,char,50;
        그룹명,grdnm,grdnm,char,30;
        외국인,foreignratio,foreignratio,float,6.2;
        주담전화,irtel,irtel,char,30;
        자본금,capital,capital,float,12.0;
        시가총액,sigavalue,sigavalue,float,12.0;
        배당금,cashsis,cashsis,float,12.0;
        배당수익율,cashrate,cashrate,float,13.2;
        현재가,price,price,long,8;
        전일종가,jnilclose,jnilclose,long,8;
        위험고지구분1_정리매매,notice1,notice1,char,1;
        위험고지구분2_투자위험,notice2,notice2,char,1;
        위험고지구분3_단기과열,notice3,notice3,char,1;
    end
    t3320OutBlock1,기업재무정보,output;
    begin
        기업코드,gicode,gicode,char,7;
        결산년월,gsym,gsym,char,6;
        결산구분,gsgb,gsgb,char,1;
        PER,per,per,float,13.2;
        EPS,eps,eps,float,13.0;
        PBR,pbr,pbr,float,13.2;
        ROA,roa,roa,float,13.2;
        ROE,roe,roe,float,13.2;
        EBITDA,ebitda,ebitda,float,13.2;
        EVEBITDA,evebitda,evebitda,float,13.2;
        액면가,par,par,float,13.2;
        SPS,sps,sps,float,13.2;
        CPS,cps,cps,float,13.2;
        BPS,bps,bps,float,13.0;
        T.PER,t_per,t_per,float,13.2;
        T.EPS,t_eps,t_eps,float,13.0;
        PEG,peg,peg,float,13.2;
        T.PEG,t_peg,t_peg,float,13.2;
        최근분기년도,t_gsym,t_gsym,char,6;
    end
    END_DATA_MAP
END_FUNCTION_MAP

따라서, 전체 주식 종목에 대해 다음과 같이 PER, PBR 정보를 가져올 수 있습니다.

SHCodeInfo[] stocks = GetStockList();
foreach (var item in GetStockInfo(stocks))
{
    Console.WriteLine($"{item.Name}: PER == {item.per}, PBR == {item.pbr}");
}

private IEnumerable<StockInfo> GetStockInfo(SHCodeInfo[] stocks)
{
    using (XQt3320 query = new XQt3320())
    {
        foreach (var item in stocks)
        {
            query.SetFieldData(XQt3320InBlock.BlockName, XQt3320InBlock.F.gicode, 0, item.SHCode);
            int resultCode = query.Request();
            if (resultCode < 0)
            {
                Console.WriteLine($"[{resultCode}] XQt3320: request failed at {item.Name}");
                break;
            }

            XQt3320OutBlock1 outBlock = query.GetBlock1();
            if (outBlock.IsValidData == false)
            {
                Console.WriteLine($"XQt3320: invalid block at {item.Name}");
                break;
            }

            yield return new StockInfo(item.Name, outBlock);
        }
    }
}

하지만, 실제로 저렇게 해보면 얼마 동안 정보를 가져오더니 어느 순간부터는 요청이 실패합니다. 이때의 resultCode 값은 -32가 나오는데요, 이것을 XingAPI의 DevCenter 도구에서 "보기" / "에러 찾기" 메뉴로 확인해 보면 다음과 같은 식의 오류 메시지를 볼 수 있습니다.

// Request() == -34인 경우

TR의 10분당 최대 전송 가능 횟수를 초가하였습니다.
이후부터는 전송가능횟수가 5배로 제한됩니다.

이러한 제약은 DevCenter의 "TR 속성" 창을 통해서도 확인할 수 있습니다.

xingapi_tr_time_limit_1.png

이러한 제약은 각각의 API마다 다를 수 있으므로 저렇게 꼭 "TR 속성" 창을 확인해 적절한 시간 지연을 해야 합니다. (아쉽게도 이러한 시간 정보가 RES 파일에는 없어서 이것을 소스 코드 자동화에 반영할 수가 없습니다. 혹시 이 정보를 어떻게 구할 수 있는지 방법을 아시는 분은 덧글 부탁드립니다. ^^)

암튼, 그래서 이런 경우에는 10분당 200건이니까, (10 * 60초 / 200 = ) 3초마다 한 번씩 정보를 요청하는 식으로 바꿔야 합니다. 이를 위해 QueryPerTime 속성을 넣어두었으니, 다음과 같이 설정해 주시면 됩니다.

private IEnumerable<StockInfo> GetStockInfo(SHCodeInfo[] stocks)
{
    using (XQt3320 query = new XQt3320())
    {
        // QueryPerTime에는 밀리 초 단위로 지정
        query.QueryPerTime = 3 * 1000; // 3초마다 한 번씩 전송

        foreach (var item in stocks)
        {
            // ...[생략]...

            yield return new StockInfo(item.Name, outBlock);
        }
    }
}

오늘 날짜 기준으로 XQt8430.Get 메서드가 반환하는 전체 코드는 3,333개, 그중에서 ETF/ETN을 제거하면 2,726개, 다시 그중에서 보통주만을 선별하면 2,370개 정도의 종목이 나옵니다. 정확히 3초씩 가져온다고 가정했을 때 7,110초(약 118분), 전체 가져오는데 약 2시간 정도의 시간이 걸립니다.

어차피, 프로그래밍으로 하는 거니까 DB에만 적절하게 저장해 두도록 그냥 실행해 두고 잊어버리면 되겠습니다. 또한, PER과 PBR 등의 수치들이 주식 가격에 따른 기업 실적의 결과로 산출되는 것이니 매일 장 마감 후 한 번만 하면 될 것입니다.

(첨부 파일은 이 글의 예제 코드를 포함합니다.)
(XingAPI github에도 t3320 프로젝트로 올렸습니다.)



이번 글에서는 기업별로 재무 정보를 가져오기 위해 (시간제한이 걸린) Request를 보내야 하는 불편함이 있었는데요, 다행히 이런 문제점을 해결하는 API가 있습니다. ^^ 그건 다음번 글에서 설명하겠습니다.

참고로, 웹 검색을 해보면 네이버 증권 등에서 웹 페이지로 제공하는 데이터를 HTML 파싱으로 PBR, PER 등의 정보를 가져오는 방법도 볼 수 있습니다. 그런 경우 시간제한이 걸린 증권사의 API 사용보다 빠르게 이용할 수 있지만 HTML 파싱의 특징상 홈페이지 개편이 있을 때마다 틀어질 수 있다는 단점이 있습니다. 다들 장/단점이 있군요. ^^




[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]

[연관 글]






[최초 등록일: ]
[최종 수정일: 5/8/2022]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 




... 76  [77]  78  79  80  81  82  83  84  85  86  87  88  89  90  ...
NoWriterDateCnt.TitleFile(s)
12011정성태8/27/201926228사물인터넷: 57. C# - Rapsberry Pi Zero W와 PC 간 Bluetooth 통신 예제 코드파일 다운로드1
12010정성태8/27/201919159VS.NET IDE: 138. VSIX - DTE.ItemOperations.NewFile 메서드에서 템플릿 이름을 다국어로 설정하는 방법
12009정성태8/26/201919989.NET Framework: 858. C#/Windows - Clipboard(Ctrl+C, Ctrl+V)가 동작하지 않는다면?파일 다운로드1
12008정성태8/26/201919658.NET Framework: 857. UWP 앱에서 SQL Server 데이터베이스 연결 방법
12007정성태8/24/201918264.NET Framework: 856. .NET Framework 버전을 올렸을 때 오류가 발생할 수 있는 상황
12006정성태8/23/201921733디버깅 기술: 129. guidgen - Encountered an improper argument. 오류 해결 방법 (및 windbg 분석) [1]
12005정성태8/13/201919352.NET Framework: 855. 닷넷 (및 VM 계열 언어) 코드의 성능 측정 시 주의할 점 [2]파일 다운로드1
12004정성태8/12/201927621.NET Framework: 854. C# - 32feet.NET을 이용한 PC 간 Bluetooth 통신 예제 코드 [14]
12003정성태8/12/201919754오류 유형: 564. Visual C++ 컴파일 오류 - fatal error C1090: PDB API call failed, error code '3'
12002정성태8/12/201919109.NET Framework: 853. Excel Sheet를 WinForm에서 사용하는 방법 - 두 번째 이야기 [5]
12001정성태8/10/201924320.NET Framework: 852. WPF/WinForm에서 UWP의 기능을 이용해 Bluetooth 기기와 Pairing하는 방법 [1]
12000정성태8/9/201923738.NET Framework: 851. WinForm/WPF에서 Console 창을 띄워 출력하는 방법파일 다운로드1
11999정성태8/1/201917993오류 유형: 563. C# - .NET Core 2.0 이하의 Unix Domain Socket 사용 시 System.IndexOutOfRangeException 오류
11998정성태7/30/201920106오류 유형: 562. .NET Remoting에서 서비스 호출 시 SYN_SENT로 남는 현상파일 다운로드1
11997정성태7/30/201920393.NET Framework: 850. C# - Excel(을 비롯해 Office 제품군) COM 객체를 제어 후 Excel.exe 프로세스가 남아 있는 문제 [2]파일 다운로드1
11996정성태7/25/201923398.NET Framework: 849. C# - Socket의 TIME_WAIT 상태를 없애는 방법파일 다운로드1
11995정성태7/23/201927122.NET Framework: 848. C# - smtp.daum.net 서비스(Implicit SSL)를 이용해 메일 보내는 방법 [2]
11994정성태7/22/201921819개발 환경 구성: 454. Azure 가상 머신(VM)에서 SMTP 메일 전송하는 방법파일 다운로드1
11993정성태7/22/201916508오류 유형: 561. Dism.exe 수행 시 "Error: 2 - The system cannot find the file specified." 오류 발생
11992정성태7/22/201918612오류 유형: 560. 서비스 관리자 실행 시 "Windows was unable to open service control manager database on [...]. Error 5: Access is denied." 오류 발생
11991정성태7/18/201915670디버깅 기술: 128. windbg - x64 환경에서 닷넷 예외가 발생한 경우 인자를 확인할 수 없었던 사례
11990정성태7/18/201917911오류 유형: 559. Settings / Update & Security 화면 진입 시 프로그램 종료
11989정성태7/18/201916759Windows: 162. Windows Server 2019 빌드 17763부터 Alt + F4 입력시 곧바로 로그아웃하는 현상
11988정성태7/18/201919278개발 환경 구성: 453. 마이크로소프트가 지정한 모든 Root 인증서를 설치하는 방법
11987정성태7/17/201925183오류 유형: 558. 윈도우 - KMODE_EXCEPTION_NOT_HANDLED 블루스크린(BSOD) 문제 [1]
11986정성태7/17/201916933오류 유형: 557. 드라이브 문자를 할당하지 않은 파티션을 탐색기에서 드라이브 문자와 함께 보여주는 문제
... 76  [77]  78  79  80  81  82  83  84  85  86  87  88  89  90  ...