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

비밀번호

댓글 작성자
 




1  2  3  4  5  6  7  [8]  9  10  11  12  13  14  15  ...
NoWriterDateCnt.TitleFile(s)
13430정성태10/30/20232666닷넷: 2153. C# - 사용자가 빌드한 ICU dll 파일을 사용하는 방법
13429정성태10/27/20232940닷넷: 2152. Win32 Interop - C/C++ DLL로부터 이중 포인터 버퍼를 C#으로 받는 예제파일 다운로드1
13428정성태10/25/20233014닷넷: 2151. C# 12 - ref readonly 매개변수
13427정성태10/18/20233199닷넷: 2150. C# 12 - 정적 문맥에서 인스턴스 멤버에 대한 nameof 접근 허용(Allow nameof to always access instance members from static context)
13426정성태10/13/20233360스크립트: 59. 파이썬 - 비동기 호출 함수(run_until_complete, run_in_executor, create_task, run_in_threadpool)
13425정성태10/11/20233169닷넷: 2149. C# - PLinq의 Partitioner<T>를 이용한 사용자 정의 분할파일 다운로드1
13423정성태10/6/20233147스크립트: 58. 파이썬 - async/await 기본 사용법
13422정성태10/5/20233284닷넷: 2148. C# - async 유무에 따른 awaitable 메서드의 병렬 및 예외 처리
13421정성태10/4/20233345닷넷: 2147. C# - 비동기 메서드의 async 예약어 유무에 따른 차이
13420정성태9/26/20235514스크립트: 57. 파이썬 - UnboundLocalError: cannot access local variable '...' where it is not associated with a value
13419정성태9/25/20233188스크립트: 56. 파이썬 - RuntimeError: dictionary changed size during iteration
13418정성태9/25/20233883닷넷: 2146. C# - ConcurrentDictionary 자료 구조의 동기화 방식
13417정성태9/19/20233421닷넷: 2145. C# - 제네릭의 형식 매개변수에 속한 (매개변수를 가진) 생성자를 호출하는 방법
13416정성태9/19/20233233오류 유형: 877. redis-py - MISCONF Redis is configured to save RDB snapshots, ...
13415정성태9/18/20233726닷넷: 2144. C# 12 - 컬렉션 식(Collection Expressions)
13414정성태9/16/20233483디버깅 기술: 193. Windbg - ThreadStatic 필드 값을 조사하는 방법
13413정성태9/14/20233681닷넷: 2143. C# - 시스템 Time Zone 변경 시 이벤트 알림을 받는 방법
13412정성태9/14/20236959닷넷: 2142. C# 12 - 인라인 배열(Inline Arrays) [1]
13411정성태9/12/20233458Windows: 252. 권한 상승 전/후 따로 관리되는 공유 네트워크 드라이브 정보
13410정성태9/11/20234977닷넷: 2141. C# 12 - Interceptor (컴파일 시에 메서드 호출 재작성) [1]
13409정성태9/8/20233834닷넷: 2140. C# - Win32 API를 이용한 모니터 전원 끄기
13408정성태9/5/20233800Windows: 251. 임의로 만든 EXE 파일을 포함한 ZIP 파일의 압축을 해제할 때 Windows Defender에 의해 삭제되는 경우
13407정성태9/4/20233556닷넷: 2139. C# - ParallelEnumerable을 이용한 IEnumerable에 대한 병렬 처리
13406정성태9/4/20233520VS.NET IDE: 186. Visual Studio Community 버전의 라이선스
13405정성태9/3/20233967닷넷: 2138. C# - async 메서드 호출 원칙
13404정성태8/29/20233477오류 유형: 876. Windows - 키보드의 등호(=, Equals sign) 키가 눌리지 않는 경우
1  2  3  4  5  6  7  [8]  9  10  11  12  13  14  15  ...