Spring Boot - JPA или SQL Запрос для поиска наиболее распространенного элемента в объединенных таблицах? - PullRequest
0 голосов
/ 09 марта 2020

Я использую Spring Boot, JPA & MySQL для создания микросервиса Spring Boot, который состоит из следующих отношений сущностей:

Owner can have multiple Cars.
Cars can only have one Owner.

pom. xml:

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>
    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.2.5.RELEASE</version>
        <relativePath/> <!-- lookup parent from repository -->
    </parent>
    <groupId>com.myapi</groupId>
    <artifactId>car-api</artifactId>
    <version>0.0.1-SNAPSHOT</version>
    <name>car-api</name>
    <description>Car REST API</description>

    <properties>
        <java.version>1.8</java.version>
    </properties>

    <dependencies>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-data-jdbc</artifactId>
        </dependency>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-data-jpa</artifactId>
        </dependency>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>

        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <scope>runtime</scope>
        </dependency>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
            <exclusions>
                <exclusion>
                    <groupId>org.junit.vintage</groupId>
                    <artifactId>junit-vintage-engine</artifactId>
                </exclusion>
            </exclusions>
        </dependency>
    </dependencies>

    <build>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
            </plugin>
        </plugins>
    </build>

</project>

Владелец:

@Entity
@Table(name = "owner")
public class Owner extends AuditModel {

    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private Long id;

    @NotNull
    private String name;


    private String address,
    private String city;
    private String state;
    private int zipCode;


    @OneToMany(cascade = CascadeType.ALL,
                fetch = FetchType.EAGER,
                mappedBy = "owner")
    private List<Car> cars = new ArrayList<>();

    public Owner() {
    }

    // Getter & Setters omitted for brevity.
}

Автомобиль:

@Entity
@Table(name="car")
public class Car extends AuditModel {

    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private Long id;

    String make;
    String model;
    String year;

    @JsonIgnore
    @ManyToOne(fetch = FetchType.LAZY)
    @JoinColumn(name = "owner_id", nullable = false)
    private Owner owner;

    // Getter & Setters omitted for brevity.
}

OwnerRepository:

@Repository
public interface OwnerRepository extends JpaRepository<Owner, Long> {
    @Query(value = "SELECT * FROM owner WHERE name = ?", nativeQuery = true)
    Owner findOwnerByName(String name);
}

CarRepository:

@Repository
public interface CarRepository extends JpaRepository<Car, Long> {
}

OwnerService:

public interface OwnerService {

    List<Owner> getAllOwners();

}

OwnerServiceImpl:

@Service
public class OwnerServiceImpl implements OwnerService {


    @Autowired
    OwnerRepository ownerRepository;

    @Override
    public List<Owner> getAllOwners() {
        return ownerRepository.findAll();
    }
}

OwnerController:

@RestController
public class OwnerController {


    private HttpHeaders headers = null;

    @Autowired
    OwnerService ownerService;

    public OwnerController() {
        headers = new HttpHeaders();
        headers.add("Content-Type", "application/json");
    }


    @RequestMapping(value = { "/owners" }, method = RequestMethod.GET, produces = "APPLICATION/JSON")
    public ResponseEntity<Object> getAllOwners() {
        List<Owner> owners = ownerService.getAllOwners();

        if (owners.isEmpty()) {
            return new ResponseEntity<Object>(HttpStatus.NOT_FOUND);
        }
        return new ResponseEntity<Object>(owners, headers, HttpStatus.OK);
    }

}

Итак, когда я вызываю эту конечную точку, используя:

GET http://localhost:8080/car-api/owners

Получает следующее JSON Полезная нагрузка:

[
    {
        "id": 1,
        "name": "Tom Brady",
        "cars": [
            {
                "id": 1,
                "make": "Honda",
                "model": "Accord",
                "year": "2020"
            },
            {
                "id": 11,
                 "make": "Nissan",
                 "model": "Maxima",
                 "year": "2019"
            },
            {

                "id": 12,
                "make": "Porsche",
                "model": "911",
                "year": "2017"
            }
        ]
    },
    {
        "id": 2,
        "name": "Kobe Bryant",
        "cars": [
            {
                "id": 2,
                "make": "Porsche",
                "model": "911",
                "year": "2017"
            }
        ]
    },
    {
        "id": 3,
        "name": "Mike Tyson",
        "cars": [
            {
                "id": 3,
                "make": "Volkswagen",
                "model": "Beatle",
                "year": "1973"
            }
        ]
    },
    {
        "id": 4,
        "name": "Scottie Pippen",
        "cars": [
            {
                "id": 4,
                "make": "Ford",
                "model": "F-150",
                "year": "2010"
            }
        ]
    },
    {
        "id": 5,
        "name": "John Madden",
        "cars": [
            {
                "id": 5,
                "make": "Chevrolet",
                "model": "Silverado",
                "year": "2020"
            }
        ]
    },
    {
        "id": 6,
        "name": "Arnold Palmer",
        "cars": [
            {
                "id": 6,
                 "make": "Toyota",
                "model": "Camary",
                "year": "2018"
            }
        ]
    },
    {
        "id": 7,
        "name": "Tiger Woods",
        "cars": [
            {
                "id": 7,
                 "make": "Alfa",
                 "model": "Romeo",
                 "year": "2017"
            }
        ]
    },
    {
        "id": 8,
        "name": "Magic Johnson",
        "cars": [
            {
                "id": 8,
                "make": "Porsche",
                "model": "911",
                "year": "2017"
            }
        ]
    },
    {
        "id": 9,
        "name": "George Foreman",
        "cars": [
            {
                "id": 9,
                "make": "Toyota",
                "model": "Camary",
                "year": "2018"
            }
        ]
    },
    {
        "id": 10,
        "name": "Charles Barkley",
        "cars": [
            {
                "id": 10,
                "make": "Alfa",
                "model": "Romeo",
                "year": "2017"
            }
        ]
    }
]

Как вы можете видеть, Том Брэйди, Коби Брайант, волхвы c У Джонсона все Porsche 911 ... Это самая распространенная машина в этом списке.


За кулисами JPA создает такие таблицы:

Автомобильный стол:

CREATE TABLE `car` (
  `id` bigint(20) NOT NULL AUTO_INCREMENT,
  `battery_capacity` int(11) NOT NULL,
  `make` varchar(255) DEFAULT NULL,
  `model` varchar(255) DEFAULT NULL,
  `year` varchar(255) DEFAULT NULL,
  `owner_id` bigint(20) NOT NULL,
  PRIMARY KEY (`id`),
  KEY `FK6wxv6hdekqn26n47pb7f1dt02` (`user_id`),
  CONSTRAINT `FK6wxv6hdekqn26n47pb7f1dt02` FOREIGN KEY (`owner_id`) REFERENCES `owner` (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=13 DEFAULT CHARSET=latin1;

Таблица владельца:

CREATE TABLE `owner` (
  `id` bigint(20) NOT NULL AUTO_INCREMENT,
  `address` varchar(255) DEFAULT NULL,
  `city` varchar(255) DEFAULT NULL,
  `name` varchar(255) NOT NULL,
  `state` varchar(255) DEFAULT NULL,
  `zip_code` int(11) NOT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=11 DEFAULT CHARSET=latin1;

SELECT * FROM OWNER:

+----+----------------+
| id | name           |
+----+----------------+
|  1 | Tom Brady      |
|  2 | Kobe Bryant    |
|  3 | Mike Tyson     |
|  4 | Scottie Pippen |
|  5 | John Madden    |
|  6 | Arnold Palmer  |
|  7 | Tiger Woods    |
|  8 | Magic Johnson  |
|  9 | George Foreman |
| 10 | Charles Barkley|
+----+----------------+

SELECT id, user_id, c.make, c.model, c.year FROM cars c;

+----------------+--------------+------+
| id | make      | model        | year |
+----+-----------+-----------+---------+
|  1 | Honda     | Accord       | 2020 |
|  2 | Nissan    | Leaf         | 2029 |
|  3 | Volkswagen| Beatle       | 1973 |
|  4 | Porsche   | Taycan       | 2020 |
|  5 | Ford      | F-150        | 2010 |
|  6 | Chevrolet | Silverado    | 2020 |
|  7 | Toyota    | Camry        | 2018 |
|  8 | Chevrolet | Bolt         | 2020 |
|  9 | Honda     | Clarity      | 2018 |
| 10 | Hyundai   | Ioniq        | 2017 |
| 11 | Nissan    | Maxima       | 2019 |
| 12 | Porsche   | 911          | 2020 |
+----+-----------+--------------+------+

Каким будет запрос JPA или SQL (может быть может быть именованным запросом), который я мог бы поместить в свой репозиторий, который будет возвращать данные с наиболее распространенным автомобилем (например, Select count (*) и, возможно, оператор Group By), который будет указывать, что наиболее принадлежащим автомобилю является Porsche 911, потому что 3 человека владеют их?

1 Ответ

1 голос
/ 09 марта 2020

Объявите класс для хранения car и его счетчика.

public class CarStatistics {

  private final Car car;
  private final Long count;

  public CarStatistics(Car car, Long count) {
    this.car = car;
    this.count = count;
  }

  public Car getCar() {
    return car;
  }

  public Long getCount() {
    return count;
  }
}

Возврат экземпляра компонента из группировки и упорядочения методов репозитория по количеству:

public interface CarRepository extends JpaRepository<Car, Long> {

  @Query("SELECT new com.example.CarStatistics(c, COUNT(c)) "
      + " FROM Car c "
      + " GROUP BY c.make, c.model "
      + " ORDER BY COUNT(c) DESC")
  List<CarStatistics> findCarCount();
}

Вызов репозитория метод и взять первый объект в наборе результатов:

@Service
public class CarService {

  private final CarRepository carRepository;

  public CarService(CarRepository carRepository) {
    this.carRepository = carRepository;
  }

  public Car findMostCommonCar() {
    List<CarStatistics> carStatistics = carRepository.findCarCount();
    return carStatistics.get(0).getCar();
  }
}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...