Django User 模塊之 AbstractUser 擴(kuò)展詳解
最近在寫博客,剛好寫到用戶注冊注銷模塊,覺得這一方面還是挺有趣的。當(dāng)嘗試掀開 Django 的源代碼時一切 API 就不會變得那么摸不著。順著讀Django 的各模塊源碼,我們可以更靈活地更改代碼以實(shí)現(xiàn)自己想要的功能。
現(xiàn)在,思考一個問題,主需求是實(shí)現(xiàn)博客中用戶的注冊登錄登出功能。如果只滿足于用戶注冊時只登記其郵箱或是用戶名,Django 自帶的 User 模塊既可以實(shí)現(xiàn)。但實(shí)際上,一個普遍的要求是注冊用戶應(yīng)該能夠修改自己的頭像信息,郵箱信息,昵稱信息等其他更靈活的需求。
可以先看一下 Django User 模塊的源碼
class User(AbstractUser): """ Users within the Django authentication system are represented by this model. Username, password and email are required. Other fields are optional. """ class Meta(AbstractUser.Meta): swappable = 'AUTH_USER_MODEL'
注意:如果你的是 Anaconda 管理,可以在路徑 C:\Users\User\Anaconda3\Lib\site-packages\django\contrib\auth\models.py 查看
Django 中的 User 模塊實(shí)際上繼承了 AbstractUser 模塊,AbstractUser 模塊下有 :
username
first_name
last_name
email
date_joined
…
你可以看出,User 模塊繼承了 AbstractUser 抽象基類,而僅僅只是繼承了,并沒有對 AbstractUser 進(jìn)行任何擴(kuò)展。所以,對于一個需要更多需求的 User 模塊信息來說,我們可以繼承 AbstractUser 并根據(jù)自己的需求進(jìn)行擴(kuò)展。
現(xiàn)在,我們對用戶屬性添加一些需求,比如支持用戶修改頭像、支持用戶昵稱、qq、wechat 以及網(wǎng)站鏈接等。
class User(AbstractUser): nickname = models.CharField(max_length=30, blank=True, null=True, verbose_name='昵稱') qq = models.CharField(max_length=20, blank=True, null=True, verbose_name='QQ號碼') url = models.URLField(max_length=100, blank=True, null=True, verbose_name='個人網(wǎng)頁地址') avatar = ProcessedImageField(upload_to='avatar', default='avatar/default.png', verbose_name='頭像') class Meta: verbose_name = '用戶' verbose_name_plural = verbose_name ordering = ['-id'] def __str__(self): return self.username
我們給自定義的用戶模塊增加 nickname(昵稱), qq, url(網(wǎng)站鏈接),avatar(頭像)屬性。
注意:為了讓 Django 能夠識別使用自定義的用戶模型,必須要在 settings.py 中設(shè)置自定義模塊位置,如在 settings.py 上添加
AUTH_USER_MODEL = 'blog.user'
其中,blog 為你對應(yīng)的應(yīng)用 app 信息,user 為 blog 應(yīng)用下的 user 模塊,在這里 blog 和 user 大小寫無關(guān)。
如果在你現(xiàn)在執(zhí)行數(shù)據(jù)庫遷移命令,可能會出現(xiàn) blog 不存在 user 模塊 的提示,而無法重新進(jìn)行數(shù)據(jù)遷移。
ValueError: The field account.EmailAddress.user was declared with a lazy reference to 'blog.user', but app 'blog' doesn't provide model 'user'. The field admin.LogEntry.user was declared with a lazy reference to 'blog.user', but app 'blog' doesn't provide model 'user'. The field blog.Article.author was declared with a lazy reference to 'blog.user', but app 'blog' doesn't provide model 'user'. The field easy_comment.Comment.user was declared with a lazy reference to 'blog.user', but app 'blog' doesn't provide model 'user'. The field easy_comment.Like.user was declared with a lazy reference to 'blog.user', but app 'blog' doesn't provide model 'user'. The field notifications.Notification.recipient was declared with a lazy reference to 'blog.user', but app 'blog' doesn't provide model 'user'. The field online_status.OnlineStatus.user was declared with a lazy reference to 'blog.user', but app 'blog' doesn't provide model 'user'. The field socialaccount.SocialAccount.user was declared with a lazy reference to 'blog.user', but app 'blog' doesn't provide model 'user'.
所以,如果之前是使用例如 AUTH_USER_MODEL = auth.user 的用戶模型,并重新將其自定義為 AUTH_USER_MODEL = blog.user 請刪掉 migrations 目錄下的所有文件 以及數(shù)據(jù)庫文件。
刪除之后,重新進(jìn)行數(shù)據(jù)庫的遷移
$ python manage.py makemigrations myapp $ python manage.py migrate
這個時候,所使用的用戶即為自定義后的用戶了。
File "C:\Users\Micky\Anaconda3\lib\site-packages\django\db\backends\utils.py", line 85, in _execute return self.cursor.execute(sql, params) File "C:\Users\Micky\Anaconda3\lib\site-packages\django\db\backends\sqlite3\base.py", line 303, in execute return Database.Cursor.execute(self, query, params) django.db.utils.OperationalError: no such table: blog_user
這里可以在模板中指定數(shù)據(jù)庫 db_table = 'user'
補(bǔ)充知識:Django學(xué)習(xí)筆記——內(nèi)置用戶類AbstractUser與內(nèi)置認(rèn)證校驗(yàn)系統(tǒng)
內(nèi)置用戶類AbstractUser
我們在之前講過了model模型的作用和父類的作用,這次介紹的內(nèi)置用戶類AbstractUser就是Django內(nèi)置的一個關(guān)于用戶操作的類,它極大地方便了我們對model模型中對User用戶類的設(shè)計(jì)。而所謂內(nèi)置用戶類的本質(zhì)也就是一個封裝好的父類,所以使用起來是相當(dāng)?shù)姆奖恪?/p>
#導(dǎo)入AbstractUser類 from django.contrib.auth.models import AbstractUser #直接繼承就可以了,如果有需要就向?qū)こodel一樣寫字段就可以 class User(AbstractUser): pass
我們通過查看AbstractUser的源碼可以知道它設(shè)有幾個字段
#用戶名 username = models.CharField( _('username'), max_length=150, unique=True, help_text=_('Required. 150 characters or fewer. Letters, digits and @/./+/-/_ only.'), validators=[username_validator], error_messages={ 'unique': _("A user with that username already exists."), }, ) #名 first_name = models.CharField(_('first name'), max_length=30, blank=True) #姓 last_name = models.CharField(_('last name'), max_length=150, blank=True) #郵箱 email = models.EmailField(_('email address'), blank=True) #權(quán)限 is_staff = models.BooleanField( _('staff status'), default=False, help_text=_('Designates whether the user can log into this admin site.'), ) #激活 is_active = models.BooleanField( _('active'), default=True, help_text=_( 'Designates whether this user should be treated as active. ' 'Unselect this instead of deleting accounts.' ), ) #日期 date_joined = models.DateTimeField(_('date joined'), default=timezone.now)
內(nèi)置認(rèn)證校驗(yàn)系統(tǒng)
django自帶的用戶認(rèn)證校驗(yàn)系統(tǒng)較為簡單,主要就是認(rèn)證用戶名密碼的正確與否
首先要在settings里面配置
#使用自帶的認(rèn)證系統(tǒng)
AUTH_USER_MODEL = "user.User"
這是配合自帶的用戶類AbstractUser一起使用的
通常使用在類視圖中的post方法校驗(yàn)用戶登錄等操作
在view中的具體代碼如下
class LoginView(View): def get(self,request): #邏輯代碼 return render(request,'login.html') def post(self,request): # 獲取前端傳遞過來的用戶名和密碼 username = request.POST.get('username') pwd = request.POST.get('pwd') record = request.POST.get('record') # 進(jìn)行數(shù)據(jù)校驗(yàn) if not all([username,pwd]): return HttpResponse('數(shù)據(jù)輸入不完整') # 驗(yàn)證用戶名和密碼是否正確 user = authenticate(username=username,password=pwd) return render(request,''index.html')
主要就是其中的user = authenticate(username=username,password=pwd)
兩個參數(shù)都是拿到前端用戶輸入的信息
以上這篇Django User 模塊之 AbstractUser 擴(kuò)展詳解就是小編分享給大家的全部內(nèi)容了,希望能給大家一個參考,也希望大家多多支持腳本之家。
- JavaSE static final及abstract修飾符實(shí)例解析
- PHP中abstract(抽象)、final(最終)和static(靜態(tài))原理與用法詳解
- Python2和Python3中@abstractmethod使用方法
- JAVA抽象類和抽象方法(abstract)實(shí)例分析
- Springboot源碼 AbstractAdvisorAutoProxyCreator解析
- java編程abstract類和方法詳解
- 淺談利用Spring的AbstractRoutingDataSource解決多數(shù)據(jù)源的問題
- Django如何繼承AbstractUser擴(kuò)展字段
相關(guān)文章
詳解python3實(shí)現(xiàn)的web端json通信協(xié)議
本篇文章主要介紹了python3實(shí)現(xiàn)的web端json通信協(xié)議,具有一定的參考價值,感興趣的小伙伴們可以參考一下。2016-12-12Python獲取央視節(jié)目單的實(shí)現(xiàn)代碼
這篇文章主要介紹了Python獲取央視節(jié)目單的實(shí)現(xiàn)代碼,涉及Python頁面采集的相關(guān)技巧,具有一定參考借鑒價值,需要的朋友可以參考下2015-07-07Python登錄并獲取CSDN博客所有文章列表代碼實(shí)例
這篇文章主要介紹了Python登錄并獲取CSDN博客所有文章列表代碼實(shí)例,具有一定借鑒價值,需要的朋友可以參考下2017-12-12Django的restframework接口框架自定義返回?cái)?shù)據(jù)格式的示例詳解
這篇文章主要介紹了Django的restframework接口框架自定義返回?cái)?shù)據(jù)格式,本文介紹了通過Django的restframework接口框架自定義Response返回對象來自定義返回?cái)?shù)據(jù)格式,本文通過示例代碼給大家介紹的非常詳細(xì),需要的朋友可以參考下2022-07-07詳解Python并發(fā)編程之從性能角度來初探并發(fā)編程
這篇文章主要介紹了詳解Python并發(fā)編程之從性能角度來初探并發(fā)編程,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-08-08打印tensorflow恢復(fù)模型中所有變量與操作節(jié)點(diǎn)方式
這篇文章主要介紹了打印tensorflow恢復(fù)模型中所有變量與操作節(jié)點(diǎn)方式,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-05-05淺談selenium如何應(yīng)對網(wǎng)頁內(nèi)容需要鼠標(biāo)滾動加載的問題
這篇文章主要介紹了淺談selenium如何應(yīng)對網(wǎng)頁內(nèi)容需要鼠標(biāo)滾動加載的問題,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-03-03Python深度學(xué)習(xí)神經(jīng)網(wǎng)絡(luò)殘差塊
這篇文章主要為大家介紹了Python深度學(xué)習(xí)中的神經(jīng)網(wǎng)絡(luò)殘差塊示例詳解有需要的 朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步2021-10-10關(guān)于Python的高級數(shù)據(jù)結(jié)構(gòu)與算法
這篇文章主要介紹了關(guān)于Python的高級數(shù)據(jù)結(jié)構(gòu)與算法,掌握這些數(shù)據(jù)結(jié)構(gòu)和算法將幫助我們在實(shí)際編程中解決各種問題,提高我們的編程技巧和水平,需要的朋友可以參考下2023-04-04