【问题标题】:Annotation for UDT by Spring data Cassandra [duplicate]Spring数据Cassandra对UDT的注释[重复]
【发布时间】:2015-01-04 23:30:57
【问题描述】:

springdatacassandra 如何支持 POJO 中 Cassandra 提供的用户自定义数据类型?我正在寻找 UDT 的注释。

【问题讨论】:

标签: user-defined-types spring-data-cassandra


【解决方案1】:

Spring data cassandra 最新版本 (1.2.0.BUILD-SNAPSHOT) 依赖于 datastax 驱动程序 2.0.4 ,其中从 datastax 驱动程序 2.1.x 开始支持 UDT。

您可以尝试将 datastax 驱动程序覆盖到 2.1.x 以使用 2.1 驱动程序提供的最新功能。

【讨论】:

  • 感谢@shatk 的回复。这就是我要找的。​​span>
  • 只是为了更清楚:@UDT 和其他在 DataStax 文档中看到的映射注释 (com.datastax.driver.mapping.annotations.*) 到目前为止不受 spring-data-cassandra 支持。这些注释由工件cassandra-driver-mapping 提供。使用 DataStax 的注解注解的类需要通过com.datastax.driver.mapping.Mapper 进行持久化,请参阅datastax.com/documentation/developer/java-driver/2.1/…
【解决方案2】:
    <dependency>
        <groupId>com.datastax.cassandra</groupId>
        <artifactId>cassandra-driver-mapping</artifactId>
        <version>2.1.9</version>
    </dependency>

是必需的,但总的来说它不起作用,spring data cassandra 不支持 UDT 映射。

在此处查看详细信息: https://jira.spring.io/browse/DATACASS-172

我遇到了这个问题,调试过程向我展示了 spring data cassandra 检查 @Table、@Persistent 或 @PrimaryKeyClass 仅注解并在另一种情况下引发异常

> 调用 init 方法失败;嵌套异常是 org.springframework.data.cassandra.mapping.VerifierMappingExceptions: Cassandra 实体必须具有@Table、@Persistent 或@PrimaryKeyClass 注解

但我找到了解决方案。 我想出了一种方法,可以让我管理包含 UDT 和不包含 UDT 的实体。在我的应用程序中,我使用 spring cassandra 数据项目以及直接 datastax 核心驱动程序。不包含 UDT 对象的存储库使用 spring cassanta 数据方法,包含 UDT 的对象使用自定义存储库。 自定义存储库使用 datastax 映射器,它们与 UDT 一起正常工作 (它们位于单独的包中,请参阅下面的说明为什么需要它):

package com.fyb.cassandra.custom.repositories.impl;

import java.util.List;
import java.util.UUID;

import javax.annotation.PostConstruct;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.cassandra.config.CassandraSessionFactoryBean;

import com.datastax.driver.core.ResultSet;
import com.datastax.driver.mapping.Mapper;
import com.datastax.driver.mapping.MappingManager;
import com.datastax.driver.mapping.Result;
import com.google.common.collect.Lists;
import com.fyb.cassandra.custom.repositories.AccountDeviceRepository;
import com.fyb.cassandra.dto.AccountDevice;

public class AccountDeviceRepositoryImpl implements AccountDeviceRepository {

    @Autowired
    public CassandraSessionFactoryBean session;

    private Mapper<AccountDevice> mapper;

    @PostConstruct
    void initialize() {
        mapper = new MappingManager(session.getObject()).mapper(AccountDevice.class);
    }

    @Override
    public List<AccountDevice> findAll() {
        return fetchByQuery("SELECT * FROM account_devices");
    }

    @Override
    public void save(AccountDevice accountDevice) {
        mapper.save(accountDevice);
    }

    @Override
    public void deleteByConditions(UUID accountId, UUID systemId, UUID deviceId) {
        final String query = "DELETE FROM account_devices where account_id =" + accountId + " AND system_id=" + systemId
                + " AND device_id=" + deviceId;
        session.getObject().execute(query);
    }

    @Override
    public List<AccountDevice> findByAccountId(UUID accountId) {
        final String query = "SELECT * FROM account_devices where account_id=" + accountId;
        return fetchByQuery(query);
    }

    /*
     * Take any valid CQL query and try to map result set to the given list of appropriates <T> types.
     */
    private List<AccountDevice> fetchByQuery(String query) {
        ResultSet results = session.getObject().execute(query);
        Result<AccountDevice> accountsDevices = mapper.map(results);
        List<AccountDevice> result = Lists.newArrayList();
        for (AccountDevice accountsDevice : accountsDevices) {
            result.add(accountsDevice);
        }
        return result;
    }
}

用于管理不包含 UDT 对象的实体的 Spring Data 相关存储库如下所示:

package com.fyb.cassandra.repositories;

import org.springframework.data.cassandra.repository.CassandraRepository;

import com.fyb.cassandra.dto.AccountUser;
import org.springframework.data.cassandra.repository.Query;
import org.springframework.stereotype.Repository;

import java.util.List;
import java.util.UUID;

@Repository
public interface AccountUserRepository extends CassandraRepository<AccountUser> {

    @Query("SELECT * FROM account_users WHERE account_id=?0")
    List<AccountUser> findByAccountId(UUID accountId);
}

我已经测试了这个解决方案,它 100% 有效。 此外,我还附加了我的 POJO 对象:

仅使用数据统计注释的 Pojo:

package com.fyb.cassandra.dto;
import java.util.List;
import java.util.Map;
import java.util.UUID;

import com.datastax.driver.mapping.annotations.ClusteringColumn;
import com.datastax.driver.mapping.annotations.Column;
import com.datastax.driver.mapping.annotations.Frozen;
import com.datastax.driver.mapping.annotations.FrozenValue;
import com.datastax.driver.mapping.annotations.PartitionKey;
import com.datastax.driver.mapping.annotations.Table;

@Table(name = "account_systems")
public class AccountSystem {

    @PartitionKey
    @Column(name = "account_id")
    private java.util.UUID accountId;

    @ClusteringColumn
    @Column(name = "system_id")
    private java.util.UUID systemId;

    @Frozen
    private Location location;

    @FrozenValue
    @Column(name = "user_token")
    private List<UserToken> userToken;

    @Column(name = "product_type_id")
    private int productTypeId;

    @Column(name = "serial_number")
    private String serialNumber;   
}

Pojo 不使用 UDT,只使用 spring data cassandra 框架:

package com.fyb.cassandra.dto;

import java.util.Date;
import java.util.UUID;

import org.springframework.cassandra.core.PrimaryKeyType;
import org.springframework.data.cassandra.mapping.Column;
import org.springframework.data.cassandra.mapping.PrimaryKeyColumn;
import org.springframework.data.cassandra.mapping.Table;

@Table(value = "accounts")
public class Account {

    @PrimaryKeyColumn(name = "account_id", ordinal = 0, type = PrimaryKeyType.PARTITIONED)
    private java.util.UUID accountId;

    @Column(value = "account_name")
    private String accountName;

    @Column(value = "currency")
    private String currency;    
}

注意,以下实体使用不同的注解:

@PrimaryKeyColumn(name = "account_id", ordinal = 0, type = PrimaryKeyType.PARTITIONED)and @PartitionKey
@ClusteringColumn and @PrimaryKeyColumn(name = "area_parent_id", ordinal = 2, type = PrimaryKeyType.CLUSTERED)

乍一看 - 这很不舒服,但它允许您使用包含 UDT 和不包含 UDT 的对象。

一个重要的注意事项。这两个 repos(使用 UDT 并且不应该驻留在不同的包中)导致 Spring config 寻找带有 repos 的基本包:

@Configuration
@EnableCassandraRepositories(basePackages = {
        "com.fyb.cassandra.repositories" })
public class CassandraConfig {
..........
}

【讨论】:

    猜你喜欢
    • 2016-04-17
    • 2015-08-03
    • 2018-08-23
    • 2017-11-30
    • 1970-01-01
    • 2018-01-10
    • 1970-01-01
    • 2014-10-10
    • 1970-01-01
    相关资源
    最近更新 更多