基于 SerializerMethodField 对 django 对象进行排序

Sorting the django objects based on SerializerMethodField

我正在尝试根据两个用户之间最后一条消息的时间戳对用户配置文件进行排序。

我正在使用 SerializerMethodField 获取最后一条消息的时间戳。

有什么方法可以对数据进行排序吗?

class UserProfileSerializer(serializers.ModelSerializer):
    lastmessage = serializers.SerializerMethodField()
    class Meta:
        model = User
        fields = ['id','lastmessage']
    def get_lastmessage(self,obj): 
        k = self.context.get('view').kwargs['sid']
        data =( Message.objects.filter(receiver=obj.id,sender=k) | Message.objects.filter(sender=obj.id,receiver=k)).order_by('-timestamp').values('message','timestamp')
        if len(data) == 0:
            return ""
        else:
            data = data.first()
            data["timestamp"] = str(data["timestamp"])
        return str(data)

我的看法:

   class UserChatViewSet(viewsets.ModelViewSet): 
       queryset = User.objects.all() 
       serializer_class = UserProfileSerializer

现在我的看法return:

[{
    "id": 4,
    "lastmessage": "{'message': 'random', 'timestamp': '2020-06-14 23:49:33.077749+00:00'}"
},
{
    "id": 5,
    "lastmessage": ""
},
{
    "id": 6,
    "lastmessage": "{'message': 'sample', 'timestamp': '2020-06-14 11:53:03.880833+00:00'}"
},
{
    "id": 7,
    "lastmessage": ""
}]

但我希望它根据最后一条消息的时间戳进行排序

您的回复顺序应在视图中处理。

  from django.db.models import Subquery, OuterRef
 lm = Message.objects.filter(sender=OuterRef("id"), receiver=self.kwargs['sid']).order_by('-timestamp')   
    data = User.objects.all().annotate(
        lastmessage=Subquery(
                    lm.values('timestamp')[:1]
                )
    ).order_by('-lastmessage__timestamp')

您可以覆盖 list 以实现此目的:

def list(self, request, *args, **kwargs):
    response = super().list(request, args, kwargs)
    # sort response.data['results']
    return response

此外,lastmessage 可以是 dict 而不是 str,因此更容易使用。