spring+struts+hibernate常见异常总结

 

Exception : No form found under 'loginForm' in locale 'en_US'

这个错误完全是我自己给自己找麻烦,我在给loginForm做表单验证时,用了validator,我在写validation.xml文件时,突然发神经,想写得比较国际化一点,这样写的:

   <formset language="zh" country="CN"> <!--这个是zh_CN的-->

   开始是一点问题都没有,因为我们大家一般都用中文系统,后来有一天我用linux,当时环境是english,结果发现页面就报错:No form found under 'loginForm' in locale 'en_US'。

当时,特别郁闷,后来仔细回忆,想起当时只在validation.xml文件里显式的写过国际化相关的东西,后来就把 language="zh" country="CN" 给删掉了。即:
<formset>。。。。。。</formset>

再用英文系统访问就没有事情了,看来你要是想国际化,就把所有的情况的配置都写全了,不要只指定中文的配置,英文的不配置,那样,别人用英文系统的人就会出错。所以,建议还是不要制定国际化,看来有的时候还是偷懒一点比较好!

 

hibernate,one to many关系时出现:Data truncation: Column was set to data type implicit default; NULL supplied for NOT NULL column

Hibernate: select max(orders_id) from orders
Hibernate: insert into orders (member_id, orderno, realname, address, zip, tel, payment, email, memo, price, time, tag, orders_id) values (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
Hibernate: select member0_.member_id as member1_0_, member0_.username as username0_, member0_.password as password0_, member0_.realname as realname0_, member0_.tel as tel0_, member0_.address as address0_, member0_.zip as zip0_, member0_.email as email0_ from member member0_ where member0_.username=?
Hibernate: select orders0_.member_id as member2_1_, orders0_.orders_id as orders1_1_, orders0_.orders_id as orders1_5_0_, orders0_.member_id as member2_5_0_, orders0_.orderno as orderno5_0_, orders0_.realname as realname5_0_, orders0_.address as address5_0_, orders0_.zip as zip5_0_, orders0_.tel as tel5_0_, orders0_.payment as payment5_0_, orders0_.email as email5_0_, orders0_.memo as memo5_0_, orders0_.price as price5_0_, orders0_.time as time5_0_, orders0_.tag as tag5_0_ from orders orders0_ where orders0_.member_id=?
1
Hibernate: update orders set member_id=null where member_id=?
SQL Error: 0, SQLState: 01004
Data truncation: Column was set to data type implicit default; NULL supplied for NOT NULL column 'member_id' at row 1
Could not synchronize database state with session
org.hibernate.exception.GenericJDBCException: Could not execute JDBC batch update
at org.hibernate.exception.SQLStateConverter.handledNonSpecificException(SQLStateConverter.java:103)
at org.hibernate.exception.SQLStateConverter.convert(SQLStateConverter.java:91)
at org.hibernate.exception.JDBCExceptionHelper.convert(JDBCExceptionHelper.java:43)
at org.hibernate.jdbc.AbstractBatcher.executeBatch(AbstractBatcher.java:202)
at org.hibernate.engine.ActionQueue.executeActions(ActionQueue.java:235)
at org.hibernate.engine.ActionQueue.executeActions(ActionQueue.java:141)
at org.hibernate.event.def.AbstractFlushingEventListener.performExecutions(AbstractFlushingEventListener.java:297)
at org.hibernate.event.def.DefaultFlushEventListener.onFlush(DefaultFlushEventListener.java:27)
at org.hibernate.impl.SessionImpl.flush(SessionImpl.java:985)
at org.hibernate.impl.SessionImpl.managedFlush(SessionImpl.java:333)
at org.hibernate.transaction.JDBCTransaction.commit(JDBCTransaction.java:106)
at net.yywn.hibernate.database.DbOperate.delete(DbOperate.java:602)
at net.yywn.hibernate.database.DbOperateTest.testOneToMany(DbOperateTest.java:265)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(Unknown Source)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(Unknown Source)
at java.lang.reflect.Method.invoke(Unknown Source)
at junit.framework.TestCase.runTest(TestCase.java:154)
at junit.framework.TestCase.runBare(TestCase.java:127)
at junit.framework.TestResult$1.protect(TestResult.java:106)
at junit.framework.TestResult.runProtected(TestResult.java:124)
at junit.framework.TestResult.run(TestResult.java:109)
at junit.framework.TestCase.run(TestCase.java:118)
at org.eclipse.jdt.internal.junit.runner.junit3.JUnit3TestReference.run(JUnit3TestReference.java:128)
at org.eclipse.jdt.internal.junit.runner.TestExecution.run(TestExecution.java:38)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:460)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:673)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.run(RemoteTestRunner.java:386)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.main(RemoteTestRunner.java:196)
Caused by: java.sql.BatchUpdateException: Data truncation: Column was set to data type implicit default; NULL supplied for NOT NULL column 'member_id' at row 1
at com.mysql.jdbc.ServerPreparedStatement.executeBatch(ServerPreparedStatement.java:657)
at org.hibernate.jdbc.BatchingBatcher.doExecuteBatch(BatchingBatcher.java:58)
at org.hibernate.jdbc.AbstractBatcher.executeBatch(AbstractBatcher.java:195)
... 25 more

问题分析:

1(这是该用户的订单数)
Hibernate: update orders set member_id=null where member_id=?(开始执行delete该用户)

这是上面代码片段的分析:可知,该delete的操作没事执行,而是先去执行update。由于member_id为null,而

数据库中相应的字段设置不为null。

问题解决:

当然不能去该数据库的那个字段(设置为null,那就是缘木求鱼了)

细看配置文件,发现member没配inverse(初始值为false)属性。

把它设为true即可!

个人认为inverse属性是用来设置主动权的。在member的set(orders)中设置其为true,表示把主动权交给了

member,这样删除member的同时就把member的orders也删除了。(其实是先删orders再member)

成功后部分sql如下:

Hibernate: delete from orders where orders_id=?
Hibernate: delete from orders where orders_id=?
Hibernate: delete from member where member_id=?

..................................................................................

Hibernate错误:org.hibernate.StaleStateException: Batch update returned unexpected row count from update: 0 actual row count: 0 expected: 1

原因多半是你的mapping文件的主键没有设置对。检查如下:

一:看你的主键类型是什么?如果是long,int等,记得设置unsaved-value="0"

二:如果是string 的话,请设置unsaved-value="null"

注意unsaved-value是个很重要的属性。Hibernate通过这个属性来判断一个对象应该save还是update,如果这个对象的id是unsaved-value的话,那说明这个对象不是 persistence object要save(insert);如果id是非unsaved-value的话,那说明这个对象是persistence object(数据库中已存在),只要update就行了。 你可以Debug跟一下你的代码,看看在 session.update(obj); 的时候,obj的主键的值是不是按你定义的那样,obj可以被hibernate认为是一个已经持久化在数据库里的对象

===========================================================

java.lang.NoClassDefFoundError: antlr/ANTLRException

原因是:缺少antlr-2.7.5H3.jar

---------------------------------------2----------------------------------------------

类 java.lang.NoSuchMethodError

java.lang.Object

   |

   +----java.lang.Throwable

           |

           +----java.lang.Error

                   |

                   +----java.lang.LinkageError

                           |

                           +----java.lang.IncompatibleClassChangeError

                                   |

                                   +----java.lang.NoSuchMethodError

public class NoSuchMethodError

extends IncompatibleClassChangeError

如果应用程序试图调用一个类的指定方法(无论是静态还是动态),但是该类不再有该方法的定义,则抛出该异常。

通常由编译器检测此错误;如果类定义不允许再作兼容性的改变,那么此错误只可能发生在运行时刻。

构造子索引

NoSuchMethodError()

NoSuchMethodError(String)

    用指定的详细信息构造 NoSuchMethodException。

构造子

NoSuchMethodError

public NoSuchMethodError()

NoSuchMethodError

public NoSuchMethodError(String s)

    用指定的详细信息构造 NoSuchMethodException。

    参数:

        s - 详细信息。

----------------------------------3--------------------------------------------------

今天用hibernate3做关系映射,当调用映射类的时候总是报

org.hibernate.LazyInitializationException: could not initialize proxy错误

原因:

hb3对many-to-one的默认处理是lazy = "proxy" 而hb2是false

所以,要这么设置

        <many-to-one name="shop_type" column="sort_id" class="liubi.ShopType"

cascade="all" outer-join="true" lazy="false" />

------------------------------------4--------------------------------------------------

java.sql.Timestamp异常

     原因:时间date错误

     更改时间java.util.Date 为java.lang.String就好了,同时修改配置文件中的内容

------------------------------------5--------------------------------------------------

Cannot find bean in any scope

分类:Struts异常

1.出现这种错误一般是资源文件没找到或是资源文件里没有对应的key值。

2.struts-config.xml里的Action配置里把redirect设置为了true;改成false,

3.在Action里一般会request.setAttribute()一些对象,然后在转向的jsp文件里

(用tag或 request.getAttribute()方法)得到这些对象并显示出来。

这个异常是说jsp要得到一个对象,

但前面的Action里并没有将对象设置到request(也可以是session、servletContext)里。

可能是名字错了,请检查jsp里的tag的一般是name属性,或getAttribute()方法的参数值;

或者是Action逻辑有问题没有执行setAttribute()方法就先转向了。

还有另外一个可能,纯粹是jsp文件的问题,例如<logic:iterate>会指定一个id值,

然后在循环里<bean:write>使用这个值作为name的值,如果这两个值不同,

也会出现此异常。(都是一个道理,request里没有对应的对象。)

4.这个情况比较特殊,应该算没有把STRUTS标记库的机制搞明白,

<%@ taglib uri="/WEB-INF/struts-logic.tld" prefix="logic"%>

如果没有在头声明这个就用logic:iterate标记,

在bean:write的时候就会出异常!

---------------------------------6----------------------------------------------------

HibernateException: Not able to obtain connection

org.hibernate.HibernateException: Not able to obtain connection

差不多等于没有连接数据库了。要保证Hibernate的Session工厂能正常工作。

这个一般我们的做法是Session工厂设置成静态的。

然后需要进行操作就打开一个会话。所以在设计的时候注意一下。这个Session工厂对象。

最好是放在

static{}块里。然后用个抽象类去实现它,然后其他的类都实现这个抽象类就搞定了!

---------------------------------7----------------------------------------------------

org.hibernate.MappingException: Error reading resource: XXXX.XML

这个异常出现的报错还有:

Could not configure datastore from input stream

org.dom4j.DocumentException: xxxxx(系统找不到指定的文件。) Nested exception: xxxx

org.hibernate.MappingException: Error reading resource: xxxx

这个异常主要出现的情况有:

(1)没有映射文件,也就是没有xxx.hbm.xml

(2)在xxx.hbm.xml文件中有错误,主要可能是书写错误,或有乱码

(3) 这个问题最恶心,一般人不会注意到,就是要注意配置文件和映射文件都必须用同一个版本的HIBERNATE,怎么看是否用同一个HIBERNATE呢,你可以查看hibernate.cfg.xml和xxx.hbm.xml开头内容:

例如

<?xml version="1.0" encoding="utf-8"?>

<!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN"

"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">

如果一个是2.0一个是3.0到时候DOM4J就会出异常。解析不了咯!

---------------------------------------8-----------------------------------------------

save the transient instance before flushing

1.

(1)做Hibernate多对一的时候,报错:

org.apache.jasper.JasperException: Error creating bean with name 'sf' defined in ServletContext resource [/WEB-INF/applicationContext.xml]: Initialization of bean failed; nested exception is org.hibernate.PropertyNotFoundException: Could not find a getter for person in class com.accp.vo.Course

因为配置文件中的一个 <many-to-one name="person" ../> 中的person属性名称写错了.

还有原因如下:

    1)getter()方法写错了,没有按照javaBean的规则写.

    2)*.hmb.xml中的属性名和pojo中的是否一致.注意:属性名是否多了空格.

(2).Hibernate多对一报错:

save the transient instance before flushing

具体错误为在one-to-many中,对many进行save时,由于没有加入one的外建,导致错误。

具体解决方法:

(1)、在one的hbm文件中 invest=true

(2)、在many保存前务必要求one的存在

---------------------------------------9-----------------------------------------------

Named query not known

在使用session.getNamedQuery("XXX");执行HQL语句的时候,没有找 XXX语句,这主要是2个原因

1:粗心,把名给写错了

2:XXX的HQL语句所在的.hbm.xml没有在hibernate.cfg.xml里注册过!

---------------------------------------10----------------------------------------------

Expected positional parameter count: X, actual parameters:

org.hibernate.QueryException: Expected positional parameter count: 2, actual parameters: [] [SELECT * from Ad where gr_id = ? and adst_id = ?]

这样的异常主要原因是:

//以下为例子

Query query =session.getNamedQuery("OnlienAd");

   query.setParameter(0,gray.getId());

   query.setParameter(1,Globals.UPED_STATE);

可以看到query.setParameter()设置了位,也可以理解成他在要执行的HQL语句中占了2个位置。

然后执行的时候会对应HQL里的?中去。

如果你并没有setParameter(),但你的HQL里却有?那么就会出这个异常。

---------------------------------------11---------------------------------------------

org.hibernate.MappingException: Association references:XXX

---------------------------------------12---------------------------------------------

org.hibernate.exception.SQLGrammarException: could not initialize

无法实例集合。

这个是问题在你当前的实体中有一个一对多需实例的对象。但这个对象却无法实例出来,说明一对多那个多的一方实体有问题,多数情况可能有:

1、数据库该动过但代码没动

2、实体代码有错误

---------------------------------------13---------------------------------------------

org.hibernate.id.IdentifierGenerationException

异常信息:org.hibernate.id.IdentifierGenerationException: ids for this class must be manually assigned before calling save():

异常原因:

<id>元素配置不正确,

<id>元素缺少其子元素<generator></generator>的配置。

解决方法:<id>元素映射了相应数据库表的主键字段,对其子元素<generator class="">,其中class的取值可以为increment、identity、sequence、hilo、native...... 等,更多的可参考hibernate参考文档,一般取其值为native 。

MySQL:identity,increment,hilo,native.

SQL Server:identity,increment,hilo,native.

Oracle:sequence,seqhilo,increment,native.

跨平台开发:native.

native将字段管理交给数据库自己定义。

---------------------------------------8-----------------------------------------------

not-null property references a null or transient value

此异常出现看开头就个单词就知道啦

的实体约束不能为空却为空造成的,那我在这说一下对于这样的情况我们时常犯的错误。

我们在对实体进行更新操作时最好是在一个会话中将数据从数据库中调出并实例BEAN的对象。

然后再对此对象进行更改后在更新。

这样的好处是:

一、数据是最新鲜的。

二、避免不必要的设置,特别是在一对多和多对多的情况下,比如你现在操作的实体并不是在一个会话中得到,而是保存在某些容器中,比如jsp的 SESSION中,虽然这样的实体对象中有很多数据,但是如果你想得到这些实体对应的1对多或多对1的实体时,就会发现会出现session已经关闭的异常。或则在你更新数据时出现not-null异常等。

---------------------------------------14----------------------------------------------

Remember that ordinal parameters are 1-based!

HQL语句有错误

---------------------------------------15---------------------------------------------

failed to lazily initialize a collection of role: XXXXXXXX no session or

这个异常大致意思是说在多对一的时候(并且lazy="false"),对象的实例失败,多数出现的情况有

1、粗心造成

实例对象类名写错之类的

2、逻辑错误

如之前就已经传递过来一个实体对象,然后调用实体对象的方法时牵涉到1对多的情况,但此时SESSION已经关闭,所以根本无法进行一对多的操作。

3、设计到跨度的问题:

这样打比方有多个实体对象,他们直接或则间接的有关联。比如有4个实体,分别是广告信息、广告、广告问答题、广告商:他们之间的关系为:

广告商 1:n 广告

广告 1:n 广告问答题

广告商 1:n 广告商信息

大家可以看到广告和广告商信息是没有直接关系的。但我要添加广告的时候我就必须将广告商的实体做为条件。那么这么一来广告商信息可能间接的就必须用上。下面看我的操作:

ad(广告),subject(题目)

     Ad ad = new Ad();

     ad.setAdProd(adform.getAdProd());

     ad.setIndustry(industry);

     ad.setAdPicture(pagefile.getFileName());

     ad.setAdFlack(adform.getAdFlack());

     ad.setAdDv(dvfile.getFileName());

     ad.setAdContent(adform.getAdContent());

     ad.setGray(gray);

     ad.setAdDate(new Date());

     ad.setOnlinetime(new Long(0));

     //以上为广告的基本信息填写,而重要的是看下面一句,在这里我的思路是subjectFormList是一个动态提交的表单,里面有若干个广告问答题。我将这些问答题变为一个Set,然后作为ad的一个属性。

     Set<Subject> subjectset=getSubjectSet(subjectFormList,ad);

     ad.setSubjects(subjectset);

//然后提交,makePersistent是一个封装的方法,用途就是save()啦。addao是一个DAO,里面有ADUS。

addao.makePersistent(ad);

表面上看来很符合逻辑,只要我们在ad的映射里面加上对subject的级联更新就可以完成这项操作。但实际上会发生我们意想不到的问题,来让我们看一下getSubjectSet()的内容:

public Set getSubjectSet(List<SubjectForm> subjectlist,Ad ad)

{

   Set<Subject> set=new HashSet<Subject>(0);

   Subject subject;

 

   for(Iterator<SubjectForm> it=subjectlist.iterator();it.hasNext();)

   {

    subject=new Subject();

    SubjectForm sf=it.next();

    subject.setSuContent(sf.getSucontent());

    subject.setSuOption(sf.getSuoption());

    subject.setSuResult(Arrays.deepToString(sf.getSuresult()));

    subject.setSuType(String.valueOf(sf.getSutype()));

    subject.setAd(ad);

    set.add(subject);

   }

 

 

   return set;

 

}

我们在这个方法上设一个断点然后跟踪,之后你会发现断点在set.add(subject)只后就会出failed to lazily initialize a collection of role: XXXXXXXX no session or session was closed这个异常,并且这个异常还是出在了广告商的广告信息上 gray.messages。是不是很不可理解?这也是Hibernate的懒汉机制问题。没有任何一样技术是完美的。那我们该怎么处理这样的问题。有很多人以为我们在广告商对广告商信息的隐射上加lazy="false"这样在对gray操作会对messages进行关联,并查询时提出数据。但你会发现改完之后会出现org.hibernate.LazyInitializationException: illegal access to loading collection这个异常。并切lazy="false"是我们不推荐的一种方法。他会降低你的查询效率。

对于这样的情况最好的解决办法就是不要偷懒,对一个实体进行操作的时候就该用那个实体的DAO,即应该有2句HQL。如下把getSubjectSet()改一改:

public void getSubjectSet(List<SubjectForm> subjectlist,Ad ad)

{

   Set<Subject> set=new HashSet<Subject>(0);

   SubjectDAO subjectdao=DAOFactory.getDao(SubjectDAO.class);

 

 

   for(Iterator<SubjectForm> it=subjectlist.iterator();it.hasNext();)

   {

    Subject subject=new Subject();

    SubjectForm sf=it.next();

    subject.setSuContent(sf.getSucontent());

    subject.setSuOption(sf.getSuoption());

    subject.setSuResult(Arrays.deepToString(sf.getSuresult()));

    subject.setSuType(String.valueOf(sf.getSutype()));

    subject.setAd(ad);

    subjectdao.makePersistent(subject);

    //set.add(subject);

   }

 

}//遍历出所有subject一个个的往数据库里加。这样便不会出问题了。

1、OpenSessionInView模式:

以下有2种方法,第1种是结合SPRING,第2种是采用了拦截器

Spring+Hibernate中,     集合映射如果使用lazy="true", 当PO传到View层时, 出现未初始化session已关闭的错误,只能在dao先初始化

parent.getChilds().size();

Spring提供Open Session In View来解决这个问题, 有两种方式

1. Interceptor

       <!--</span><span style="COLOR: rgb(0,128,0)"> =========== OpenSession In View pattern ==============</span><span style="COLOR: rgb(0,128,0)">-->

    <bean id="openSessionInViewInterceptor"

             class="org.springframework.orm.hibernate3.support.OpenSessionInViewInterceptor">

        <property name="sessionFactory" ref="sessionFactory"/>

    </bean>

    <bean id="urlMapping" class="org.springframework.web.servlet.handler.SimpleUrlHandlerMapping">

        <property name="interceptors" ref="openSessionInViewInterceptor"/>

        <property name="mappings">

            <props>

               ......

            </props>

        </property>

    </bean>

2. Filter

<web-app>

<filter>

<filter-name>hibernateFilter</filter-name>

<filter-class>

org.springframework.orm.hibernate.support.OpenSessionInViewFilter

</filter-class>

</filter>

<filter-mapping>

<filter-name>hibernateFilter</filter-name>

<url-pattern>*.do</url-pattern>

</filter-mapping>

</web-app>

第2种解决方法:

Hibernate.initialize()强制加载关联对象

---------------------------------------16----------------------------------------------

Hibernate的No CurrentSessionContext configured 解决方案

Hibernate No CurrentSessionContext configured!

当你使用 Hibernate session factory的getCurrentSession() 函数时候,你可能见到过"No CurrentSessionContext configured!"

出现这个问题,是因为你没有正确配置 Hibernate internal session context management .

你只要在Hibernate.cfg.xml文件中添加下面的配置就可以了:

<property name="current_session_context_class">thread</property>

---------------------------------------17---------------------------------------------

你可能感兴趣的:(spring,Hibernate,bean,struts,JUnit)