6.Jfinal分页实例

1.创建web项目,导入jfinal,mysql,jsp相关依赖

这里写图片描述

2.项目结构
这里写图片描述

3.在res中添加jdbc.txt
修改为自己的数据库

jdbcUrl = jdbc:mysql://localhost/test?characterEncoding=utf8
user = root
password = admin
devMode = true
showSql = true

我的数据库如下,这里可以自己定义
这里写图片描述

4.创建主配置类
DemoConfig.java

public class DemoConfig extends JFinalConfig{
    @Override
    public void configConstant(Constants arg0) {
        arg0.setViewType(ViewType.JSP);
        loadPropertyFile("jdbc.txt");   
    }
    @Override
    public void configHandler(Handlers arg0) {
        //项目根路径
        arg0.add(new ContextPathHandler("baseUrl"));   
    }
    @Override
    public void configInterceptor(Interceptors arg0) {

    }
    @Override
    public void configPlugin(Plugins arg0) {
        C3p0Plugin c3p0=new C3p0Plugin(getProperty("jdbcUrl"),
                getProperty("user"),               
                getProperty("password"));      
        arg0.add(c3p0);
        ActiveRecordPlugin activeRecord=new ActiveRecordPlugin(c3p0);
        activeRecord.addMapping("user",User.class);
        arg0.add(activeRecord);    
    }
    @Override
    public void configRoute(Routes arg0) {
        arg0.add("page",PageController.class,"view");
     //对应第八步                                        
    }
    public static void main(String[] args) {
        JFinal.start("WebContent", 80, "/", 5);
    }

}

5.修改web.xml,加入jfinal过滤器(servlet)

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://java.sun.com/xml/ns/javaee" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd" id="WebApp_ID" version="3.0">

  <display-name>jfinal_demo</display-name>
    <filter>
        <filter-name>jfinal</filter-name>
        <filter-class>com.jfinal.core.JFinalFilter</filter-class>
        <init-param>
            <param-name>configClass</param-name>
            <param-value>com.cjq.config.DemoConfig</param-value>
        </init-param>
    </filter>
    <filter-mapping>
        <filter-name>jfinal</filter-name>
        <url-pattern>/*</url-pattern>
    </filter-mapping>
</web-app>

6.在com.cjq.model包下创建User.java

public class User extends Model<User>{

    public static final User dao=new User();

}

7.在com.cjq.service中创建UserService.java

public class UserService {
    /**
     * 
     * @param pageNum页号
     * @param pageSize每页几条记录
     * @return
     */
    public static Page<User> pagInate(int pageNum,int pageSize){               
        return User.dao.paginate(pageNum, pageSize,"select *","from user");
    }
}

8.在com.cjq.controller创建PageController.java

public class PageController extends Controller {
   final  int PAGE_SIZE=2;//默认每页2条记录
    /**
     * 初始页为第一页
     */
    public void index(){
        System.out.println("--------------");
        List<User> list=UserService.pagInate(1,PAGE_SIZE).getList();
        //renderJson(list);
        setAttr("list", list);
        setAttr("pageNum",UserService.pagInate(1,PAGE_SIZE).getPageNumber());
        setAttr("totalPage",UserService.pagInate(1,PAGE_SIZE).getTotalPage());
        render("page.jsp");
    }
    /**
     * 上一页
     */
     public void getBefore(){

            int pageNum=getParaToInt();
            List<User> list=UserService.pagInate(pageNum,PAGE_SIZE).getList();

            setAttr("list", list);
            //获取当前页号
            setAttr("pageNum",UserService.pagInate(pageNum,PAGE_SIZE).getPageNumber());
            //获取总页数
            setAttr("totalPage",UserService.pagInate(pageNum,PAGE_SIZE).getTotalPage());
            render("page.jsp");
         }
     /**
      * 下一页
      */
     public void getAfter(){

           int pageNum=getParaToInt();
            List<User> list=UserService.pagInate(pageNum,PAGE_SIZE).getList();

            setAttr("list", list);
            setAttr("pageNum",UserService.pagInate(pageNum,PAGE_SIZE).getPageNumber());
            setAttr("totalPage",UserService.pagInate(pageNum,PAGE_SIZE).getTotalPage());
            render("page.jsp");
         }
}

9.在view目录下新建page.jsp

<%@ page language="java" contentType="text/html; charset=UTF-8"
    pageEncoding="UTF-8"%>
<%@taglib prefix="c"  uri="http://java.sun.com/jsp/jstl/core"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>Insert title here</title>
</head>
<body>

        <table border="1px">
            <tr >
                <th>id</th>
                <th>姓名</th>
                <th>年龄</th>
                <th>密码</th>
            </tr>
            <c:forEach items="${list}" var="l">
            <tr >
                <td>${l.id}</td>
                <td>${l.name}</td>
                <td>${l.age}</td>
                <td>${l.password}</td>
            </tr>
            </c:forEach>
        </table>


        <div class="page">
            <a class="prev" href="#">上一页</a> <input type="text" id="pageNum"
                value="${pageNum}" readonly="true" /> <a class="next" href="#">下一页</a><input type="text" id="totalPage" value="${totalPage}"
                readonly="true"></div>



    <script type="text/javascript" src="${baseUrl}/js/jquery.js"></script>
    <script type="text/javascript">
        $(function() {
            /**
            点击上一页触发
            **/
            $(".prev").click(function() {
                    var pageNum = $("#pageNum").val();
                    pageNum = parseInt(pageNum);
                    if (pageNum == 1) {
                        alert("当前已经是第一页");
                    } else {
                        pageNum--;
window.location.href = "${baseUrl}/page/getBefore/"+ pageNum;
                        }
               });
            /**
        点击下一页触发
            **/
            $(".next").click(function() {
                    var pageNum = $("#pageNum").val();
                    var totalPage = $("#totalPage").val();
                    pageNum = parseInt(pageNum);
                    if (pageNum == totalPage) {
                        alert("当前已经是最后一页");
                    } else {
                        pageNum++;
  window.location.href = "${baseUrl}/page/getAfter/"+ pageNum;
                            }
                    });

        })
    </script>

</body>
</html>

10.运行DemoConfig.java类中main方法
这里写图片描述

11.页面有些难看,在这里不展示,可以自行调节样式

### 问题分析 用户在使用 `maven-shade-plugin` 打包 JFinal 5.2.2 应用时,运行生成的 JAR 文件出现 `java.lang.NullPointerException` 异常,并且堆栈信息指向 `PathKit.getWebRootPath()` 方法。这一异常通常与资源路径解析失败有关,尤其是在打包为 fat jar 后,原始文件系统路径可能无法被正确识别。 JFinal 的 `PathKit.getWebRootPath()` 方法依赖于运行时上下文来确定 Web 应用的根目录。当项目被打包为可执行 JAR 文件后,传统的文件系统路径访问方式不再适用,而是需要通过类路径(classpath)或特定机制访问资源。若未对资源路径进行适配处理,则可能导致 `getWebRootPath()` 返回 `null`,从而引发空指针异常[^1]。 --- ### 根本原因 1. **资源路径未正确打包** 在使用 `maven-shade-plugin` 进行打包时,若未将 `webapp` 目录内容正确嵌入 JAR 文件中,会导致运行时找不到资源路径。JFinal 默认期望从文件系统中加载 `webapp` 路径,但在 fat jar 中该路径应为内部资源路径。 2. **PathKit 行为不兼容 fat jar 环境** `PathKit.getWebRootPath()` 本质上是调用 `ClassUtils.getWebRootPath()`,其底层逻辑依赖于 ServletContext 或当前类路径下的相对路径。在普通 Web 容器中运行正常,但在独立 JAR 包中运行时会因找不到合适的上下文而返回 `null`。 3. **Manifest 配置不完整** 即使配置了主类,如果未显式指定 Class-Path 或资源过滤策略,也可能导致资源加载失败,进一步影响 `PathKit` 的初始化过程。 --- ### 解决方案 #### 1. 显式设置 WebRoot 路径 在应用启动前,手动设置 `PathKit` 的 `webRootPath`,避免依赖自动检测机制: ```java public class App { public static void main(String[] args) { // 手动设置 webroot 路径为 classpath 下的 webapp 目录 PathKit.setWebRootPath(App.class.getResource("/webapp").getFile()); JFinal.start("webapp", 80, "/", 5); } } ``` 此方法确保即使在 fat jar 中也能正确定位资源路径,避免 `NullPointerException` 的发生[^1]。 #### 2. 确保资源正确打包 在 `pom.xml` 中添加资源过滤配置,确保 `webapp` 目录随 JAR 文件一起打包: ```xml <resources> <resource> <directory>src/main/webapp</directory> <targetPath>webapp</targetPath> <filtering>false</filtering> </resource> <resource> <directory>src/main/resources</directory> <targetPath>.</targetPath> <filtering>false</filtering> </resource> </resources> ``` 这样可以保证 `webapp` 内容在构建过程中被复制到 JAR 文件的根目录下,供运行时访问[^4]。 #### 3. 使用 JettyServer 启动方式(推荐) JFinal 提供了更简洁的启动方式,使用 `JettyServer` 可以避免手动拼接路径的问题: ```java public class App { public static void main(String[] args) { JettyServer.create().start(); } } ``` 这种方式由框架内部处理资源路径问题,无需手动干预,适用于大多数基于内嵌 Jetty 的 JFinal 应用[^1]。 #### 4. 构建后验证 JAR 结构 使用以下命令解压 JAR 文件并检查结构是否正确: ```bash unzip target/your-app.jar -d jartest/ ``` 确认以下几点: - `META-INF/MANIFEST.MF` 中包含正确的 `Main-Class` - `webapp` 目录存在且内容完整 - 没有遗漏关键资源文件(如 `WEB-INF/web.xml`) --- ### 总结 在使用 `maven-shade-plugin` 打包 JFinal 项目时,由于资源路径处理不当,容易导致 `PathKit.getWebRootPath()` 抛出 `NullPointerException`。解决方法包括: - 手动设置 `PathKit.setWebRootPath()` - 正确配置 Maven 资源过滤策略,确保 `webapp` 目录被打包进 JAR - 推荐使用 `JettyServer` 启动方式以避免路径问题 - 构建后验证 JAR 文件结构和清单信息 通过上述调整,可以有效解决打包后运行异常的问题,提升应用的稳定性和可维护性。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值