热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

python金融建模大赛_【用Python金融建模】从二叉树谈起:衍生品Option期权定价模型的构建...

内容首发乐学偶得(http:lexueoude.com)公众号:乐学Fintech用代码理解分析解决金融问题在金融里面很多地方都出现过一个理念就是“货币的时间价值”&

内容首发

乐学偶得(http://lexueoude.com) 公众号: 乐学Fintech

用代码理解分析解决金融问题

在金融里面很多地方都出现过一个理念就是“货币的时间价值”,例如我们之前聊过的利用Python对项目进行估值判断 就是利用这一重要的思想:我们做出的决定,都是把未来的一系列现金流的【流入】和【流出】进行折现,通过我们理性人在做决定的时候,是选择对我们有利的事情——也就是折现到现在NPV为正的项目去做。

那么这个世界如果真的这么简单就好了,我们只需要把未来的现金流折现到现在,然后就可以愉快的进行判断了。

可是问题来了:未来并不确定。

而且万事万物都是相互联系的,也就是说A可能导致B,B导致C,这样蝴蝶效应传递下去,一个很小的偏差或者错误的估计会导致巨大的差错。

二叉树是我们对未来进行预判的一种思考方式,也是金融里面常用的模型,二叉树的思想如下:

1.事情的发展分成一个阶段一个阶段

2.每一个阶段只有两个结果:变好和变坏

如果这是一个项目,我们可以把项目通过二叉树进行估计,最好的情况,好成什么样,最差的情况,差成什么样,一目了然:项目估值

还有一个非常重要的运用就是在拟合或者模拟未来股价的趋势上,股价本质上就是一系列的时间序列,所以说与项目的估值也是一类东西:随着时间的变化,价格随之改变股价预测

我们下面就来介绍这样一个思考问题的方法与原理,并且通过Python进行建模计算:

二叉树的思想就是把未来时间轴分为一期一期,在每一期内,只有价格只有两种变化方向,向上,或者向下。我们从一期二叉树开始思考:

一期二叉树

假设初始价格为S0,在第一期的时间内,有P的概率价格会上升,成为S0*u 的价格,因为在这段时间内只有两种可能,所以剩下

的1-P的概率价格会下跌,成为S0*d的价格。

u与d分别为up factor(上涨因子)与down factor(下跌因子),(我明白,涉及到公式经常会加什么什么因子,导致这个名字很山寨,但是因为多期二叉树中会用到,两个因子造成的影响也可能不一样,所以我们暂且尽量用一般公式与常见叫法去表示)

另外,我们为了多期二叉树能够重合,也就是上涨下跌与下跌上涨的结果一样,我们将u与d两个因子设立一个约束条件u*d=1

这样,我们从一期推广到多期,就可以得到如下流程图:多期二叉树

也就是说,我们只要能知道p(上涨概率)与u(上涨因子),就能求出1-p(下跌概率),与d(因为约束关系),这样整个未来的价格的路径就能够知道了。

好了,这样我们就知道最简单的标的资产,通过二叉树的方法进行预测未来变动,是什么样的路径了。

我们接下来再把问题弄复杂一点,假设我们研究的不是简单的标的资产,比如说并不是一个单一的项目(可以看做一系列的未来现金流)或者是一个简单的股票,而是衍生品,也就是说定价是衍生在原标的资产上的产品,那我们如何进行定价呢?

这就涉及到了二叉树运用非常多的一个方面——衍生品定价。

我们接下来还是以例子,只不过我们研究的不仅仅是标的资产(比如说股票stock),而是衍生在标的资产上的金融产品(比如说期权option),我们该如何定价呢?标的资产二叉树标的资产上衍生的call option

对于Call Option(看涨期权),其实价格非常好计算,X为执行价格,就是max(X-S0u^1,0)或者是max(X-S0d^1,0),也就是说要么为执行价格与股价的差价,要么为0.

这样,我们在0时刻的期权价格也非常好计算了,

期权价格= [ p ×上涨时期权价格 + (1−p) ×下跌时期权价格] × 折现因子

当然折现因子在这里可以用连续复利也可以用离散复利的方法折现,比较推荐连续复利,因为更加准确。利用连续复利的方法进行折现

那么我们到这里已经完全具备了建模的理论基础了,我们接下来用Python建立二叉树模型进行定价计算,这样不仅能巩固我们的概念,也能锻炼我们的建模能力:

我们首先可以先调整numpy的显示,因为如果期数较多的情况下jupyter notebook中会截断大量数据,我们想让数据全部显示出来:

import sys

import numpy

numpy.set_printoptions(threshold=sys.maxsize)

然后我们只需要numpy帮我们储存一下数据并且计算即可,当然也可以用pandas,但是因为数据结构并不复杂,有点大材小用,我们用array就能搞定了。

import numpy as np

我们现在可以自定义一个函数,把可以在市场上观测到的信息作为参数:

def binomial_tree_call_option_pricing_model(N,T,S0,sigma,r,K,show_array=False):

dt=T/N

u=np.exp(sigma*np.sqrt(dt))

d=1/u

p=(np.exp(r*dt)-d)/(u-d)

其中N为期数,T为总时间,S0为初始价格,sigma为波动率,r为无风险利率,K为执行价格,show_array为一个开关,我们可以用show_array 来控制,是否显示array。

dt为将总时间通过除以期数切分为每一个小段,每一个小段为一期二叉树。

u为上涨多少,这个是根据上述讲到的公式来确定的,同样,为了二叉树能重合,d=1/u,通过约束关系也能表示出来。

p为上涨概率,同样根据上述讲到的公式来确定的上述讲到的公式

接着我们需要创建一个容器去储存我们的数据,我可以用numpy的zero生成全部是zero的矩阵用作存放数据的容器:

但是因为这个容器是需要随着数据的多少而随之变化,也就是不能是固定的大小,于是我们可以通过变量进行解决:

price_tree=np.zeros([N+1,N+1])

然后我们就可以将这个容器里面的数据填满了,我们首先填入的是价格,也就是比如标的资产是股票的话,股票在未来T时间段里的每一个dt期间变动的可能价格,我们全部放入:

for i in range(N+1):

for j in range(i+1):

price_tree[j,i]=S0*(d**j)*(u**(i-j))

这里有一个nested for loop的技巧,我们在很多list of lists, array等里面都会用到。另外,价格上升也下降我们通过N+1和i+1确定range,i和j的约束关系控制上涨个下跌的次数

接着我们依葫芦画瓢的方法先建立起option价格的tree:

我们首先计算每一个期权价格:

option_tree=np.zeros([N+1,N+1])

option_tree[:,N]=np.maximum(np.zeros(N+1),price_tree[:,N]-K)

然后我们把期权价格通过反向进行折现,求出t=0时刻的价格:

for i in np.arange(N-1,-1,-1):

for j in np.arange(0,i+1):

option_tree[j,i]=np.exp(-r*dt)*(p*option_tree[j,i+1]+(1-p)*option_tree[j+1,i+1])

然后我们就可以直接返回array的值就可以了,因为我们在参数中直接将show_array设置成了False,所以是默认不显示出整个树状结构的:

if show_array:

return [option_tree[0,0],np.round(price_tree),np.round(option_tree)]

else:

return option_tree[0,0]

这样我们能直接返回这个期权定价的初始价值,如果我们将show_array调成True,则执行第一段,将整个树状结构也就是每一个标的资产价值显示出,当然,我们为了美观,将array中的结果进行了round,没有保留小数。

当然,我们还没有讨论一个问题,就是上涨概率和下跌概率与风险中性定价原则,我们在接下来为大家介绍......未完待续......

以上为《Python零基础入门编程的新世界》部分内容笔记。用Python做爬虫,数据分析,全栈建设,Fintech金融量化,机器学习,办公自动化,树莓派,美好生活DIY,......2000+连载,不仅有编程,还有更多原理讲解。零基础Python入门编程全栈量化AI - 网易云课堂​study.163.com

另外,也欢迎关注一个非常干的干货公众号: 乐学Fintech用代码理解分析解决金融问题



推荐阅读
  • 微软头条实习生分享深度学习自学指南
    本文介绍了一位微软头条实习生自学深度学习的经验分享,包括学习资源推荐、重要基础知识的学习要点等。作者强调了学好Python和数学基础的重要性,并提供了一些建议。 ... [详细]
  • 提升Python编程效率的十点建议
    本文介绍了提升Python编程效率的十点建议,包括不使用分号、选择合适的代码编辑器、遵循Python代码规范等。这些建议可以帮助开发者节省时间,提高编程效率。同时,还提供了相关参考链接供读者深入学习。 ... [详细]
  • 学习SLAM的女生,很酷
    本文介绍了学习SLAM的女生的故事,她们选择SLAM作为研究方向,面临各种学习挑战,但坚持不懈,最终获得成功。文章鼓励未来想走科研道路的女生勇敢追求自己的梦想,同时提到了一位正在英国攻读硕士学位的女生与SLAM结缘的经历。 ... [详细]
  • Spring源码解密之默认标签的解析方式分析
    本文分析了Spring源码解密中默认标签的解析方式。通过对命名空间的判断,区分默认命名空间和自定义命名空间,并采用不同的解析方式。其中,bean标签的解析最为复杂和重要。 ... [详细]
  • 向QTextEdit拖放文件的方法及实现步骤
    本文介绍了在使用QTextEdit时如何实现拖放文件的功能,包括相关的方法和实现步骤。通过重写dragEnterEvent和dropEvent函数,并结合QMimeData和QUrl等类,可以轻松实现向QTextEdit拖放文件的功能。详细的代码实现和说明可以参考本文提供的示例代码。 ... [详细]
  • 本文分享了一个关于在C#中使用异步代码的问题,作者在控制台中运行时代码正常工作,但在Windows窗体中却无法正常工作。作者尝试搜索局域网上的主机,但在窗体中计数器没有减少。文章提供了相关的代码和解决思路。 ... [详细]
  • 推荐系统遇上深度学习(十七)详解推荐系统中的常用评测指标
    原创:石晓文小小挖掘机2018-06-18笔者是一个痴迷于挖掘数据中的价值的学习人,希望在平日的工作学习中,挖掘数据的价值, ... [详细]
  • 计算机存储系统的层次结构及其优势
    本文介绍了计算机存储系统的层次结构,包括高速缓存、主存储器和辅助存储器三个层次。通过分层存储数据可以提高程序的执行效率。计算机存储系统的层次结构将各种不同存储容量、存取速度和价格的存储器有机组合成整体,形成可寻址存储空间比主存储器空间大得多的存储整体。由于辅助存储器容量大、价格低,使得整体存储系统的平均价格降低。同时,高速缓存的存取速度可以和CPU的工作速度相匹配,进一步提高程序执行效率。 ... [详细]
  • Python正则表达式学习记录及常用方法
    本文记录了学习Python正则表达式的过程,介绍了re模块的常用方法re.search,并解释了rawstring的作用。正则表达式是一种方便检查字符串匹配模式的工具,通过本文的学习可以掌握Python中使用正则表达式的基本方法。 ... [详细]
  • 本文介绍了Java的集合及其实现类,包括数据结构、抽象类和具体实现类的关系,详细介绍了List接口及其实现类ArrayList的基本操作和特点。文章通过提供相关参考文档和链接,帮助读者更好地理解和使用Java的集合类。 ... [详细]
  • 本文讨论了微软的STL容器类是否线程安全。根据MSDN的回答,STL容器类包括vector、deque、list、queue、stack、priority_queue、valarray、map、hash_map、multimap、hash_multimap、set、hash_set、multiset、hash_multiset、basic_string和bitset。对于单个对象来说,多个线程同时读取是安全的。但如果一个线程正在写入一个对象,那么所有的读写操作都需要进行同步。 ... [详细]
  • C++语言入门:数组的基本知识和应用领域
    本文介绍了C++语言的基本知识和应用领域,包括C++语言与Python语言的区别、C++语言的结构化特点、关键字和控制语句的使用、运算符的种类和表达式的灵活性、各种数据类型的运算以及指针概念的引入。同时,还探讨了C++语言在代码效率方面的优势和与汇编语言的比较。对于想要学习C++语言的初学者来说,本文提供了一个简洁而全面的入门指南。 ... [详细]
  • 本文介绍了H5游戏性能优化和调试技巧,包括从问题表象出发进行优化、排除外部问题导致的卡顿、帧率设定、减少drawcall的方法、UI优化和图集渲染等八个理念。对于游戏程序员来说,解决游戏性能问题是一个关键的任务,本文提供了一些有用的参考价值。摘要长度为183字。 ... [详细]
  • 如何实现织梦DedeCms全站伪静态
    本文介绍了如何通过修改织梦DedeCms源代码来实现全站伪静态,以提高管理和SEO效果。全站伪静态可以避免重复URL的问题,同时通过使用mod_rewrite伪静态模块和.htaccess正则表达式,可以更好地适应搜索引擎的需求。文章还提到了一些相关的技术和工具,如Ubuntu、qt编程、tomcat端口、爬虫、php request根目录等。 ... [详细]
  • Tomcat/Jetty为何选择扩展线程池而不是使用JDK原生线程池?
    本文探讨了Tomcat和Jetty选择扩展线程池而不是使用JDK原生线程池的原因。通过比较IO密集型任务和CPU密集型任务的特点,解释了为何Tomcat和Jetty需要扩展线程池来提高并发度和任务处理速度。同时,介绍了JDK原生线程池的工作流程。 ... [详细]
author-avatar
手机用户2602923713
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有