java.lang.NoSuchMethodException:最易忽略的地方

本文介绍了一个使用Struts框架实现的用户注册功能,重点在于如何通过多层架构(包括Action层、Service层和DAO层)来检查用户名是否重复,并详细展示了具体的Java代码实现。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

http://669341085.iteye.com/blog/777171

action层

Java代码   收藏代码
  1. @Controller  
  2. @Scope("prototype")  
  3. @SuppressWarnings("serial")  
  4. public class UserAction extends CommonAction implements ModelDriven<MemoryUser>{  
  5.     @Resource  
  6.     private UserService userService;  
  7.     private MemoryUser memoryUser=new MemoryUser();  
  8.     //用户名是否重复的标识,“Y”标识重复,“N”标识不重复  
  9.     private String isUserNameRepeat;  
  10.     @Override  
  11.     public MemoryUser getModel() {  
  12.         return memoryUser;  
  13.     }  
  14.       
  15.     public String getIsUserNameRepeat() {  
  16.         return isUserNameRepeat;  
  17.     }  
  18.   
  19.     public void setIsUserNameRepeat(String isUserNameRepeat) {  
  20.         this.isUserNameRepeat = isUserNameRepeat;  
  21.     }  
  22.   
  23.     /** 
  24.      * @Description: 用户注册时检测用户名是否重复 
  25.      * @author: 
  26.      * @Date: 
  27.      * @param: 
  28.      * @return: 
  29.      * @throws: 
  30.      */  
  31.     private String checkUserName() throws Exception {  
  32.         isUserNameRepeat="N";  
  33.         System.out.println("执行++++++++++");  
  34.         System.out.println(userService.checkUserName(getModel().getUserName()));  
  35.         if(userService.checkUserName(getModel().getUserName())) {  
  36.             isUserNameRepeat="Y";  
  37.         }  
  38.         return SUCCESS;  
  39.     }  

 

Service层

Java代码   收藏代码
  1. @Service  
  2. public class UserService {  
  3.     @Resource  
  4.     private UserDao userDao;  
  5.       
  6.     /** 
  7.      * @Description: 用户注册时检测用户名是否重复 
  8.      * @author: 
  9.      * @Date: 
  10.      * @param:用户输入的用户名 
  11.      * @return:是否重复 
  12.      * @throws: 
  13.      */  
  14.     public Boolean checkUserName(String userName) throws Exception {  
  15.         return userDao.checkUserName(userName);  
  16.     }  
  17. }  

 

Dao层

Java代码   收藏代码
  1. import javax.annotation.Resource;  
  2.   
  3. import org.hibernate.Session;  
  4. import org.hibernate.SessionFactory;  
  5. import org.hibernate.Transaction;  
  6. import org.springframework.stereotype.Repository;  
  7.   
  8. import cn.cd.model.MemoryUser;  
  9.   
  10. @Repository  
  11. public class UserDao {  
  12.     @Resource  
  13.     private SessionFactory sessionFactory;  
  14.   
  15.     public SessionFactory getSessionFactory() {  
  16.         return sessionFactory;  
  17.     }  
  18.   
  19.     public void setSessionFactory(SessionFactory sessionFactory) {  
  20.         this.sessionFactory = sessionFactory;  
  21.     }   
  22.       
  23.     /** 
  24.      * @Description: 用户注册时检测用户名是否重复 
  25.      * @author: 
  26.      * @Date: 
  27.      * @param:用户输入的用户名 
  28.      * @return:是否重复 
  29.      * @throws: 
  30.      */  
  31.     public Boolean checkUserName(String userName) throws Exception {  
  32.         Boolean flag=false;  
  33.         Session session=sessionFactory.openSession();  
  34.         Transaction tx=null;  
  35.         tx=session.beginTransaction();  
  36.         MemoryUser memoryUser=(MemoryUser) session.get(MemoryUser.class, userName);  
  37.         tx.commit();  
  38.         if(null!=memoryUser) {  
  39.             flag=true;  
  40.         }  
  41.         return flag;  
  42.     }  
  43.       
  44. }  

 

Struts配置

Java代码   收藏代码
  1. <?xml version="1.0" encoding="UTF-8" ?>  
  2. <!DOCTYPE struts PUBLIC "-//Apache Software Foundation//DTD Struts Configuration 2.1//EN" "http://struts.apache.org/dtds/struts-2.1.dtd">  
  3. <struts>  
  4.     <!-- 与用户有关的action的配置 -->  
  5.     <package name="memoryUser" extends="json-default" namespace="/memoryUser">  
  6.         <!-- 验证用户名是否重复-->  
  7.         <action name="checkUserName" class="cn.cd.action.UserAction" method="checkUserName">  
  8.             <result name="success" type="json">  
  9.                 <param name="includeProperties">  
  10.                     isUserNameRepeat  
  11.                 </param>  
  12.                 <param name="noCache">true</param>    
  13.                 <param name="ignoreHierarchy">false</param>  
  14.             </result>  
  15.         </action>   
  16.     </package>  
  17. </struts>      

 

一直报上面的错误,找了好久发现配置都是对的,最后无意中发现Action中的checkUserName()居然是private……大概是写属性写的太多,写顺手了……

<think>好的,我现在需要处理用户关于Android运行时InvocationTargetException的问题。首先,我得回忆一下这个异常的基本信息。InvocationTargetException通常在反射调用方法时抛出,比如通过Method.invoke()调用某个方法,如果被调用的方法内部抛出了异常,就会包装成InvocationTargetException。所以根本原因在于被反射调用的方法本身出现了问题。 接下来,用户需要的是原因和解决方案。我需要分步骤分析可能的原因。首先,常见的情况是不是被调用的方法本身抛出了异常,比如空指针、数组越界等。这时候需要查看异常的堆栈跟踪,找到根本原因。其次,可能是在使用动态代理或AOP框架时,代理方法中的错误导致的。另外,参数不匹配也可能导致问题,比如反射调用时传递的参数类型或数量不正确。还有可能涉及版本兼容性问题,某些方法在不同API级别中的行为不同,导致调用失败。 然后,解决方案方面,首先要检查堆栈跟踪,找到被包裹的原始异常。用户可能需要使用getCause()方法获取根本原因。其次,验证反射调用的正确性,确认方法名、参数类型是否正确,特别是泛型擦除可能导致的问题。另外,检查动态代理或AOP相关的代码,确保代理逻辑无误。参数校验也很重要,确保传递的参数类型和数量与方法签名一致。版本兼容性方面,需要使用适合的API方法,或者进行版本判断。 还需要考虑用户提供的引用内容。不过看起来引用内容可能和这个问题关联不大,比如引用[1]提到的MissingFormatArgumentException和ProGuard的问题,引用[3]关于Go语言的作用域。可能需要忽略这些,除非用户的问题中涉及ProGuard优化导致反射异常,这时候可以建议关闭优化或更新ProGuard版本,如引用[2]提到的使用-dontoptimize选项。这可能是一个相关点,如果异常是在混淆后出现的,可能需要调整ProGuard配置。 另外,用户可能需要代码示例来说明如何正确处理异常,比如在try-catch块中捕获InvocationTargetException,并通过getCause()获取原始异常。同时,提醒用户在日志中详细记录原始异常的信息,以便调试。 总结下来,回答的结构应该是先解释异常的原因,然后分点列出常见原因和对应的解决方案,后给出示例代码和注意事项。确保使用正确的中文术语,并按照用户的格式要求使用行内公式(如果有的话,但这个问题可能不太涉及数学公式),但在这个问题中可能不需要,所以可能忽略数学格式部分。同时生成相关问题,帮助用户进一步排查问题,比如关于反射调用、ProGuard配置、异常处理等。</think>### Android中InvocationTargetException错误的原因与解决方案 **原因分析** `InvocationTargetException` 是Java反射调用(如通过 `Method.invoke()`)时抛出的异常,其核心原因是**被调用的方法内部抛出了其他异常**。该异常本身仅作为包装器,实际错误需通过 `getCause()` 方法获取[^1]。 #### 常见具体原因: 1. **被调用方法自身异常** - 空指针(`NullPointerException`)、数组越界(`ArrayIndexOutOfBoundsException`)等运行时异常。 - 示例:反射调用一个未正确处理参数的方法时触发崩溃。 2. **反射调用参数不匹配** - 参数类型或数量与方法签名不一致(如传递 `String` 代替 `int`)。 3. **动态代理/AOP框架问题** - 代理类逻辑错误导致目标方法执行失败。 4. **兼容性问题** - 高版本API方法在低版本设备调用时引发 `NoSuchMethodError`(可能被包装为 `InvocationTargetException`)。 --- ### 解决方案 #### 1. 定位根本原因 通过 `getCause()` 提取原始异常: ```java try { method.invoke(obj, args); } catch (InvocationTargetException e) { Throwable rootCause = e.getCause(); // 关键:获取实际异常 rootCause.printStackTrace(); } ``` #### 2. 验证反射调用 - **检查方法签名**:确认方法名、参数类型和数量,尤其注意泛型擦除问题。 ```java // 示例:获取方法时需明确参数类型 Method method = MyClass.class.getMethod("calculate", int.class, String.class); ``` - **参数校验**:使用日志输出参数值,确保非空且类型正确。 #### 3. 处理动态代理问题 若使用动态代理(如Retrofit、Spring AOP),检查代理逻辑是否拦截或修改了原始方法调用。 #### 4. 兼容性适配 - **API版本检查**: ```java if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { // 调用高版本API } else { // 降级处理 } ``` #### 5. ProGuard混淆问题 若异常出现在混淆后的APK中,需保留反射调用的类和方法(添加ProGuard规则): ```proguard -keepclassmembers class com.example.MyClass { public void myMethod(**); } ``` 若优化引发问题,可尝试禁用优化(引用[2]): ```proguard -dontoptimize ``` --- ### 示例代码 ```java try { Method method = MyClass.class.getMethod("processData", String.class); method.invoke(myInstance, "input"); } catch (NoSuchMethodException | IllegalAccessException e) { e.printStackTrace(); } catch (InvocationTargetException e) { // 关键:打印实际异常 e.getCause().printStackTrace(); } ``` ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值