1. 程式人生 > >Django(三) ORM 資料庫操作

Django(三) ORM 資料庫操作

大綱

一、DjangoORM 建立基本型別及生成資料庫表結構
1、簡介
2、建立資料庫 表結構
二、Django ORM基本增刪改查
1、表資料增刪改查
2、表結構修改
三、Django ORM 欄位型別
1、欄位型別介紹
2、欄位引數介紹
3、Django ORM 外來鍵操作

一、DjangoORM 建立基本型別及生成資料庫表結構

1、簡介

ORM:關係物件對映。定義一個類自動生成資料庫的表結構。

建立資料庫的時候,一般有以下幾種常用資料型別:數字、字串以及時間。

ORM分為兩種:

  • DB First 資料庫裡先建立資料庫表結構,根據表結構生成類,根據類操作資料庫
  • Code First 先寫程式碼,執行程式碼建立資料庫表結構

主流的orm都是code first。django 的orm也是code first,所以學的時候,本質就分為兩塊:

  • 根據類自動建立資料庫表
  • 根據類對資料庫表中的資料進行各種操作

2、建立資料庫 表結構

  • 先寫類:

app下models.py檔案:

from django.db import models

class UserInfo(models.Model):  # 必須繼承models.Model
    # 不寫則,django預設建立ID列,自增,主鍵
    # 使用者名稱列,字串型別,指定長度
username = models.CharField(max_length=32) password = models.CharField(max_length=64)
  • 註冊APP

執行命令python manage.py makemigrations,會提示No changes detected,這是因為:執行命令時會找所有models,但是django不知道找哪個,所有需要指定一下。

settings.py,

INSTALLED_APPS = [
    'django.contrib.admin',
    'django.contrib.auth'
, 'django.contrib.contenttypes', 'django.contrib.sessions', 'django.contrib.messages', 'django.contrib.staticfiles', 'app01', # 這裡新增app ]
  • 執行命令生成表結構
python manage.py  makemigrations    # 生成migrations臨時檔案
python manage.py  migrate           # 根據migrations直接生成資料庫

執行後,從migrations便會產生操作記錄。生成資料庫,預設情況下用的sqlite3。可以用navicat等軟體直接開啟。

db.sqlite3裡面包含快取、session、cookie、靜態檔案以及後臺管理。像咱們建立的表名叫:app01_userinfo

  • 連線mysql資料庫

如果不使用sqlite,使用mysql資料庫。程式碼不用改、命令也不用改,只需要改下配置檔案。

# https://docs.djangoproject.com/en/1.10/ref/settings/#databases  官網文件
# DATABASES = {             # sqlite 預設
#     'default': {
#         'ENGINE': 'django.db.backends.sqlite3',
#         'NAME': os.path.join(BASE_DIR, 'db.sqlite3'),
#     }
# }
DATABASES = {               # mysql
    'default': {
        'ENGINE': 'django.db.backends.mysql',
        'NAME': 'mydatabase',
        'USER': 'mydatabaseuser',
        'PASSWORD': 'mypassword',
        'HOST': '127.0.0.1',
        'PORT': '3306',
    }
}

注意:

  • 資料庫名django不能建立,需要自己提前建立

  • Django預設使用MySQLdb模組連結MySQL,但python3現在還沒有MySQLdb,所以改為用pymysql去連。在project專案名下的__init__.py裡面

    import pymysql
    pymysql.install_as_MySQLdb()

二、Django ORM基本增刪改查

1、表資料增刪改查

urls.py

    url(r'^orm/', views.orm),

app01/views.py

from app01 import models  # 匯入models模組
def orm(request):
# 建立資料
    # 第一種方式
    # models.UserInfo.objects.create(username="root",password="123")
    # 第二種方式
    # obj = models.UserInfo(username='fzh', password="iajtag")
    # obj.save()
    # 第三種方式
    # dic = {'username':'fgf', 'password':'666'}
    # models.UserInfo.objects.create(**dic)

# 查詢資料
    # result = models.UserInfo.objects.all()  # 查詢所有,為QuerySet型別,可理解成列表
    # result = models.UserInfo.objects.filter(username="fgf",password="666")  # 列表
    # result = models.UserInfo.objects.filter(username="fgf").first()  # 物件
    # 條件查詢。filter 相當於where查詢條件,裡面的","會組成and條件
    # for row in result:  # 列印查詢到資料。
    #     print(row.id,row.username,row.password)

    # 檢視QuerySet型別具體做了什麼事情,可以: print(result.query)

# 刪除資料
    # models.UserInfo.objects.all().delete()  # 刪除所有
    # models.UserInfo.objects.filter(id=4).delete()  # 刪除所有

# 更新資料
    # models.UserInfo.objects.all().update(password=8888)
    # models.UserInfo.objects.filter(id=3).update(password=888888)

    return HttpResponse('orm')

2、表結構修改

  • 修改列

把最初定義的類中欄位password = models.CharField(max_length=64)改為password = models.CharField(max_length=60)。重新執行python manage.py makemigrationspython manage.py migrate,則資料表結構更改了。如果列內內容超過定義大小,則資料就丟了。

  • 增加一列

類中增加一列,執行命令,會有提示資訊,因為預設情況下是不允許為空的。這裡提供兩個選項。

  1. 輸入一個值,新增列已存在的行預設加上輸入的內容。
  2. email = models.CharField(max_length=32, null=True),允許為空

查看錶結構變化時,重新整理看不出來,需要重新開啟表看效果。

  • 刪除列

類內刪掉相應的欄位,執行命令即可。

三、Django ORM 欄位型別

1、欄位型別介紹

Django ORM 欄位型別中,有CharField、EmailField、URLField、GenericIPAddressField等,實際上在資料庫裡都是字串。不能做檢查語法,做不了驗證。那這個有什麼用呢?

這些是給Django 的 admin 用的。在admin那個網頁上做驗證。如果不用admin,那那些都是字串,效果都一樣。

自定義自增列models.AutoField(primary_key=True)

1、models.AutoField  自增列 = int(11)
  如果沒有的話,預設會生成一個名稱為 id 的列,如果要顯示的自定義一個自增列,必須將給列設定為主鍵 primary_key=True2、models.CharField  字串欄位
  必須 max_length 引數
3、models.BooleanField  布林型別=tinyint(1)
  不能為空,Blank=True
4、models.ComaSeparatedIntegerField  用逗號分割的數字=varchar
  繼承CharField,所以必須 max_lenght 引數
5、models.DateField  日期型別 date
  對於引數,auto_now = True 則每次更新都會更新這個時間;auto_now_add 則只是第一次建立新增,之後的更新不再改變。
6、models.DateTimeField  日期型別 datetime
  同DateField的引數
7、models.Decimal  十進位制小數型別 = decimal
  必須指定整數位max_digits和小數位decimal_places
8、models.EmailField  字串型別(正則表示式郵箱) =varchar
  對字串進行正則表示式
9、models.FloatField  浮點型別 = double
10、models.IntegerField  整形
11、models.BigIntegerField  長整形
  integer_field_ranges = {
    'SmallIntegerField': (-32768, 32767),
    'IntegerField': (-2147483648, 2147483647),
    'BigIntegerField': (-9223372036854775808, 9223372036854775807),
    'PositiveSmallIntegerField': (0, 32767),
    'PositiveIntegerField': (0, 2147483647),
  }
12、models.IPAddressField  字串型別(ip4正則表示式)(已棄用,用13、)
13、models.GenericIPAddressField  字串型別(ip4和ip6是可選的)
  引數protocol可以是:both、ipv4、ipv6
  驗證時,會根據設定報錯
14、models.NullBooleanField  允許為空的布林型別
15、models.PositiveIntegerFiel  正Integer
16、models.PositiveSmallIntegerField  正smallInteger
17、models.SlugField  減號、下劃線、字母、數字
18、models.SmallIntegerField  數字
  資料庫中的欄位有:tinyint、smallint、int、bigint
19、models.TextField  字串=longtext
20、models.TimeField  時間 HH:MM[:ss[.uuuuuu]]
21、models.URLField  字串,地址正則表示式
22、models.BinaryField  二進位制
23、models.ImageField   圖片
24、models.FilePathField 檔案

如上那麼多的欄位,可大致分為 字串、數字、時間、二進位制、自增(primary_key=True)幾類。

2、欄位引數介紹

  • 資料庫中的欄位引數
null                # 是否可以為空
default             # 預設值
primary_key         # 主鍵
db_column           # 列名
db_index            # 索引(db_index=True)
unique              # 唯一索引(unique=True)
unique_for_date     # 只對日期索引
unique_for_month    # 只對月份索引
unique_for_year     # 只對年做索引
auto_now            # 建立時,自動生成時間
auto_now_add        # 更新時,自動更新為當前時間
        # 更新時間不支援這種
        obj = UserGroup.objects.filter(id=1).update(caption='CEO')  
        obj = UserGroup.objects.filter(id=1).first()  # 自動更新時間需要這樣寫
        obj.caption = "CEO"
        obj.save()
  • 以下是隻針對admin的欄位引數
choices             # 作用:1、django admin中顯示下拉框;2、避免連表查詢
        user_type_choices = (  # 資料庫只存1、2、3,後面的資訊存在記憶體裡。
            (1, '超級使用者'),
            (2, '普通使用者'),
            (3, '普普通使用者'),
        )
        user_type_id = models.IntegerField(choices=user_type_choices,default=1)
blank              # django admin是否可以為空
verbose_name       # django admin顯示欄位中文
editable           # django admin是否可以被編輯
error_messages     # 錯誤資訊
    # error_messages={"required":"密碼不能為空",}  # 注意必須有逗號
help_text          # django admin提示
validators         # django form ,自定義錯誤資訊

python manage.py createsuperuser    # 建立 Django 使用者

3、Django ORM 外來鍵操作

連表關係之一對多,models.ForeignKey(ColorDic)

models.py

  • 表關聯
class UserGroup(models.Model):
    uid = models.AutoField(primary_key=True)
    caption = models.CharField(max_length=32,unique=True)
    ctime = models.DateTimeField(auto_now_add=True, null=True)
    uptime = models.DateTimeField(auto_now=True, null=True)

class UserInfo(models.Model):
    username = models.CharField(max_length=32,blank=True,verbose_name='使用者名稱')
    password = models.CharField(max_length=60, help_text='pwd')
    email = models.CharField(max_length=60)
    test = models.EmailField(max_length=19,null=True,error_messages={'invalid': '請輸入密碼',})
    # UserInfo表中沒有user_group欄位,而是 user_group_id 列 值為 uid 數字
    user_group = models.ForeignKey("UserGroup",to_field='uid')      # 外來鍵關聯 **********
  • 資料查詢
user_list = Userinfo.objects.all()  # 獲取Userinfo物件
for row in user_list:               # 
    print(row.user_group_id)        # 資料庫裡真實存在的資料
    # user_group:代指UserGroup物件。類UserGroup物件裡封裝了(uid,catption,ctime,uptime)
    print(row.user_group.uid)       # 通過物件獲取uid,和user_group_id一樣
    print(row.user_group.caption)   # 通過物件獲取caption
  • 建立資料

UserInfo表建立資料,怎麼寫呢?

    models.UserInfo.objects.create(
        username='root1',
        password='123',
        email="asdfasdf",
        test="asdfasdf",
        # 第一種方式:再次查詢資料庫,不推薦
        # user_group = models.UserGroup.objects.filter(id=1).first()
        # 第二種方式:通過 外來鍵欄位_id
        user_group_id = 1
    )