数据库中的字段和Java里面的类型匹配

当在hibernate中运用原始的sql语句进行操作的的时候,常常会报错,错误很多都是关于
No Dialect mapping for JDBC type:?这里的问号是不同的数字如:-1,1,-9,3等等,每种数字代表一种类型。经过分析和查找,问题常常发生在几个地方
1、错误的配置了hibernate的方言
2、就是你的数据库中的字段和Java里面的类型不匹配的问题 
错误如下:
org.hibernate.MappingException: No Dialect mapping for JDBC type: -1
Error message:
org.hibernate.MappingException: No Dialect mapping for JDBC type: -1
at org.hibernate.dialect.TypeNames.get(TypeNames.java:56)
at org.hibernate.dialect.TypeNames.get(TypeNames.java:81)
at org.hibernate.dialect.Dialect.getHibernateTypeName(Dialect.java:370)
at org.hibernate.loader.custom.CustomLoader$Metadata.getHibernateType(CustomLoader.java:559)
at org.hibernate.loader.custom.CustomLoader$ScalarResultColumnProcessor.performDiscovery(CustomLoader.java:485)
at org.hibernate.loader.custom.CustomLoader.autoDiscoverTypes(CustomLoader.java:501)
at org.hibernate.loader.Loader.getResultSet(Loader.java:1796)
at org.hibernate.loader.Loader.doQuery(Loader.java:674)
at org.hibernate.loader.Loader.doQueryAndInitializeNonLazyCollections(Loader.java:236)
at org.hibernate.loader.Loader.doList(Loader.java:2220)
at org.hibernate.loader.Loader.listIgnoreQueryCache(Loader.java:2104)
at org.hibernate.loader.Loader.list(Loader.java:2099)
at org.hibernate.loader.custom.CustomLoader.list(CustomLoader.java:289)
at org.hibernate.impl.SessionImpl.listCustomQuery(SessionImpl.java:1695)
at org.hibernate.impl.AbstractSessionImpl.list(AbstractSessionImpl.java:142)
at org.hibernate.impl.SQLQueryImpl.list(SQLQueryImpl.java:152)
at com.tilsi.solr.dao.summary.EntryDAO.queryByEntryId(EntryDAO.java:164)
at com.tilsi.solr.service.db.DbSearchThread.searchSummary(DbSearchThread.java:69)
at com.tilsi.solr.service.db.DbSearchThread.call(DbSearchThread.java:48)
at com.tilsi.solr.service.db.DbSearchThread.call(DbSearchThread.java:1)
at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask.java:303)
at java.util.concurrent.FutureTask.run(FutureTask.java:138)
at java.util.concurrent.ThreadPoolExecutor$Worker.runTask(ThreadPoolExecutor.java:886)
at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:908)
at java.lang.Thread.run(Thread.java:662)

解决方法为:通过查询hibernate的源码,找到问题的根本所在,就是hibernate中不支持对应的数据类型,具体数据类型的值,请查看下表
查看 java.sql.Types 类,可以找到对应类型,调整相应的数据库里的类型,或者其他支持这个类型的方言。 
public final static int BIT   =  -7;
public final static int TINYINT  =  -6;
public final static int SMALLINT =   5;
public final static int INTEGER  =   4;
public final static int BIGINT   =  -5;
public final static int FLOAT   =   6;
public final static int REAL   =   7;
public final static int DOUBLE   =   8;
public final static int NUMERIC  =   2;
public final static int DECIMAL  =   3;
public final static int CHAR  =   1;
public final static int VARCHAR  =  12;
public final static int LONGVARCHAR  =  -1;
public final static int DATE   =  91;
public final static int TIME   =  92;
public final static int TIMESTAMP  =  93;
public final static int BINARY  =  -2;
public final static int VARBINARY  =  -3;
public final static int LONGVARBINARY  =  -4;
public final static int NULL  =   0;
public final static int OTHER  = 1111;
public final static int JAVA_OBJECT         = 2000;
public final static int DISTINCT            = 2001;
public final static int STRUCT              = 2002;
public final static int ARRAY               = 2003;
public final static int BLOB                = 2004;
public final static int CLOB                = 2005;
public final static int REF                 = 2006;
public final static int DATALINK = 70;
public final static int BOOLEAN = 16;
public final static int ROWID = -8;
public static final int NCHAR = -15;
public static final int NVARCHAR = -9;
public static final int LONGNVARCHAR = -16;
public static final int NCLOB = 2011;
public static final int SQLXML = 2009; 

这类错误,最后的问号表示不确定的数字,但是解决的方法都一样。
首先,自定义一个方言类——Hibernate Dialect,该类需要继承与我们使用的数据库相应的方言类。比如:如果我们用的是mysql(版本为5.x.x),我们需要继承“org.hibernate.dialect.MySQL5Dialect”;如果我们使用的是DB2,那么我们应该继承“org.hibernate.dialect.DB2Dialect”;我用的是SqlServer2008,所以我要继承“org.hibernate.dialect.SQLServerDialect”,参考代码如下:

    import java.sql.Types;   
      
    import org.hibernate.Hibernate;   
    import org.hibernate.dialect.SQLServerDialect;   
      
    public class SqlServer2008Dialect extends SQLServerDialect {   
      
        public SqlServer2008Dialect() {   
            super();   
            registerHibernateType(Types.CHAR, Hibernate.STRING.getName());   
            registerHibernateType(Types.NVARCHAR, Hibernate.STRING.getName());   
            registerHibernateType(Types.LONGNVARCHAR, Hibernate.STRING.getName());   
            registerHibernateType(Types.DECIMAL, Hibernate.DOUBLE.getName());   
        }   
    }  
总之大家可以在“org.hibernate.dialect”这个package中找到与数据库相对应的方言类。在其中,我们需要注意三点:

a、在默认构造方法中继承父类构造方法,同时调用“registerHibernateType(int code, String name)”方法将数据库中该数据类型映射到相应的java类型。code表示数据库中的数据类型整数表示,可以在“java.sql.Types”类中查到相应的数据库类型。name表示我们要映射的java类型。可以从“org.hibernate.Hibernate”中查到。

b、Types类。在Types中定义了数据库常用的字段类型,如:

    ……   
    public final static int LONGVARCHAR =  -1;   
    public final static int TIMESTAMP =  93;   
    ……  
也就是上面我列出的java.sql.Types。
我们可以根据“No Dialect mapping for JDBC type : ”后面紧跟的数字在该类(Types)中找到相应的类型。我们也可以根据数据表中字段的类型找到相应的值。这个值就是registerHibernateType(int code, String name)的第一个参数。

c、Hibernate类。Hibernate中定义了转换的目的类型,如第一段代码所示。能够转化成什么类型,可以在该类中查找。通过调用“getName()”方法得到一个String型。当然,如果你记住了,我们还可以这样写

    import org.hibernate.dialect.SQLServerDialect;   
      
    public class SqlServer2008Dialect extends SQLServerDialect {   
      
        public SqlServer2008Dialect() {   
            super();   
            registerHibernateType(1, "string");   
            registerHibernateType(-9, "string");   
            registerHibernateType(-16, "string");   
            registerHibernateType(3, "double");   
        }   
    }  
其实和上面是一样的,只是把上面所代表的值直接写出来了。需要注意的是super()方法的调用,不调用该方法是否会出现错误,这个我就不知道了,我没有测试,所以最好调一下。
然后,我们还需要在配置文件中作修改,我将连接数据库的方法放在了spring配置文件中了,这段代码就写在applicationContext.xml里面了:
<property name="hibernateProperties">
<props>
<!-- <prop key="hibernate.dialect">org.hibernate.dialect.SQLServerDialect </prop> -->
<prop key="hibernate.dialect">com.sense.workflow.util.SqlServer2008Dialect</prop>
<prop key="hibernate.hbm2ddl.auto">update</prop>
                <prop key="hibernate.show_sql">true</prop>
                <prop key="hibernate.cache.use_query_cache">false</prop> 
                <!-- 显示sql格式 -->
                <prop key="hibernate.format_sql">false</prop> 
</props>
</property>

最后还有一个解决方法:如果你的数据库表中某个字段不可以不用text类型的,将其改成其他类型(如:varchar等)也可以,这个方法虽然简单,当数据库字段必须用text类型的时候该方法就不行了。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: MySQL字段类型长度和Java类型有以下对应关系: 1. 字符串类型: - CHAR(n):Java类型为String,长度为n。 - VARCHAR(n):Java类型为String,长度为n。 - TEXT:Java类型为String。 2. 数值类型: - INT:Java类型为int。 - BIGINT:Java类型为long。 - DECIMAL(p, s):Java类型为BigDecimal,其p表示总位数,s表示小数位数。 3. 日期和时间类型: - DATE:Java类型java.sql.Date。 - TIME:Java类型java.sql.Time。 - DATETIME:Java类型java.sql.Timestamp。 4. 布尔类型: - BOOL、BOOLEAN:Java类型为boolean。 5. 二进制类型: - BLOB:Java类型为byte[]。 需要注意的是,Java的数据类型和MySQL的字段类型并不完全一致,因此在进行数据类型转换时需要注意类型匹配。在使用JDBC连接MySQL进行数据操作时,可以根据需要使用ResultSet的get方法和PreparedStatement的set方法来进行数据类型的转换。同时,在设计数据库表结构时也需要根据实际需求选择合适的MySQL字段类型,以确保数据的存储和查询效率,并防止数据溢出或损失。 ### 回答2: MySQL的字段类型长度和Java类型对应如下: 1. 字符串类型(VARCHAR、CHAR): - VARCHAR(n) 对应 Java 的 String 类型,表示一个可变长度的字符串,最大长度为 n,例如:VARCHAR(50) 对应 Java 的 String。 - CHAR(n) 对应 Java 的 String 类型,表示一个固定长度的字符串,长度为 n,例如:CHAR(20) 对应 Java 的 String。 2. 数值类型(INT、BIGINT、FLOAT、DOUBLE、DECIMAL): - INT 对应 Java 的 int 类型。 - BIGINT 对应 Java 的 long 类型。 - FLOAT 对应 Java 的 float 类型。 - DOUBLE 对应 Java 的 double 类型。 - DECIMAL(precision, scale) 对应 Java 的 BigDecimal 类型,其 precision 表示总长度,scale 表示小数点后的位数,例如:DECIMAL(10, 2) 对应 Java 的 BigDecimal。 3. 日期和时间类型(DATE、TIME、DATETIME、TIMESTAMP): - DATE 对应 Java 的 LocalDate 类型。 - TIME 对应 Java 的 LocalTime 类型。 - DATETIME 和 TIMESTAMP 都对应 Java 的 LocalDateTime 类型。 4. 布尔类型(BOOLEAN、BIT): - BOOLEAN 对应 Java 的 boolean 类型。 - BIT(n) 对应 Java 的 byte[] 类型。 5. 其他类型: - BLOB 对应 Java 的 byte[] 类型,用于存储二进制数据。 - TEXT 对应 Java 的 String 类型,用于存储大文本数据。 需要注意的是,MySQL的字段类型长度和Java类型对应是一种常见的映射方式,但不是唯一的方式。在不同的数据库和编程语言,可能会有一些差异或变化。此外,还可以使用数据库连接工具或框架进行字段类型Java类型的自动映射。 ### 回答3: MySQL字段类型的长度和Java类型对应关系如下: 1. 数值类型: - TINYINT:Java类型为byte,长度为1字节。 - SMALLINT:Java类型为short,长度为2字节。 - MEDIUMINT:Java类型为int,长度为3字节。 - INT:Java类型为int,长度为4字节。 - BIGINT:Java类型为long,长度为8字节。 - FLOAT:Java类型为float,长度为4字节。 - DOUBLE:Java类型为double,长度为8字节。 - DECIMAL:Java类型为BigDecimal,长度可自定义。 2. 字符串类型: - CHAR:Java类型为String,长度可自定义,最大为255字节。 - VARCHAR:Java类型为String,长度可自定义,最大为65535字节。 - TEXT:Java类型为String,长度可自定义,最大为65535字节。 - BLOB:Java类型为byte[],长度可自定义,最大为65535字节。 3. 日期和时间类型: - DATE:Java类型java.sql.Date,长度为3字节。 - TIME:Java类型java.sql.Time,长度为3字节。 - DATETIME:Java类型java.sql.Timestamp,长度为8字节。 - TIMESTAMP:Java类型java.sql.Timestamp,长度为4字节。 - YEAR:Java类型java.util.Date,长度为1字节。 4. 其他类型: - BOOLEAN:Java类型为boolean,长度为1字节。 - ENUM:Java类型为String,长度可自定义。 需要注意的是,MySQL的字段类型长度可能会受到存储引擎和字符编码的影响,所以实际长度可能有所不同。对于字符串类型,长度指的是字符数,而不是字节数。另外,为了更准确地处理日期和时间类型,建议使用对应的Java日期时间类。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值