Как hibernate генерирует имена ограничений внешнего ключа?

Как hibernate генерирует имена ограничений внешнего ключа?

Если я не определяю имя, спящий режим генерирует что-то вроде этого

CONSTRAINT fk_2ocepcfwpr1v18dg1ieoe6bau

как рождается это имя? Может быть, из MD5 хэша имен полей или что-то в этом роде? Мне нужно знать, одинаково ли имя во всех экземплярах.


person wutzebaer    schedule 21.04.2016    source источник


Ответы (1)


Hibernate генерирует имя ограничения путем объединения имен таблиц и свойств и преобразования результата в MD5. Это необходимо из-за ограничения длины имен ограничений в некоторых базах данных. Например, в базе данных Oracle длина имени внешнего ключа не может превышать 30 символов.

Этот фрагмент кода из исходного кода Hibernate org.hibernate.mapping.Constraint

/**
 * If a constraint is not explicitly named, this is called to generate
 * a unique hash using the table and column names.
 * Static so the name can be generated prior to creating the Constraint.
 * They're cached, keyed by name, in multiple locations.
 *
 * @return String The generated name
 */
public static String generateName(String prefix, Table table, Column... columns) {
    // Use a concatenation that guarantees uniqueness, even if identical names
    // exist between all table and column identifiers.

    StringBuilder sb = new StringBuilder( "table`" + table.getName() + "`" );

    // Ensure a consistent ordering of columns, regardless of the order
    // they were bound.
    // Clone the list, as sometimes a set of order-dependent Column
    // bindings are given.
    Column[] alphabeticalColumns = columns.clone();
    Arrays.sort( alphabeticalColumns, ColumnComparator.INSTANCE );
    for ( Column column : alphabeticalColumns ) {
        String columnName = column == null ? "" : column.getName();
        sb.append( "column`" ).append( columnName ).append( "`" );
    }
    return prefix + hashedName( sb.toString() );
}

/**
 * Hash a constraint name using MD5. Convert the MD5 digest to base 35
 * (full alphanumeric), guaranteeing
 * that the length of the name will always be smaller than the 30
 * character identifier restriction enforced by a few dialects.
 * 
 * @param s
 *            The name to be hashed.
 * @return String The hased name.
 */
public static String hashedName(String s) {
    try {
        MessageDigest md = MessageDigest.getInstance( "MD5" );
        md.reset();
        md.update( s.getBytes() );
        byte[] digest = md.digest();
        BigInteger bigInt = new BigInteger( 1, digest );
        // By converting to base 35 (full alphanumeric), we guarantee
        // that the length of the name will always be smaller than the 30
        // character identifier restriction enforced by a few dialects.
        return bigInt.toString( 35 );
    }
    catch ( NoSuchAlgorithmException e ) {
        throw new HibernateException( "Unable to generate a hashed Constraint name!", e );
    }
}

Вы можете сгенерировать свои собственные имена ограничений (уникальный и внешний ключ), используя ImplicitNamingStrategy. Вы можете сослаться на Hibernate5NamingStrategy, например.

person v.ladynev    schedule 21.04.2016