后端,为什么要进行校验?

1.“后端进行参数校验,是防止别人通过接口乱刷服务”。反正我是信第二种说法的,如果一些不安好心的人,通过接口刷我们的服务,随便哪个参数我们都允许填入,会导致数据库中导致大量的脏数据、风险。如果注入的是病毒怎么办?所以,后端是要进行参数校验的。

怎么进行后端参数校验?

我目前也是了解到了两种做法。

1、通过注解的方法,在实体上加注解。通过工具,当客户端传入参数的时候,自动进行校验。

大概的方法:pom中引入工具类;然后实体中添加注解;controller层传入数据时添加@validator进行参数校验。

2、通过工具类,进行参数校验。

思路:将常用的校验方法放在一个类中,然后哪个参数需要校验,就调用工具类中的方法。我自认为很好用。

下面提供一个工具类,正则表达式工具类:RegexUtils.java

下面就第二种方式进行个简单的举例说明:



import java.util.regex.Matcher;
import java.util.regex.Pattern;

/**
 * @author puxiaozhe
 * @Date 2010-02-14 
 */
public class RegexUtils {

        /**
         * 验证Email
         * @param email email地址,格式:zhangsan@zuidaima.com,zhangsan@xxx.com.cn,xxx代表邮件服务商
         * @return 验证成功返回true,验证失败返回false
         */
        public static boolean checkEmail(String email) {
            String regex = "\\w+@\\w+\\.[a-z]+(\\.[a-z]+)?";
            return Pattern.matches(regex, email);
        }

        /**
         * 验证身份证号码
         * @param idCard 居民身份证号码15位或18位,最后一位可能是数字或字母
         * @return 验证成功返回true,验证失败返回false
         */
        public static boolean checkIdCard(String idCard) {
            String regex = "[1-9]\\d{13,16}[a-zA-Z0-9]{1}";
            return Pattern.matches(regex,idCard);
        }

        /**
         * 验证手机号码(支持国际格式,+86135xxxx...(中国内地),+00852137xxxx...(中国香港))
         * @param mobile 移动、联通、电信运营商的号码段
         *<p>移动的号段:134(0-8)、135、136、137、138、139、147(预计用于TD上网卡)
         *、150、151、152、157(TD专用)、158、159、187(未启用)、188(TD专用)</p>
         *<p>联通的号段:130、131、132、155、156(世界风专用)、185(未启用)、186(3g)</p>
         *<p>电信的号段:133、153、180(未启用)、189</p>
         * @return 验证成功返回true,验证失败返回false
         */
        public static boolean checkMobile(String mobile) {
            String regex = "(\\+\\d+)?1[34578]\\d{9}$";
            return Pattern.matches(regex,mobile);
        }

        /**
         * 验证固定电话号码
         * @param phone 电话号码,格式:国家(地区)电话代码 + 区号(城市代码) + 电话号码,如:+8602085588447
         * <p><b>国家(地区) 代码 :</b>标识电话号码的国家(地区)的标准国家(地区)代码。它包含从 0 到 9 的一位或多位数字,
         *  数字之后是空格分隔的国家(地区)代码。</p>
         * <p><b>区号(城市代码):</b>这可能包含一个或多个从 0 到 9 的数字,地区或城市代码放在圆括号——
         * 对不使用地区或城市代码的国家(地区),则省略该组件。</p>
         * <p><b>电话号码:</b>这包含从 0 到 9 的一个或多个数字 </p>
         * @return 验证成功返回true,验证失败返回false
         */
        public static boolean checkPhone(String phone) {
            String regex = "(\\+\\d+)?(\\d{3,4}\\-?)?\\d{7,8}$";
            return Pattern.matches(regex, phone);
        }

        /**
         * 验证整数(正整数和负整数)
         * @param digit 一位或多位0-9之间的整数
         * @return 验证成功返回true,验证失败返回false
         */
        public static boolean checkDigit(String digit) {
            String regex = "\\-?[1-9]\\d+";
            return Pattern.matches(regex,digit);
        }

        /**
         * 验证整数和浮点数(正负整数和正负浮点数)
         * @param decimals 一位或多位0-9之间的浮点数,如:1.23,233.30
         * @return 验证成功返回true,验证失败返回false
         */
        public static boolean checkDecimals(String decimals) {
            String regex = "\\-?[1-9]\\d+(\\.\\d+)?";
            return Pattern.matches(regex,decimals);
        }

        /**
         * 验证空白字符
         * @param blankSpace 空白字符,包括:空格、\t、\n、\r、\f、\x0B
         * @return 验证成功返回true,验证失败返回false
         */
        public static boolean checkBlankSpace(String blankSpace) {
            String regex = "\\s+";
            return Pattern.matches(regex,blankSpace);
        }

        /**
         * 验证中文
         * @param chinese 中文字符
         * @return 验证成功返回true,验证失败返回false
         */
        public static boolean checkChinese(String chinese) {
            String regex = "^[\u4E00-\u9FA5]+$";
            return Pattern.matches(regex,chinese);
        }

        /**
         * 验证日期(年月日)
         * @param birthday 日期,格式:1992-09-03,或1992.09.03
         * @return 验证成功返回true,验证失败返回false
         */
        public static boolean checkBirthday(String birthday) {
            String regex = "[1-9]{4}([-./])\\d{1,2}\\1\\d{1,2}";
            return Pattern.matches(regex,birthday);
        }

        /**
         * 验证URL地址         * @param url 格式:http://blog.***.net:80/xyang81/article/details/7705960? 或 http://www.***.net:80
         * @return 验证成功返回true,验证失败返回false
         */
        public static boolean checkURL(String url) {
            String regex = "(https?://(w{3}\\.)?)?\\w+\\.\\w+(\\.[a-zA-Z]+)*(:\\d{1,5})?(/\\w*)*(\\??(.+=.*)?(&.+=.*)?)?";
            return Pattern.matches(regex, url);
        }

        /**
         * <pre>
         * 获取网址 URL 的一级域
         * </pre>
         *
         * @param url
         * @return
         */
        public static String getDomain(String url) {
            Pattern p = Pattern.compile("(?<=http://|\\.)[^.]*?\\.(com|cn|net|org|biz|info|cc|tv)", Pattern.CASE_INSENSITIVE);
            // 获取完整的域名
            // Pattern p=Pattern.compile("[^//]*?\\.(com|cn|net|org|biz|info|cc|tv)", Pattern.CASE_INSENSITIVE);
            Matcher matcher = p.matcher(url);
            matcher.find();
            return matcher.group();
        }
        /**
         * 匹配中国邮政编码
         * @param postcode 邮政编码
         * @return 验证成功返回true,验证失败返回false
         */
        public static boolean checkPostcode(String postcode) {
            String regex = "[1-9]\\d{5}";
            return Pattern.matches(regex, postcode);
        }

        /**
         * 匹配IP地址(简单匹配,格式,如:192.168.1.1,127.0.0.1,没有匹配IP段的大小)
         * @param ipAddress IPv4标准地址
         * @return 验证成功返回true,验证失败返回false
         */
        public static boolean checkIpAddress(String ipAddress) {
            String regex = "[1-9](\\d{1,2})?\\.(0|([1-9](\\d{1,2})?))\\.(0|([1-9](\\d{1,2})?))\\.(0|([1-9](\\d{1,2})?))";
            return Pattern.matches(regex, ipAddress);
        }


}

然后直接在controller层进行调用的。!RegexUtils.checkEmail(obj.getContactsEmail())

@RequestMapping(value = "/submit", method = RequestMethod.POST)
    @ResponseBody
    public BaseResponse submitTenantInfo(@RequestBody WebObjRequest<TenantInfo> request) {
        TenantInfo obj = request.getObj();
        if (obj == null || !request.isCorrectParams()|| !RegexUtils.checkEmail(obj.getContactsEmail())) {
            return new BaseResponse(RespConstants.GLOBAL_PARAM_ERROR);
        }
       ```````
        return new BaseResponse(RespConstants.GLOBAL_SUCCESS);
    }

 

最后,我想说的是。客户端校验和服务端校验的不同:我的理解是,客户端校验的提示,是提示用户正确输入,密码的输入等准确性来加强用户体验。服务端校验是为了接口安全,防止非法用户乱刷。所以,服务端进行参数校验的时候,返回的错误码不要具体到具体的错误,返回“参数异常”等相关异常错误即可。