3
10

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 5 years have passed since last update.

Pythonでゴールデンクロスの情報を1日1回メールで通知する

Last updated at Posted at 2019-10-05

目次

株のデータ収集についての記事一覧をこちらに記載しております。

目的

  • 過去にゴールデンクロスが起きた日付を特定する機能の実装
  • ゴールデンクロスが起きている株式の情報をメールで通知する機能の実装

ゴールデンクロスとは?

ゴールデンクロスは短期線が長期線を下から上へ突き抜ける状態のことを言います。直近の価格傾向が上向きに転じたと捉えられるので「買いシグナル」です。
pic4.gif
###デッドクロスとは?
ゴールデンクロスとは逆に、短期移動平均線が長期移動平均線を上から下へ突き抜けることを言います。為替レートが下降トレンドに転じたと捉えられるため「売りシグナル」となります。
pic5.gif
引用:外為ドットコム

ゴールデンクロスとデッドクロスの算出方法

ゴールデンクロスとデッドクロスの算出方法は単純で、以下の条件で求めることが可能です。

ゴールデンクロス
前日の短期線 < 前日の長期線 AND 当日の短期線 > 当日の長期戦

デッドクロス
前日の長期線 < 前日の短期線 AND 当日の長期線 > 当日の短期戦

過去にゴールデンクロスが起きた日付を特定する

前提

  • 前もってデータベースに株価の情報と25日平均線と75日平均線が登録されている状態から開始します
  • Djangoを使用します
  • Djangoのmodelは以下のようになっております
models.py
class Company(models.Model):
    code = models.IntegerField("銘柄コード", primary_key=True)
    name = models.CharField("会社名", max_length=200)

class RawPrices(models.Model):
    code = models.IntegerField("銘柄コード")
    date = models.DateField("日付")
    open_price = models.IntegerField("始値")
    close_price = models.IntegerField("終値")
    high_price = models.IntegerField("高値")
    low_price = models.IntegerField("安値")
    volume = models.IntegerField("出来高")
    adjustment_close_price = models.FloatField("調整後終値")
    moving_averages25 = models.FloatField("25日移動平均線")
    moving_averages75 = models.FloatField("75日移動平均線")

実装

  • サイボウズの2017/1/1から2019/9/27までの間でゴールデンクロスが起きた日を特定します
  • DjangoのCommand機能を使用します
from django.core.management.base import BaseCommand
from app.models import RawPrices
import pandas as pd

import logging
logger = logging.getLogger(__name__)

class Command(BaseCommand):
    help = ''

    def handle(self, *args, **options):
        self.print_cross_date()

    def print_cross_date(self):
        """
        ゴールデンクロスが起きた日付を表示
        """
        CODE = 4776 #サイボウズの証券コード
        START_DATE = "2017-12-29" #東証が2017/12/30から2018/1/3まで休みの為、2018/1/1からのゴールデンクロスを求める為に2017/12/29を指定
        raw_prices = self.get_raw_price(CODE, START_DATE)
        # 25日平均線のDataFrameを作成
        series1 = pd.DataFrame(list(map(lambda x: [x.date, x.moving_averages25], raw_prices)),
                               columns=["date", "sma"])
        series1.set_index("date", inplace=True)
        # 75日平均線のDataFrameを作成
        series2 = pd.DataFrame(list(map(lambda x: [x.date, x.moving_averages75], raw_prices)),
                               columns=["date", "sma"])
        series2.set_index("date", inplace=True)
        cross_date_list, cross_result_list = self.search_golden_cross_date(series1, series2)
        # 結果を表示
        for i in cross_date_list:
            print(i)

    def get_raw_price(self, code, start_date):
        # データベースから株価のデータを取得
        raw_prices = RawPrices.objects.filter(code=code)\
            .filter(date__gte=start_date).exclude(moving_averages75=-1).order_by("date")
        return raw_prices

    def search_golden_cross_date(self, series1: pd.Series, series2: pd.Series):
        cross_date_list = []
        cross_result_list = []
        for i in range(series1.size):
            # 2日毎のデータに分割してゴールデンクロスを算出
            result = self.crossover(series1=series1[i:i+2], series2=series2[i:i+2])
            cross_result_list.append([series2.iloc[[i]].index.values[0], result])
            if result:
                cross_date_list.append(series2.iloc[[i+1]].index.values[0])
        return cross_date_list, cross_result_list

    def crossover(self, series1: pd.Series, series2: pd.Series) -> bool:
        """
        ゴールデンクロスが起きたらTrueを返す
        """
        if(series1.size < 2):
            return False
        try:
            return series1.sma[0] < series2.sma[0] and series1.sma[1] > series2.sma[1]
        except IndexError:
            return False

実行結果

python manage.py golden_cross_notification
2018-01-04
2018-05-21
2018-08-20
2019-03-08

実際に楽天証券のチャートで確認。正しく算出できているようです。
テクニカルチャート.png

ゴールデンクロスの情報をメールで通知する

  • 1日1回ゴールデンクロスが起きている株式の情報をメールで通知する機能を実装してみました
  • 以下のコードをcronなどで一日一回実行するとゴールデンクロスが起きた株式の情報がメールで届きます(DBに保存されている株価の情報は別途更新する必要があります)
from django.core.management.base import BaseCommand
from app.models import RawPrices, Company
import pandas as pd
from django.core.mail import send_mail
from django.conf import settings

import logging
logger = logging.getLogger(__name__)

class Command(BaseCommand):
    help = ''

    def handle(self, *args, **options):
        self.notification()

    def notification(self):
        """
        ゴールデンクロスの情報をメールで送信
        """
        companies = Company.objects.all()
        notification_list = []
        for company in companies:
            raw_prices = RawPrices.objects.filter(code=company.code).order_by("-date")[:2]
            series1 = pd.DataFrame(list(map(lambda x: [x.date, x.moving_averages25], raw_prices)),
                                   columns=["date", "sma"])
            series1.set_index("date", inplace=True)
            series2 = pd.DataFrame(list(map(lambda x: [x.date, x.moving_averages75], raw_prices)),
                                   columns=["date", "sma"])
            series2.set_index("date", inplace=True)
            if self.crossover(series1, series2):
                notification_list.append(company)
        if len(notification_list) > 0:
            self.send_mail(notification_list)

    def crossover(self, series1: pd.Series, series2: pd.Series) -> bool:
        """
        ゴールデンクロスが起きたらTrueを返す
        """
        if(series1.size < 2):
            return False
        try:
            return series1.sma[1] < series2.sma[1] and series1.sma[0] > series2.sma[0]
        except IndexError:
            return False

    def send_mail(self, company_list):
        message = ""
        for company in company_list:
            text = "証券コード : {}\n".format(company.code)
            message+=text
        subject = "ゴールデンクロス情報"
        from_email = settings.FROM_EMAIL  # 送信者
        recipient_list = settings.RECIPIENT_LIST  # 宛先リスト
        send_mail(subject, message, from_email, recipient_list)

実行すると以下のようなメールが届きます。

タイトル:ゴールデンクロス情報
本文
証券コード : 4443
証券コード : 4776

Djangoでメールを送信するための設定はこちらの記事が参考になります。
https://narito.ninja/blog/detail/64/

3
10
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
3
10

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?