JWeb框架快速入门

写了个框架。欢迎大家指点、提意见、讨论、反馈bug.
前言:
1.JWeb框架使用servlet3.0+版本以上。无需配置web.xml文件即可自动注册拦截器。
2.**支持原生**Servlet3.0+JWeb强大的过滤、校验相结合。
3.不差struts2分毫的拦截器(我们还是亲切点,叫回过滤器),却比其更简单!更灵活!!更高效!!更易维护!!!
4.基本无多少学习代价。设计之初便考虑到最初的Servlet习惯!但我们做到了简单使用,不简单功能实现!!

一个简单的而强大的框架,往往设计上会复杂得多,并耗性能。但我们做到了简单、高速、易用相结合。这得益于作者的耐心与数学功底。
不只是代码,连同框架设计流程图也一起奉献。以顺让用户完全把握框架,从而从多而杂的开源框架中,拥有自己的框架,离开框架的变动、出现bug无从下手……等等问题脱离出来,如同请了位架构师,在为您公司无偿加班了一段时间

本框架最大的好处,
一是基于高效的数据模型
二是基于强大的过滤器(完全可以定制过各种同用视图!);
三支持扫描式配置(比如/xx/xx开关的请求路径统一要经过xx过滤器/过滤组。是不是很自然地想到权限的管理呢?),
四是没有封装过度!!不会出现隔年无人会用的尴尬!而且封装过度,还消耗性能,加大维护难度,二次开发及后期人员变动,投入的预备阶段的成本!!最关键的是,没有封装过度,还非常好用、简单、可以轻松完全复杂业务!


一、hello!
建一个类,加入@H与@M标注即可进行访问!通过@H+@M的值进行访问。

@H("/say")
public class HI {
    @M("/yousay")// 通过 /say/yousay访问。
    public static void selectOne(JWeb jw) {
        jw.printOne("hello!!!!!");
    }
}

二、获得用户传参
1. 直接获取。
2. 通过对象,自动装箱。
对象的属性名,就是用户传过来的参数名;属性值,就是用户传过来的参数值。
例子:

package wx.web;

import java.text.SimpleDateFormat;
import system.web.JWeb;
import system.web.hm.annotation.H;
import system.web.hm.annotation.M;
import system.web.validate.annotation.Validate;

/**
 *
 * @author wangchunzi
 */
@H("/login")
public class UserLogin {

    @M("/user")//
    @Validate(UserLoginValidate.class)
    public static void selectOne(JWeb jw) {
        //直接取参数
//        String account = jw.getString("account");
//        String passord = jw.getString("passord");
        //注意,取日期、时间的格式,如果没有配置,默认采用框架的全局配置格式。
        //置于是用日期格式来转换还是用时间格式来转换,框架会对比值与格式的长度决定 。
//         jw.getDate("mydate");//取mydate字段的时间
//        jw.getDate("mydate", "yyyy-MM-dd"); 指定格式进行取日期
//------------------------当然,如果参数多,我们通过对象式取参(根据传入的类型,自动装箱)---------------------------------------------------------        
        //对象式取参
        UserVO vo = jw.getObject(UserVO.class);
        //此对象取时间时,因为我们没有传时间格式给取参对象的方法,所以,它默认取全局配置的时间配置。
        jw.request.setAttribute("ok_return", "你发送的数据:"
                + vo.account
                + "//" + vo.passord
                + "//" + new SimpleDateFormat("yyyy/MM/dd").format(vo.mydate)
        );

        jw.forward("/index.jsp");
    }
}

[/code]
如果要演示对象传参,还需要建一个vo对象:
[code=java]
package wx.web;

import java.util.Date;

public class UserVO {

public String account;//   或者私有化访问域也能取取值的如 private String account;
public String passord;
public Date mydate;

}
[/code]
三、校验用户传参:ValidateModel
第一步:
定义一个类,然后继承ValidateModel抽象类。
实现其中的抽象方法即可。
public abstract void iniValidate();
public abstract ValidateResultModel recheck(JWeb jw, Map

package configuration.file;

public class ImgFileModel extends system.web.file.FileModel {

    @Override
    public void configuration(system.web.file.temp.FileConfig fc) {
        //web根目录的绝对路径.直接采用全局配置实例中,取得即可。
        fc.path_web_real = system.web.WebContext.getWebContext().WEB_PATH;
        fc.path_save = "upload/img/";            //上传的目录
        fc.fileNameSuffix_alloy = "gif,jpg,jpeg,bmp,png";//图片类型后缀
        fc.fileSize_max = 2 * 1024 * 1024;      //文件大小
        //文件缓冲,及写入硬盘临时文件的临界值
        //如果采用Servlet原生支持,两处fileSizeThreshold的值都必须一样!
        fc.fileSizeThreshold = 2 * 1024 * 1024; 
        //等其他参数。
    }
}

2使用Servlet原生的上传。
跟上例差不多。区别就是,下面直接使用原生的Servlet(注意,下例子的 extends Servlet,实际就是,就是扩展HttpServlet.框架提供Servlet类封装,是方便使用JWeb对象。)
另外要注意的就是,一定要配置fileSizeThreshold的值。同时,还要跟配置ImgFileModel类的fileSizeThreshold一样!!

package wx.web.filemanager;

import configuration.file.ImgFileModel;
import java.io.IOException;
import javax.servlet.ServletException;
import javax.servlet.annotation.MultipartConfig;
import javax.servlet.annotation.WebServlet;
import system.web.JWeb;
import system.web.file.engine.PartEngine;
import system.web.servlet.Servlet;

@WebServlet(name = "UpImgFile", urlPatterns = {"/UpImgFile"})
@MultipartConfig(location = "", fileSizeThreshold = 2 * 1024 * 1024)
public class UpImgFile extends Servlet {

    @Override
    protected void doService(JWeb jw) throws ServletException, IOException {
        PartEngine pe = new PartEngine(jw.request);
        pe.upOne(ImgFileModel.class);
    }
}

五、定义过滤器
(1)、通过继承FilterModel,来自定义过滤器
如下。过滤默认放在顶层。如果不执行super.setLocation(LOCATION_BUTTOM);修改的话

package configuration.filter.buttom;

import system.web.JWeb;
import system.web.filter.chain.FilterModel;

public class DemoFilter extends FilterModel{
    {
        //表示此过滤器放在
        //顶层LOCATION_TOP,或中层LOCATION_CENTER,或底层LOCATION_BUTTOM
        //执行
        super.setLocation(LOCATION_BUTTOM);//底层。执行完@HM服务类或Servlet后执行的过滤器
    }
    @Override
    public void doFilter(JWeb jw) {
        //具体实现
    }

}

我们看一下通过super.setLocation能配置的空间位置 一共三个
这里写图片描述

(2)、通过JWFilter使用我们定义的过滤器
如下例子:通过@JWFilter(我们定义的过滤类),么,所有此类的方法请求,都会被加入这个过器。

@H("/login")
@JWFilter(DemoFilter.class)//看我,在类处。则所有此类方法下,都会使用我的过滤
public class UserLogin {

当然,可以直接放在方法那里,这样只作用在此方法。

public class UserLogin {

    @M("/user")//
    @Validate(UserLoginValidate.class)
    @JWFilter(DemoFilter.class)//看我,在方法处
    public static void selectOne(JWeb jw) {

(3)、过滤器是单例模式的。如图说明
这里写图片描述
(4)、创建过滤组
为什么要创建过滤组?
比如要多个过滤器共同完成一个功能。
比如一个类、方法要配置多个不同功能的过滤器。

我们通过继承FilterModels来创建一个过滤组。如下代码。
然后加入我们其他的过滤器,或过滤组。

package configuration.filter;

import configuration.filter.buttom.DemoFilter;
import java.util.Set;
import system.web.filter.chain.FilterModels;

/**
 *xx功能过滤组
 * @author wangchunzi
 */
public class MyFilters extends FilterModels {

    @Override
    public void put(Set<Class> modelClass) {
        modelClass.add(DemoFilter.class);
        modelClass.add(null);//或其他过滤组的类
        //可以继续加入...
    }
}

(5)、通过JWFilter使用我们定义的过滤器组
用法与过滤器一样!可放在类处,亦可放在方法处。都是同个@JWFilter
(6)、实战过滤之定义文件下载视图!
先看成果。是的!只要把文件相对路径放入request属性中。即可实例下载了!!!为什么可以这样方便?
主要是因为我们用了自己定义的低层过滤器(在@H@M服务类-方法或Servlet后执行的过滤器) BigFileDownload.class!!

package wx.web.filemanager;

import configuration.filter.FilterKey;
import configuration.filter.buttom.BigFileDownload;
import system.web.JWeb;
import system.web.filter.annotation.JWFilter;
import system.web.hm.annotation.H;
import system.web.hm.annotation.M;
@H("/filedow")
public class Dow {

    @M("/big/myfile")
    @JWFilter(BigFileDownload.class)
    public static void doww(JWeb jw) {
        //真实项目中,我们都是从数据库取出文件的相对路径或由用户提交要下载的文件名
        //假设我们要下载的文件放在项目upload/file.如图所示,我们就能输出文件流给用户了!
        //当然,也可能放在其他直接url访问不到的地方。这时,就需要我们流来处理了。而不是给用户一个可访问的文件url
        //具体原理请看操作手册。这得益于我们强大而灵活的拦截器,就这样可以下载文件了!!
        jw.request.setAttribute(FilterKey.file.code, "upload/file/123.txt");
    }
}

现在,我们所有类似的下载,都可以使用此过滤器了。
再看看我们自己这个过滤器的实例:

package configuration.filter.buttom;

import configuration.filter.FilterKey;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
import system.web.JWeb;
import system.web.WebContext;
import system.web.filter.chain.FilterModel;

/**
 *
 * @author wangchunzi
 */
public class BigFileDownload extends FilterModel {

    {
        super.setLocation(LOCATION_BUTTOM);
    }

    public static final int buff = 1024 * 1024 * 3;

    @Override
    public void doFilter(JWeb jw) {
        Object attribute = jw.request.getAttribute(FilterKey.file.code);
        if (null != attribute) {
            String str = attribute.toString();
            jw.request.removeAttribute("file");

            String filename = str.substring(str.lastIndexOf("/")+1);
            FileInputStream fis = null;
            try {
                fis = new FileInputStream(WebContext.getWebContext().WEB_PATH + str);
            } catch (FileNotFoundException ex) {
                jw.printOne("你访问的资源已经失效!");
                return;
            }
            // 设置输出的格式
            jw.response.reset();
            jw.response.addHeader("Content-Disposition", "attachment; filename=\"" + filename + "\"");
            try {
                jw.response.setContentLength(fis.available());
            } catch (IOException ex) {
                jw.printOne("获得文件大小成错!");
                return;
            }
            // 循环取出流中的数据
            byte[] b = new byte[buff];
            int len;
            try {
                while ((len = fis.read(b)) > 0) {
                    jw.response.getOutputStream().write(b, 0, len);
                }
                jw.response.flushBuffer();
                jw.response.getOutputStream().close();
                fis.close();
            } catch (IOException e) {
            }
        }
    }

}

当然,我们可以发挥我们的想象力,定义统一的,与ui 框架统一的视图。比如说json… 常见有 grid 、tree等数据。

(6)、使用扫描配置过滤器、过滤组!
之有我们都是使用@JWFilter来配置使用我们的过滤器、过滤组。现在,我们可以通过扫描来配置!

首先,我们要继承一个ConfigurationFilter类。

package configuration.filter;

import system.web.filter.chain.config.ConfigurationFilter;
import system.web.filter.chain.config.LinkFilters;

public class MyConfigurationFilter extends ConfigurationFilter {

    @Override
    public void configuration(LinkFilters lf) {

//        //第一个参数[test.action.*]表示以此开头的(类+方法)路径,将绑定第二个参数所有的过滤器
//        lf.addFiltersByMethod("test.action.*", new Class[]{ 
//            //test.configuration.filter.f1.TwoFilter.class, Filters1.class, Filters8.class
//        });
//        第一个参数/x/ss表示以此开头的请求路径,将绑定第二个参数所有的过滤器
        lf.addFiltersByURL("/filedow/big/*", configuration.filter.buttom.BigFileDownload.class);
    }

}

再看先前的一例子,通过扫描配置,我们已经可以去除
@JWFilter(configuration.filter.buttom.BigFileDownload.class)的手工绑定了!!

@H("/filedow")
public class Dow {
    @M("/big/myfile")
    public static void doww(JWeb jw) {
            jw.request.setAttribute(FilterKey.file.code, "upload/file/123.txt");
    }
}

六、全局配置:WebConfigModel
下面我们来自定义配置。如,我们要修改HM_SUFFIX值。如下,表示我们只管理带.jw后缀的请求路径。
此种模式下,效率相对高些。

package configuration;

public class WebConfigModel extends system.web.config.temp.WebConfigModel {

    @Override
    public void config(system.web.config.temp.WebConfig config) {
        config.HM_SUFFIX = "*.jw";//表示拦截url后缀为jw的请求
    }

}

另外,我们发现还有时间、日期的配置。
这两个参数主要是在获得用户传参数时(有日期、或时间),如果没有指定日期、时间的格式,系统将自动采用默认的的公共配置来转化,如将 1988-09-01转化成日期。

后:
因为现在脚本、sql注入拦截,已经成了服务器安全软件的基本功,甚至一些jdbc封装的jar包又进行一次防注入过滤,
我还在想要不要在框架再实现一次,最后想想,还是留下用户自己选择吧。
当然,如果你们觉得还是很有必要,我会在后期加入的。
有问题的读者,请联系我pankeng1988w@163.com

源码地址
https://git.oschina.net/it_weixin/JWeb

源码地址空间的附件 中有框架设计图及演示项目。

视频教学
http://pan.baidu.com/s/1gf61Xh5

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值