2017-09-04 5 views
2

내가 장고를 배우고, 나는 약간의 문제에 직면하고있어 : I 행을 관련 한 모델이Groupping와 장고 모델에 집계 관련 기록

을, 나는 합계를 좀하고 싶습니다 관련 행의 열을 두 번째 열의 값으로 그룹화합니다.

내가 함께 일하고 있어요 모델의는 다음과 같다 :

class Integrante(models.Model): 
    nombre = models.CharField(max_length=80, db_index=True) 

class EstadoCuenta(models.Model): 
    num_edc = models.PositiveIntegerField(validators=[MinValueValidator(1),]) 
    fecha_edc = models.DateField(null=True, db_index=True) 

class Movimiento(models.Model): 
    TIPOS_MOVIMIENTO = (
     # Choices for column 'tipo' 
    ) 
    estado_cuenta = models.ForeignKey(EstadoCuenta, on_delete=models.CASCADE) 
    integrante = models.ForeignKey(Integrante, on_delete=models.CASCADE) 
    fecha = models.DateField(db_index=True) 
    tipo = models.SmallIntegerField(db_index=True, choices=TIPOS_MOVIMIENTO) 
    descripcion = models.CharField(max_length=200, blank=True) 
    importe = models.DecimalField(max_digits=8, decimal_places=2) 

    class Meta: 
     ordering = ['integrante', 'fecha', 'tipo', 'pk'] 

나는 다음 시도했다 :

for x in edc.movimiento_set.values('integrante').annotate(Sum('importe')): 
    print(x) 

# {'integrante': 28, 'importe__sum': Decimal('-20.00')} 
# {'integrante': 28, 'importe__sum': Decimal('23.00')} 
# {'integrante': 28, 'importe__sum': Decimal('9.60')} 
# {'integrante': 28, 'importe__sum': Decimal('20.00')} 
# {'integrante': 28, 'importe__sum': Decimal('-0.60')} 
# {'integrante': 24, 'importe__sum': Decimal('96.00')} 
# {'integrante': 24, 'importe__sum': Decimal('28.80')} 
# {'integrante': 24, 'importe__sum': Decimal('48.00')} 
# {'integrante': 24, 'importe__sum': Decimal('28.80')} 
# {'integrante': 24, 'importe__sum': Decimal('96.00')} 
# {'integrante': 24, 'importe__sum': Decimal('48.00')} 
# {'integrante': 24, 'importe__sum': Decimal('288.00')} 
# {'integrante': 24, 'importe__sum': Decimal('144.00')} 
# {'integrante': 24, 'importe__sum': Decimal('19.20')} 
# {'integrante': 24, 'importe__sum': Decimal('-510.00')} 
# {'integrante': 24, 'importe__sum': Decimal('48.00')} 
# {'integrante': 24, 'importe__sum': Decimal('48.00')} 
# {'integrante': 24, 'importe__sum': Decimal('48.00')} 
# {'integrante': 24, 'importe__sum': Decimal('48.00')} 
# {'integrante': 24, 'importe__sum': Decimal('48.00')} 
# {'integrante': 24, 'importe__sum': Decimal('48.00')} 
# {'integrante': 24, 'importe__sum': Decimal('35.00')} 
# {'integrante': 24, 'importe__sum': Decimal('235.00')} 
# {'integrante': 24, 'importe__sum': Decimal('-0.80')} 
# ... 
그러나

edc = EstadoCuenta.objects.filter(grupo__nombre_grupo='A group').latest() 
edc.movimiento_set.values('integrante').annotate(Sum('importe')) 

, 나는 다음과 같은 결과를 얻고있다

내가 얻으려고하는 결과는 다음과 같습니다 (데이터베이스 서버에서 직접 수행됨) :

select integrante_id, sum(importe) 
from core_movimiento 
where estado_cuenta_id=233 
group by integrante_id; 

-- | integrante_id | sum(importe) | 
-- +---------------+--------------+ 
-- |   24 |  844.00 | 
-- |   25 |   0.00 | 
-- |   26 |   0.00 | 
-- |   27 |  -232.00 | 
-- |   28 |  32.00 | 
-- |   29 |   0.00 | 
-- |   30 |  20.00 | 

올바른 방향으로 나를 가리킬 수 있습니까?

I 파이썬 3.6 장고 사용하고 1.11

보너스

I는 열 tipo의 값에 따라, 컬럼 분리 importe의 합을 얻고 자하는 것이다. SQL에서는 이렇게 할 수 있습니다.

select integrante_id 
    , sum(case when tipo=1 then importe else 0 end) as pagos 
    , sum(case when tipo in (11, 12) then importe else 0 end) as ventas 
    , sum(case when tipo in (70, 71) then importe else 0 end) as paquetes 
    , sum(case when tipo=99 then importe else 0 end) as ajustes 
    , sum(importe) as total 
from core_movimiento 
where estado_cuenta_id = 233 
group by integrante_id; 

-- | integrante_id | pagos | ventas | paquetes | ajustes | total | 
-- +---------------+---------+---------+----------+---------+---------+ 
-- |   24 | -510.00 | 1084.80 | 270.00 | -0.80 | 844.00 | 
-- |   25 | -35.00 | 0.00 | 35.00 | 0.00 | 0.00 | 
-- |   26 | -20.00 | 0.00 | 20.00 | 0.00 | 0.00 | 
-- |   27 | -422.00 | 190.00 |  0.00 | 0.00 | -232.00 | 
-- |   28 | -20.00 | 32.60 | 20.00 | -0.60 | 32.00 | 
-- |   29 | -200.00 | 0.00 | 200.00 | 0.00 | 0.00 | 
-- |   30 | 0.00 | 0.00 | 20.00 | 0.00 | 20.00 | 

아이디어가 있습니까?

+0

''meta' 모델에'주문 중 '이 있습니까? –

+1

이것은 큰 질문입니다. 특히 보너스 부분! 좋은 대답은 django ORM을 사용하여 데이터베이스를 왕복하지 않고이 작업을 수행하는 것입니다. 당신은 장고 ORM 납치 고려해 볼까요 resonable 대답을? django를 통해 원시 SQL 쿼리를 작성하는 것과 같습니다. https://docs.djangoproject.com/en/1.11/topics/db/sql/ – Tico

+0

@BearBrown 'Movimiento'에 'Meta'클래스가 추가되었습니다. 주문에''integrante''만을 남겨 두어야합니까? – Barranka

답변

1

업데이트됩니다 두 번째 질문은 django 1.8 이후에 새로운 답변을 추가합니다. conditional-expressions

from django.db.models import Sum, When, Case, IntegerField, F 

edc.movimiento_set.values('integrante' 
    ).annotate(total=Sum('importe') 
    ).annotate(pagos=Sum(
     Case(
      When(tipo=1, then=F('importe')), 
      default=0, 
      output_field=IntegerField() 
      ) 
     ) 
    ).annotate(ventas=Sum(
     Case(
      When(tipo__in=(11, 12), then=F('importe')), 
      default=0, 
      output_field=IntegerField() 
      ) 
     ) 
    ).annotate(paquetes=Sum(
     Case(
      When(tipo__in=(70, 71), then=F('importe')), 
      default=0, 
      output_field=IntegerField() 
      ) 
     ) 
    ).annotate(ajustes=Sum(
     Case(
      When(tipo=99, then=F('importe')), 
      default=0, 
      output_field=IntegerField() 
      ) 
     ) 
    ).order_by() 

괜찮습니다.

+0

인상적! 나는 두 답을 모두 받아들이고 싶다! 고맙습니다! – Barranka

2

단지로 빈 추가 주문 먼저하셔야합니다 : 주문 필드 내가 생각하는 보너스 ordering-or-order-by

에 대한 기본 장고에 의해

edc.movimiento_set.values('integrante').annotate(Sum('importe')).order_by() 
#                ^^^^^^^^^^ 

에 의해 그룹을 추가, 대한 대답 나중에