Mybatis获取SqlSession时报空指针错误解决

今天开始接触Mybatis,在学习第一步时就遇到了一个空指针的错误,所以记录一下,免得以后再犯:


具体错误如下
Exception in thread "main" org.apache.ibatis.exceptions.PersistenceException: 
### Error opening session.  Cause: java.lang.NullPointerException
### Cause: java.lang.NullPointerException
    at org.apache.ibatis.exceptions.ExceptionFactory.wrapException(ExceptionFactory.java:30)
    at org.apache.ibatis.session.defaults.DefaultSqlSessionFactory.openSessionFromDataSource(DefaultSqlSessionFactory.java:100)
    at org.apache.ibatis.session.defaults.DefaultSqlSessionFactory.openSession(DefaultSqlSessionFactory.java:47)
    at com.mybatis.test.UserDao.init(UserDao.java:32)
    at com.mybatis.test.UserDao.main(UserDao.java:41)
Caused by: java.lang.NullPointerException
    at org.apache.ibatis.session.defaults.DefaultSqlSessionFactory.openSessionFromDataSource(DefaultSqlSessionFactory.java:95)
    ... 3 more
具体的代码
public class UserDao {

    private SqlSession session;

    public void select(int id){
        User user = session.selectOne("findById", id);
        System.out.println(user);
    }

    public void init(){
        String resource = "com/mybatis/map/mybatisConfig.xml";
        try {
            Reader reader = Resources.getResourceAsReader(resource);
            SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(reader);
            session = sqlSessionFactory.openSession();    <----错误定位代码

        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    public static void main(String[] args) {
        UserDao userDao = new UserDao();
        userDao.init();
        userDao.select(1);

    }
}
mybatis配置文件如下
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE configuration
    PUBLIC "-//mybatis.org/DTD Config 3.0//EN"
    "http://mybatis.org/dtd/mybatis-3-config.dtd">
<configuration>
    <environments default="development">
        <environment id="database1">
            <transactionManager type="JDBC"></transactionManager>
            <dataSource type="POOLED">
                <property name="driver" value="com.mysql.jdbc.Driver"/>
                <property name="url" value="jdbc:mysql://localhost/mybatis_test1"/>
                <property name="username" value="root"/>
                <property name="password" value="root"/>
            </dataSource>
        </environment>
        <environment id="database2">
            <transactionManager type="JDBC"></transactionManager>
            <dataSource type="POOLED">
                <property name="driver" value="com.mysql.jdbc.Driver"/>
                <property name="url" value="jdbc:mysql://localhost/mybatis_test2"/>
                <property name="username" value="root"/>
                <property name="password" value="root"/>
            </dataSource>
        </environment>
        <environment id="database3">
            <transactionManager type="JDBC"></transactionManager>
            <dataSource type="POOLED">
                <property name="driver" value="com.mysql.jdbc.Driver"/>
                <property name="url" value="jdbc:mysql://localhost/mybatis_test3"/>
                <property name="username" value="root"/>
                <property name="password" value="root"/>
            </dataSource>
        </environment>
    </environments>
    <mappers>
        <mapper resource="com/mybatis/map/User.xml"/>
    </mappers>
</configuration>

通过调试发现,SqlSessionFactory不为 null,但是在

session = sqlSessionFactory.openSession();

时报了错误

后来查看mybatis文档才了解到,在mybatis配置时,可能由于我们开发时的数据库环境和最终上线时的数据库环境不同,因此可以在配置文件中配置多个数据库环境;
即在 < enviroments >标签下可以配置多个< enviroment>标签,每一个< enviroment >标签对应一个数据库环境
而不同的数据库环境通过< enviroment > 标签的 id 属性用以区分
那么具体开发时如果知道使用的是哪一个环境呢?
在< envirments> 标签里有一个default属性,该属性对应的是下面的不同的< enviroment > 的id属性
default的值为哪一个id,则代表此时使用的是哪一个environment数据库环境

由于一开始我照着教程写的代码,直接填的default=”development”,我以为该值是一个mybatis框架下的一个默认值,< environments default=”development”>,因此它在获取environment时,在我的配置中未查找到id为 development的enviroment (database1,databse2,database3),在openSession时会导致获取为空

因此解决办法

< environment default=”“> 该值为下面配置的< environment id=”“>中需要使用的环境的id即可

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE configuration
    PUBLIC "-//mybatis.org/DTD Config 3.0//EN"
    "http://mybatis.org/dtd/mybatis-3-config.dtd">
<configuration>
    <environments default="database1">  <!-- default属性选择此次开发时需要的数据库环境的id值即可 (database1、database2、database3) -->
        <environment id="database1">
            <transactionManager type="JDBC"></transactionManager>
            <dataSource type="POOLED">
                <property name="driver" value="com.mysql.jdbc.Driver"/>
                <property name="url" value="jdbc:mysql://localhost/mybatis_test1"/>
                <property name="username" value="root"/>
                <property name="password" value="root"/>
            </dataSource>
        </environment>
        <environment id="database2">
            <transactionManager type="JDBC"></transactionManager>
            <dataSource type="POOLED">
                <property name="driver" value="com.mysql.jdbc.Driver"/>
                <property name="url" value="jdbc:mysql://localhost/mybatis_test2"/>
                <property name="username" value="root"/>
                <property name="password" value="root"/>
            </dataSource>
        </environment>
        <environment id="database3">
            <transactionManager type="JDBC"></transactionManager>
            <dataSource type="POOLED">
                <property name="driver" value="com.mysql.jdbc.Driver"/>
                <property name="url" value="jdbc:mysql://localhost/mybatis_test3"/>
                <property name="username" value="root"/>
                <property name="password" value="root"/>
            </dataSource>
        </environment>
    </environments>
    <mappers>
        <mapper resource="com/mybatis/map/User.xml"/>
    </mappers>
</configuration>

最终问题解决

### 回答1: Mybatis获取SqlSession对象的方法有两种: 1. 通过SqlSessionFactory获取SqlSession对象 首先需要创建一个SqlSessionFactory对象,可以通过读取mybatis的配置文件来创建,然后通过SqlSessionFactory对象的openSession()方法获取SqlSession对象。 示例代码: //读取mybatis配置文件 InputStream inputStream = Resources.getResourceAsStream("mybatis-config.xml"); //创建SqlSessionFactory对象 SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream); //获取SqlSession对象 SqlSession sqlSession = sqlSessionFactory.openSession(); 2. 通过Spring框架获取SqlSession对象 如果项目中使用了Spring框架,可以通过Spring的容器来获取SqlSession对象。需要在Spring的配置文件中配置SqlSessionFactorySqlSessionTemplate两个Bean,然后通过@Autowired注解来注入SqlSessionTemplate对象。 示例代码: <!-- 配置SqlSessionFactory --> <bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean"> <property name="dataSource" ref="dataSource"/> <property name="configLocation" value="classpath:mybatis-config.xml"/> </bean> <!-- 配置SqlSessionTemplate --> <bean id="sqlSessionTemplate" class="org.mybatis.spring.SqlSessionTemplate"> <constructor-arg ref="sqlSessionFactory"/> </bean> //注入SqlSessionTemplate对象 @Autowired private SqlSessionTemplate sqlSessionTemplate; ### 回答2: MyBatis获取SqlSession对象是非常重要的,SqlSessionMyBatis操作数据库的核心对象,用来执行SQL语句、提交事务、获取映射器等。 三种获取SqlSession对象的方式: 1.使用SqlSessionFactoryBuilder来读取配置文件,创建SqlSessionFactory对象,再通过SqlSessionFactory对象创建SqlSession对象: ``` String resource = "mybatis-config.xml"; InputStream inputStream = Resources.getResourceAsStream(resource); SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream); SqlSession sqlSession = sqlSessionFactory.openSession(); ``` 2.使用MyBatis提供的Resources类,直接获取配置文件流,创建SqlSessionFactory对象,再通过SqlSessionFactory对象创建SqlSession对象: ``` String resource = "mybatis-config.xml"; Reader reader = Resources.getResourceAsReader(resource); SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(reader); SqlSession sqlSession = sqlSessionFactory.openSession(); ``` 3.使用Spring框架集成MyBatis,通过SqlSessionFactoryBean配置SqlSessionFactory,再通过SqlSessionFactory来创建SqlSession对象: ``` <bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource"> <property name="driverClassName" value="com.mysql.jdbc.Driver"/> <property name="url" value="jdbc:mysql://localhost:3306/test?useSSL=false"/> <property name="username" value="root"/> <property name="password" value="password"/> </bean> <bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean"> <property name="dataSource" ref="dataSource"/> <property name="configLocation" value="classpath:mybatis-config.xml"/> </bean> <bean id="sqlSession" class="org.mybatis.spring.SqlSessionTemplate"> <constructor-arg ref="sqlSessionFactory"/> </bean> ``` 无论是哪种方式获取SqlSession对象,都要注意调用close方法将SqlSession对象关闭,释放数据库资源。 否则,容易出现数据库连接泄漏的问题,影响系统性能。因此,在使用MyBatis操作数据库时,一定要注意SqlSession的使用和关闭。 ### 回答3: Mybatis是一个基于Java的开源数据访问框架,提供了面向对象的持久化操作方式,是许多Java Web应用的首选框架。其核心是SqlSessionMybatis通过SqlSession与数据库进行交互。因此,获取SqlSession对象是使用Mybatis的第一步。 获取SqlSession对象主要有以下两种方式: 1. SqlSessionFactoryBuilder 首先,需要创建SqlSessionFactoryBuilder对象,通过它配置Mybatis并创建SqlSessionFactory对象,再通过SqlSessionFactory对象生成SqlSession对象。示例代码如下: ``` InputStream inputStream = Resources.getResourceAsStream("mybatis-config.xml"); SqlSessionFactoryBuilder builder = new SqlSessionFactoryBuilder(); SqlSessionFactory sqlSessionFactory = builder.build(inputStream); SqlSession sqlSession = sqlSessionFactory.openSession(); ``` 这里,首先使用Resources.getResourceAsStream加载Mybatis配置文件mybatis-config.xml,然后创建SqlSessionFactoryBuilder对象builder,调用其build方法创建SqlSessionFactory对象sqlSessionFactory,最后通过SqlSessionFactory对象的openSession方法获取SqlSession对象sqlSession。 2. @Inject注解 另外一种获取SqlSession对象的方式是使用@Inject注解。通过在Mapper接口的字段或setter方法上添加@Inject注解,Mybatis会自动注入SqlSession对象。示例代码如下: ``` public interface UserMapper { @Inject SqlSession sqlSession; // 或者 void setSqlSession(SqlSession sqlSession); } ``` 这里,通过@Inject注解,Mybatis会自动注入SqlSession对象sqlSession。另外也可以通过setter方法进行注入。 总结起来,获取SqlSession对象有两种常用方式:通过SqlSessionFactoryBuilder对象创建,或在Mapper接口中使用@Inject注解。根据实际情况选择合适的方式进行使用即可。
评论 16
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值