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

비밀번호

댓글 작성자
 




... 121  122  123  124  125  126  127  128  129  130  131  132  [133]  134  135  ...
NoWriterDateCnt.TitleFile(s)
1731정성태8/11/201427081개발 환경 구성: 235. 점(.)으로 시작하는 파일명을 탐색기에서 만드는 방법
1730정성태8/11/201422165개발 환경 구성: 234. Royal TS의 터미널(Terminal) 연결에서 한글이 깨지는 현상 해결 방법
1729정성태8/11/201418224오류 유형: 236. SqlConnection - The requested Performance Counter is not a custom counter, it has to be initialized as ReadOnly.
1728정성태8/8/201430288.NET Framework: 453. C# - 오피스 파워포인트(Powerpoint) 파일을 WinForm에서 보는 방법파일 다운로드1
1727정성태8/6/201420514오류 유형: 235. SignalR 오류 메시지 - Counter 'Messages Bus Messages Published Total' does not exist in the specified Category. [2]
1726정성태8/6/201419399오류 유형: 234. IIS Express에서 COM+ 사용 시 SecurityException - "Requested registry access is not allowed" 발생
1725정성태8/6/201421353오류 유형: 233. Visual Studio 2013 Update3 적용 후 Microsoft.VisualStudio.Web.PageInspector.Runtime 모듈에 대한 FileNotFoundException 예외 발생
1724정성태8/5/201426102.NET Framework: 452. .NET System.Threading.Thread 개체에서 Native Thread Id를 구하는 방법 - 두 번째 이야기 [1]파일 다운로드1
1723정성태7/29/201458367개발 환경 구성: 233. DirectX 9 예제 프로젝트 빌드하는 방법 [3]파일 다운로드1
1722정성태7/25/201421059오류 유형: 232. IIS 500 Internal Server Error - NTFS 암호화된 폴더에 웹 애플리케이션이 위치한 경우
1721정성태7/24/201424073.NET Framework: 451. 함수형 프로그래밍 개념 - 리스트 해석(List Comprehension)과 순수 함수 [2]
1720정성태7/23/201422079개발 환경 구성: 232. C:\WINDOWS\system32\LogFiles\HTTPERR 폴더에 로그 파일을 남기지 않는 설정
1719정성태7/22/201426039Math: 13. 동전을 여러 더미로 나누는 경우의 수 세기(Partition Number) - 두 번째 이야기파일 다운로드1
1718정성태7/19/201435296Math: 12. HTML에서 수학 관련 기호/수식을 표현하기 위한 방법 - MathJax.js [4]
1716정성태7/17/201435009개발 환경 구성: 231. PC 용 무료 안드로이드 에뮬레이터 - genymotion
1715정성태7/13/201430608기타: 47. 운영체제 종료 후에도 USB 외장 하드의 전원이 꺼지지 않는 경우 [3]
1714정성태7/11/201420891VS.NET IDE: 92. Visual Studio 2013을 지원하는 IL Support 확장 도구
1713정성태7/11/201444607Windows: 98. 윈도우 시스템 디스크 용량 확보를 위한 "Package Cache" 폴더 이동 [1]
1712정성태7/10/201432861.NET Framework: 450. 영문 윈도우에서 C# 콘솔 프로그램의 유니코드 출력 방법 [3]
1711정성태7/10/201438052Windows: 97. cmd.exe 창에서 사용할 폰트를 추가하는 방법 [1]
1710정성태7/8/201430582개발 환경 구성: 230. 유니코드의 Surrogate Pair, Supplementary Characters가 뭘까요?파일 다운로드2
1709정성태7/8/201427395VS.NET IDE: 91. Visual Studio에서 32/64비트 IIS Express 실행하는 방법
1708정성태7/7/201424760VS.NET IDE: 90. Visual Studio - 사용자 정의 정적 분석 규칙 만드는 방법 [3]파일 다운로드1
1707정성태7/4/201423027.NET Framework: 449. C#에서 C++로 VARIANT 넘겨주는 방법파일 다운로드1
1706정성태7/3/201421432.NET Framework: 448. .NET SmartClient 컨트롤을 윈도우 8/2012에서 활성화하는 방법파일 다운로드1
1705정성태7/2/201435065VC++: 78. 보이어-무어(Boyer-Moore) 알고리즘이 정말 빠를까? [6]파일 다운로드1
... 121  122  123  124  125  126  127  128  129  130  131  132  [133]  134  135  ...