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

비밀번호

댓글 작성자
 




... [16]  17  18  19  20  21  22  23  24  25  26  27  28  29  30  ...
NoWriterDateCnt.TitleFile(s)
13227정성태1/23/20234870개발 환경 구성: 659. Windows - IP MTU 값을 바꿀 수 있을까요? [1]
13226정성태1/23/20234547.NET Framework: 2088. .NET 5부터 지원하는 GetRawSocketOption 사용 시 주의할 점
13225정성태1/21/20233801개발 환경 구성: 658. Windows에서 실행 중인 소켓 서버를 다른 PC 또는 WSL에서 접속할 수 없는 경우
13224정성태1/21/20234143Windows: 221. Windows - Private/Public/Domain이 아닌 네트워크 어댑터 단위로 방화벽을 on/off하는 방법
13223정성태1/20/20234342오류 유형: 838. RDP 연결 오류 - The two computers couldn't connect in the amount of time allotted
13222정성태1/20/20233998개발 환경 구성: 657. WSL - DockerDesktop.vhdx 파일 위치를 옮기는 방법
13221정성태1/19/20234229Linux: 57. C# - 리눅스 프로세스 메모리 정보파일 다운로드1
13220정성태1/19/20234370오류 유형: 837. NETSDK1045 The current .NET SDK does not support targeting .NET ...
13219정성태1/18/20233932Windows: 220. 네트워크의 인터넷 접속 가능 여부에 대한 판단 기준
13218정성태1/17/20233869VS.NET IDE: 178. Visual Studio 17.5 (Preview 2) - 포트 터널링을 이용한 웹 응용 프로그램의 외부 접근 허용
13217정성태1/13/20234460디버깅 기술: 185. windbg - 64비트 운영체제에서 작업 관리자로 뜬 32비트 프로세스의 덤프를 sos로 디버깅하는 방법
13216정성태1/12/20234716디버깅 기술: 184. windbg - 32비트 프로세스의 메모리 덤프인 경우 !peb 명령어로 나타나지 않는 환경 변수
13215정성태1/11/20236260Linux: 56. 리눅스 - /proc/pid/stat 정보를 이용해 프로세스의 CPU 사용량 구하는 방법 [1]
13214정성태1/10/20235823.NET Framework: 2087. .NET 6부터 SourceGenerator와 통합된 System.Text.Json [1]파일 다운로드1
13213정성태1/9/20235367오류 유형: 836. docker 이미지 빌드 시 "RUN apt install ..." 명령어가 실패하는 이유
13212정성태1/8/20235120기타: 85. 단정도/배정도 부동 소수점의 정밀도(Precision)에 따른 형변환 손실
13211정성태1/6/20235147웹: 42. (https가 아닌) http 다운로드를 막는 웹 브라우저
13210정성태1/5/20234200Windows: 219. 윈도우 x64의 경우 0x00000000`7ffe0000 아래의 주소는 왜 사용하지 않을까요?
13209정성태1/4/20234091Windows: 218. 왜 윈도우에서 가상 메모리 공간은 64KB 정렬이 된 걸까요?
13208정성태1/3/20234053.NET Framework: 2086. C# - Windows 운영체제의 2MB Large 페이지 크기 할당 방법파일 다운로드1
13207정성태12/26/20224355.NET Framework: 2085. C# - gpedit.msc의 "User Rights Assignment" 특권을 코드로 설정/해제하는 방법파일 다운로드1
13206정성태12/24/20224578.NET Framework: 2084. C# - GetTokenInformation으로 사용자 SID(Security identifiers) 구하는 방법 [3]파일 다운로드1
13205정성태12/24/20224934.NET Framework: 2083. C# - C++과의 연동을 위한 구조체의 fixed 배열 필드 사용 (2)파일 다운로드1
13204정성태12/22/20224219.NET Framework: 2082. C# - (LSA_UNICODE_STRING 예제로) CustomMarshaler 사용법파일 다운로드1
13203정성태12/22/20224381.NET Framework: 2081. C# Interop 예제 - (LSA_UNICODE_STRING 예제로) 구조체를 C++에 전달하는 방법파일 다운로드1
13202정성태12/21/20224790기타: 84. 직렬화로 설명하는 Little/Big Endian파일 다운로드1
... [16]  17  18  19  20  21  22  23  24  25  26  27  28  29  30  ...