[英]Django - proper way to implement threaded comments
我正在使用Django开发博客站点。 我的网站将允许用户在我的任何博客文章中发表评论并互相回复,并使用“线程评论”结构进行显示(我尚未启动用户功能,仅是评论)。 我已经使用django-mptt使线程注释正常工作(至少现在是这样),但是如果我选择的路线或方向正确,我将一无所知。 当涉及到注释时,我所经历的几乎所有教程都只是从头开始,而没有涉及django中的线程注释。 对于我可能做错了什么以及我可以做得更好的事情,我需要一些经验丰富/专业的建议。 我要做的最后一件事是,经过数小时的工作,发现有一种更可接受的方法。
因此,这是我需要澄清的清单:
django-mptt:
一个HTML页面上有多种表单
很抱歉,如果这些不是菜鸟问题,并且我的帖子这么长。 我只想为初学者使用可行的解决方案以最佳方式做事。 任何反馈将有所帮助。 谢谢! 这是我的博客应用程序的代码。
models.py
from django.db import models
from mptt.models import MPTTModel, TreeForeignKey
class Post(models.Model):
"""Blog post"""
title = models.CharField(max_length=200)
body = models.TextField()
date_added = models.DateTimeField(auto_now_add=True)
def __str__(self):
return self.body[:50] + '...'
class Comment(MPTTModel):
"""User comment"""
post = models.ForeignKey(Post, related_name='comments',on_delete=models.CASCADE)
parent = TreeForeignKey('self', null=True, blank=True, related_name='children',db_index=True, on_delete=models.CASCADE)
user_comment = models.CharField(max_length=500, unique=True)
date_added = models.DateTimeField(auto_now_add=True)
# approved = models.BooleanField(default=False)
class MPTTMeta:
order_insertion_by = ['date_added']
def __str__(self):
return self.user_comment[:20]
由于某些奇怪的原因,我收到“没有这样的专栏:已批准”错误,因此将“批准”注释掉。
表格
from django import forms
from .models import Post, Comment
class CommentForm(forms.ModelForm):
class Meta:
model = Comment
fields = ['user_comment']
views.py
from django.shortcuts import render
from django.http import HttpResponseRedirect
from django.urls import reverse
from .models import Post
from .forms import CommentForm
def posts(request):
"""Show all blog posts"""
posts = Post.objects.order_by('-date_added')
context = {
'posts': posts
}
return render(request, 'posts/posts.html', context)
def post(request, post_id):
"""Show single blog post"""
post = Post.objects.get(id=post_id)
comments = post.comments.all()
if request.method != 'POST':
comment_form = CommentForm()
else:
comment_form = CommentForm(data=request.POST)
try:
parent_id = request.POST['comment_id']
except:
pass
if comment_form.is_valid():
comment = comment_form.save(commit=False)
comment.post = post
comment.parent = comments.get(id=parent_id)
comment.save()
return HttpResponseRedirect(reverse('posts:post', args=[post_id]))
context = {
'post': post,
'comment_form': comment_form,
'comments': comments,
}
return render(request, 'posts/post.html', context)
post.html
{% extends 'posts/base.html' %}
{% block blog_content %}
<h1>Post page!</h1>
<h3>{{ post.title }}</h3>
<h4>{{ post.date_added }}</h4>
<p>{{ post.body }}</p>
<form method="post" action="{% url 'posts:post' post.id %}">
{% csrf_token %}
{{ comment_form.as_p }}
<button type="submit">Add comment</button>
</form>
{% load mptt_tags %}
{% recursetree comments %}
<h5>{{ node.date_added }}</h5>
<p>{{ node.user_comment }}</p>
<form method="post" action="{% url 'posts:post' post.id %}">
{% csrf_token %}
{{ comment_form.as_p }}
<input type="hidden" name="comment_id" value="{{ node.id }}">
<button type="submit">Reply</button>
</form>
{% if not node.is_leaf_node %}
<div style="padding-left: 20px">
{{ children }}
</div>
{% endif %}
{% endrecursetree %}
{% endblock %}
urls.py
from django.urls import path
from . import views
app_name = 'posts'
urlpatterns = [
path('posts/', views.posts, name='posts'),
path('posts/<int:post_id>/', views.post, name='post'),
]
MPTT树非常适合获取子节点列表或节点数。 它们增加/插入节点的成本很高,并且成本随着三个节点的大小线性增加。 它们旨在使树数据适合关系数据库。 另外,不要被“我的读写要多得多”所迷惑。 理想情况下,大多数读取应命中高速缓存,而不是高速缓存下的数据库。
为什么不跳过关系数据库,而选择可以原生存储树的NoSQL数据库呢? Django和几乎每个NoSQL数据库都可以轻松集成。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.