1. 事情起因

今天在写代码的时候,在django 的models目录中新增了一个pkg.py文件,里面定义了一个class, 在执行 makemigrations 的时候,发现这张表始终无法被识别,去网上找了找资料,没有找到这方面的问题的答案。。。

2. 排查经过

  1. 尝试把这个class放到models目录中的其他文件中,可以识别到这张表,由此怀疑可能是我新加的文件名称的原因,遂把pkg换成其他名字后,发现还是无法识别,尝试失败。。。
  2. 本着搞明白为什么会出现这种问题的心理,尝试去读makemigrations的源码,发现里面有个apps对象,保存着所有django app的信息,apps对象中有一个app_config对象,里面放的是AppConfig实例字典,后来读AppConfig源码时发现models_module和models对象嫌疑比较大,通过调试得知models_module对象只是存放了APP的models模块对象,但是models对象始终找不到是从哪里初始化的
  3. 接着扫描Apps对象的源码,发现了一个可疑的方法,get_registered_model,怀疑class的识别和这个方法应该有关系。在django源码里面全局搜索哪里调用了这个方法,发现了猫腻。这个方法在django源码里面只有在models.base.ModelBase类里面出现过,并且是在__new__方法里面调用。ModelBase类是models.Model的metaclass,于是想明白了整个过程。
  4. models里面的class都是继承自models.Model,在把这个类装载到内存中的时候,就会触发ModelBase中的new方法,也就是会自动把我们定义的class注册到AppConfig对象中,现在是没有注册进去,则说明我们定义的class没有被加载到内存中,在看源码的过程中发现AppConfig实例化的时候,会初始化models_module,这个操作会将APP中的models目录import到内存中,于是找到了解决办法,在我们的APP的models目录中的__init__.py文件中添加一行 from . import pkg,在初始化models_module的时候,导入了models目录的同时,会去执行__init__.py,而执行import的时候,会执行pkg.py文件,也就是把class加载到内存中,实现注册了。

3. 总结

前面的逻辑很绕,我也看了两个小时才把关系大致屡清楚,结论就是如果models目录中定义的py文件中的class类无法在makemigrations的时候被识别到,解决办法就是在models目录的__init__.py文件中import这个文件,就能识别到这个class类。

总的来说这次看源码大致的解决了面临的问题,但是至今无法找到为什么只有我新加的pkg.py文件未加载到内存中,被识别到并注册的原因。。。django的源码太复杂太绕了。

写这篇文章的目的是为了帮助那些采坑的童鞋们再踩到这个坑的时候,能在网上搜到这篇文章,解决你的问题,并对自己解决这个问题做一个记录,希望能够帮助到你。

django models class 不识别问题解决方案的更多相关文章

  1. Django models 操作高级补充

    Django models 操作高级补充 字段参数补充: 外键 约束取消 ..... ORM中原生SQL写法: raw connection extra

  2. Django models Form model_form 关系及区别

    Django models Form model_form

  3. Django models .all .values .values_list 几种数据查询结果的对比

    Django models .all .values .values_list 几种数据查询结果的对比

  4. django models数据类型

    Django Models的数据类型 AutoField IntegerField BooleanField true/false CharField maxlength,必填 TextField C ...

  5. django models 类型整理 version:1.8.3

    django models 类型整理 version:1.8.3 网上百度到的最上面的一篇已经是11年的了,django变化很大,现在把1.8.3版的models类型大致整理了下贴出来 普通键部分 F ...

  6. django models的点查询/跨表查询/双下划线查询

    django models 在日常的编程中,我们需要建立数据库模型 而往往会用到表与表之间的关系,这就比单表取数据要复杂一些 在多表之间发生关系的情形下,我们如何利用models提供的API的特性获得 ...

  7. Django models中关于blank与null的补充说明

    Django models中关于blank与null的补充说明 建立一个简易Model class Person(models.Model): GENDER_CHOICES=( (1,'Male'), ...

  8. Django Models的数据类型汇总

    https://blog.csdn.net/devil_2009/article/details/41735611 Django Models的数据类型 汇总 AutoField IntegerFie ...

  9. Eclipse导入MyEclipse创建的WEB项目无法识别的解决方案

    Eclipse导入MyEclipse创建的WEB项目无法识别的解决方案

随机推荐

  1. 二、Android应用的界面编程(六)ProgressBar及其子类[SeekBar、RatingBar]er

    通常用于向用户显示某个耗时操作完成的百分比.Android支持几种风格的进度条,通过style属性可以为ProgressBar指定风格.该属性支持如下几个属性值. # @android:style/W ...

  2. smarty模板 变量 运算符 表达式 流程控制 函数

    ① 从配置文件中读取配置: 1,在模板页面加载配置文件 html页面 不是php页面<{config_load file='fo.conf'}> 2,在需要用到配置的地方加<{#si ...

  3. 九度OJ 1254:N皇后问题 (N皇后问题、递归、回溯)

    时间限制:1 秒 内存限制:128 兆 特殊判题:否 提交:765 解决:218 题目描述: N皇后问题,即在N*N的方格棋盘内放置了N个皇后,使得它们不相互攻击(即任意2个皇后不允许处在同一排,同一 ...

  4. 【译】快速高效学习Java编程在线资源Top 20

    想要加强你的编程能力吗?想要提升你的 Java 编程技巧和效率吗? 不用担心.本文将会提供快速高效学习 Java 编程的 50 多个网站资源: 开始探索吧: 1.MKyong:许多开发者在这里可以找到 ...

  5. ElasticSearch(二十)定位不合法的搜索及其原因

    GET /test_index/test_type/_validate/query?explain { "query": { "math": { "t ...

  6. redis持久化AOF详细操作步骤

    1.切换到redis目录下面,创建文件 s3-redis.conf 2.编辑文件s3-redis.conf 3.终止当前redis服务端 4.登录redis客户端失败,说明服务端已停止 5.重启red ...

  7. ubuntun下安装Fiddler

    对于分析网页或者写爬虫的时候经常需要用到抓包工具进行网页数据的抓包.在Windows下可以安装Fiddler来抓包.在ubuntun下不能直接安装Fiddler.需要先安装mono 1 首先安装mon ...

  8. centos下写Symfony

    之前都是在windows上写SY,现在要部署到Linux上了,提前测试一下. 第一步,要有台Centos机器,安装过程略 第二步,安装数据库,PostgreSQL,过程; 第三步,安装版本控制器,GI ...

  9. 【C语言】Linux C调用系统命令

    最近研究深度学习,做视频分析和检测,用到C语言,以前都是写python的,不过没关系,计算机语言都是相通的,差不多原理是一样的,只是语法不太一样. 下面介绍linux C语言种调用本地命令,访问一个地 ...

  10. python基础19 -------面向对象终结篇(介绍python对象中各种内置命令)

    一.isinstance()和issubclass()命令 1.isinstance(对象,类型) 用来判定该对象是不是此类型或者说是该对象是不是此类的对象,返回结果为True和False,如图所示. ...