自定义Spring中的java.lang.NumberFormatException: null的问题

一、带你们看一下出现错的截图

图所示

二、分析一下为何会出现这个错误

1、从报错来说,应该是括号里面的“变量”是null,当然也就无法转换为Integer类型了

我所写的代码如下:
当然这里的【val】值事先是要通过自定义的Value.class获取到的

图所示
2、不管你代码怎么写,只需要注意一个地方

就是你最后要【Integer.parseInt(value);】或者你可以多写一个【Double.parseDouble(value);】

括号里的值value是绝对不能为空的,如果value是null,就会出现截图中的错误,出了这种错误你要看value有没有被赋值即可,再需要注意的是如果你写的不仅仅是【Integer.parseInt(value);】,那你就要检查value也不能是其他无法转换为数字的字符串。

三、eclipse创建的Spring工程结构【Maven仓库配置的】

  • 工程名:Spring03Imitate
    • src/main/resources
      • spring-bean.xml
    • src/main/java
      • package: com.yc.main.bean
        • Person.java
      • package: com.yc.main.core
        • ApplicationContext.java
        • Bean.java
        • Value.java
        • YcClassPathXmlApplicationContext.java
      • package: com.yc.main.test
        • SpringTest.java
    • pom.xml

四、下面将模仿Spring【自定义Spring】代码分享给大家

spring-bean.xml

<?xml version="1.0" encoding="UTF-8"?>
<beans>
    <!-- 需要扫描的包路径 -->
	<component-scan base-package="com.yc.main"></component-scan>	
</beans>

Person.java

package com.yc.main.bean;

import com.yc.main.core.Bean;
import com.yc.main.core.Value;

@Bean
public class Person {
	@Value("1002")
	private String pid;
	@Value("W")
	private String pname;
	@Value("20")
	private Integer age;
	
	@Override
	public String toString() {
		return "Person [pid=" + pid + ", pname=" + pname + ", age=" + age + "]";
	}

	public String getPid() {
		return pid;
	}

	public void setPid(String pid) {
		this.pid = pid;
	}

	public String getPname() {
		return pname;
	}

	public void setPname(String pname) {
		this.pname = pname;
	}

	public Integer getAge() {
		return age;
	}

	public void setAge(Integer age) {
		this.age = age;
	}

	@Override
	public int hashCode() {
		final int prime = 31;
		int result = 1;
		result = prime * result + ((age == null) ? 0 : age.hashCode());
		result = prime * result + ((pid == null) ? 0 : pid.hashCode());
		result = prime * result + ((pname == null) ? 0 : pname.hashCode());
		return result;
	}

	@Override
	public boolean equals(Object obj) {
		if (this == obj)
			return true;
		if (obj == null)
			return false;
		if (getClass() != obj.getClass())
			return false;
		Person other = (Person) obj;
		if (age == null) {
			if (other.age != null)
				return false;
		} else if (!age.equals(other.age))
			return false;
		if (pid == null) {
			if (other.pid != null)
				return false;
		} else if (!pid.equals(other.pid))
			return false;
		if (pname == null) {
			if (other.pname != null)
				return false;
		} else if (!pname.equals(other.pname))
			return false;
		return true;
	}
}

ApplicationContext.java

package com.yc.main.core;

public interface ApplicationContext {
	public Object getBean(String name);
}

Bean.java

package com.yc.main.core;

import java.lang.annotation.Documented;
import java.lang.annotation.ElementType;
import java.lang.annotation.Inherited;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
public @interface Bean {
	String value() default "";
}

Value.java

package com.yc.main.core;

import java.lang.annotation.Documented;
import java.lang.annotation.ElementType;
import java.lang.annotation.Inherited;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
public @interface Value {
	String value() default "";
}

YcClassPathXmlApplicationContext.java

package com.yc.main.core;

import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.lang.reflect.Field;
import java.net.URL;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import org.dom4j.Document;
import org.dom4j.DocumentException;
import org.dom4j.Element;
import org.dom4j.XPath;
import org.dom4j.io.SAXReader;

public class YcClassPathXmlApplicationContext implements ApplicationContext {

	private static String locationConfig = "spring-bean.xml";
	private List<String> classNames = new ArrayList<String>(); // 存放所有扫描的类路径
	private Map<String, Object> beans = new HashMap<String, Object>();

	public YcClassPathXmlApplicationContext() {
		this(locationConfig);
	}

	public YcClassPathXmlApplicationContext(String config) {
		locationConfig = config;
		parseXml();
	}

	/**
	 * 解析xml文件
	 */
	@SuppressWarnings("unchecked")
	private void parseXml() {
		if (locationConfig == null || "".equals(locationConfig)) {
			throw new RuntimeException("读取bean的配置文件错误...");
		}

		SAXReader reader = new SAXReader();
		Document doc = null;

		try (InputStream is = this.getClass().getClassLoader().getResourceAsStream(locationConfig)) {
			doc = reader.read(is);

			XPath xpath = doc.createXPath("//component-scan");
			List<Element> nodes = xpath.selectNodes(doc);

			if (nodes == null || nodes.isEmpty()) {
				return;
			}

			String basePackage = nodes.get(0).attributeValue("base-package");

			// 获取这包下的所有类及子包下面的类
			doScannerPackage(basePackage);

			doInstanceObject(); // 实例化对象
		} catch (DocumentException e) {
			e.printStackTrace();
		} catch (IOException e1) {
			e1.printStackTrace();
		}
	}

	/**
	 * 扫描包
	 * 
	 * @param basePackage
	 */
	private void doScannerPackage(String basePackage) {
		if (basePackage == null || "".equals(basePackage)) {
			throw new RuntimeException("配置文件读取失败,请配置要扫描的包...");
		}

		URL url = this.getClass().getClassLoader().getResource(basePackage.replaceAll("\\.", "/")); // 将包路径转成目录路径
		File file = new File(url.getFile());
		if (!file.exists() && !file.isDirectory()) {
			throw new RuntimeException("您配置的要扫描的包路径有误...");
		}
		getClassInfo(basePackage, file);
//		classNames.forEach(System.out::println);
	}

	/**
	 * 扫描获取包下的类
	 * 
	 * @param basePackage 这个类的包路径
	 * @param file        开始扫描的目录
	 */
	private void getClassInfo(String basePackage, File file) {
		if (file.isDirectory()) {
			for (File f : file.listFiles()) { // 获取指定目录下在目录或文件
				if (f.isDirectory()) {
					getClassInfo(basePackage + "." + f.getName(), f);
				} else {
					classNames.add(basePackage + "." + f.getName().replace(".class", ""));
				}
			}
		}
	}

	private void doInstanceObject() {
		// 先判断哪些上有@Bean注解
		if (classNames.isEmpty()) {
			return;
		}

		Class<?> cls = null;
		String beanName = null;
		Bean bean = null;
		Value value = null;
		Field[] fields = null;
		Object obj = null;
		Class<?> type = null;

		String val = null;
		for (String className : classNames) {
			try {
				cls = Class.forName(className);

				// 现在这个类上有没有这个注解
				if (!cls.isAnnotationPresent(Bean.class)) {
					continue;
				}

				obj = cls.newInstance(); // 创建一个对象

				bean = cls.getAnnotation(Bean.class);

				// 这个注解的value值有给吗
				beanName = bean.value();
				if ("".equals(beanName)) { // 默认情况下是将整个类的第一个字母转成小写
					beanName = toFirstLowerCase(cls.getSimpleName());
				}

				// 获取整个类下的所有属性
				fields = cls.getDeclaredFields();

				if (fields == null || fields.length <= 0) {
					continue;
				}

				// 循环判断这些属性有没有Value注解
				for (Field fd : fields) {
					// 循环判断这些属性上有没有Value注解
					value = fd.getAnnotation(Value.class);
					if (value == null) {
						continue;
					}

					val = value.value();

					if ("".equals(val)) {
						continue;
					}

					//我想给这个属性直接注值,不想调用这个属性的setter方法
					fd.setAccessible(true);

					//获取这个属性的类型
					type = fd.getType();
					if (type == Integer.class || type == Integer.TYPE) {
						//如果有,则判断Value注解中的value属性有没有给值,如果有则需要注值
						fd.set(obj, Integer.parseInt(val));
					} else if (type == Double.class || type == Double.TYPE) {
						fd.set(obj, Double.parseDouble(val));
					} else {
						fd.set(obj, val);
					}
				}

				beans.put(beanName, obj);
			} catch (ClassNotFoundException e) {
				e.printStackTrace();
			} catch (IllegalArgumentException e) {
				e.printStackTrace();
			} catch (IllegalAccessException e) {
				e.printStackTrace();
			} catch (InstantiationException e) {
				e.printStackTrace();
			}
		}
	}

	/**
	 * 将给定字符串的第一个字母转成小写
	 * 
	 * @param simpleName
	 * @return
	 */
	private String toFirstLowerCase(String name) {
		char[] chs = name.toCharArray();
		chs[0] += 32;
		return String.valueOf(chs);
	}

	@Override
	public Object getBean(String name) {
		return beans.getOrDefault(name, null);
	}

}

SpringTest.java

package com.yc.main.test;

import com.yc.main.bean.Person;
import com.yc.main.core.ApplicationContext;
import com.yc.main.core.YcClassPathXmlApplicationContext;

public class SpringTest {
	public static void main(String[] args) {
		ApplicationContext ioc = new YcClassPathXmlApplicationContext("spring-bean.xml");
		Person person = (Person) ioc.getBean("person");
		System.out.println(person);
	}
}

pom.xml

<project xmlns="http://maven.apache.org/POM/4.0.0"
	xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
	xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
	<modelVersion>4.0.0</modelVersion>

	<groupId>com.yc.main</groupId>
	<artifactId>Spring03Imitate</artifactId>
	<version>0.0.1-SNAPSHOT</version>
	<packaging>jar</packaging>

	<name>Spring03Imitate</name>
	<url>http://maven.apache.org</url>

	<properties>
		<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
	</properties>

	<dependencies>
		<dependency>
			<groupId>dom4j</groupId>
			<artifactId>dom4j</artifactId>
			<version>1.6</version>
		</dependency>

		<dependency>
			<groupId>jaxen</groupId>
			<artifactId>jaxen</artifactId>
			<version>1.2.0</version>
		</dependency>
	</dependencies>
</project>

五、总结

以上就是所有的内容,希望可以帮到大家。
【注意】我所创的只是Maven工程下的视图,创什么工程无所谓,只是Maven仓库对管理一些用到的包比较方便。

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: java.lang.NumberFormatException:null 意味着尝试将空值(null)转换为数字时发生了错误。这通常是因为您在使用Integer.parseInt()或Double.parseDouble()等转换方法时传入了空值。应该在转换之前检查给定值是否为空。 ### 回答2: Java.lang.NumberFormatException: nullJava编程常见的错误异常之一。这个异常通常发生在程序试图将一个字符串转换为数字类型的时候,而这个字符串不符合数字格式的要求。 当程序试图把某个字符串转换成数字类型的时候,Java会根据字符串的内容来判断其是否符合数字格式,如果不符合则会抛出NumberFormatExceptio:null异常。 例如,以下代码,当将字符串“abc”转换为整数时,会抛出上述的异常: ``` String str = "abc"; int num = Integer.parseInt(str); ``` 这个异常的原因是Java无法将字符串“abc”解析成整数。如果想要解决这个问题,有以下几种办法: 1.检查字符串格式是否正确,确保它只包含数字字符。 2.使用try-catch语句来捕捉异常,然后在catch块进行处理,例如输出错误信息。 3.对于无法处理的非数字字符串,可以用默认值替代,或者返回错误提示给用户,让其重新输入正确的值。 总之,当开发Java程序的时候,遇到java.lang.NumberFormatException交错应当保持冷静,仔细检查代码,找出问题的根源,然后采取对应的解决措施来避免这个异常的发生。 ### 回答3: java.lang.numberformatexception:nullJava一个常见的异常(Exception),它通常出现在需要将字符串转换为数字时。当程序尝试将一个空(null)对象转换为数字时,就会出现这个异常。 一般来说,如果一个程序需要进行数字计算,那么输入的数据必须是数字类型。而当用户输入的数据是字符串类型时,需要将其转换为数字类型,这就需要使用Java的方法来实现。例如,我们可以使用Java的Integer.parseInt()方法或Double.parseDouble()方法将字符串类型转换为int或double类型。 当我们尝试使用这些方法将一个空的字符串转换为数字时,就会出现java.lang.numberformatexception:null异常。这是因为这些方法要求输入的字符串必须是一个有效的数字格式,如果输入的字符串为空,则无法进行转换。 为了避免出现这个异常,我们需要在使用这些方法进行字符串转换时,加入一些判断条件,比如判断输入的字符串是否为空,或者是否是一个合法的数字格式等。在代码进行这些判断,可以避免程序出现异常,并提高程序的健壮性。 总之,java.lang.numberformatexception:nullJava常见的一个异常,在处理数字计算时,我们需要特别注意避免出现这个异常,并做好相关的异常处理工作,在程序加入异常处理代码,可以有效提高程序的稳定性和鲁棒性。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值