资讯专栏INFORMATION COLUMN

前端数据校验从建模开始

韩冰 / 2321人阅读

摘要:为了能够把这部分代码更有条理我们把数据校验部分通过预先定义一个数据模型,把数据扔进去,返回校验结果。接下来我介绍一下这个工具,是一个数据建模及数据验证工具它可以非常方便的设计的表单数据结构,当然它不限于在表单使用。

前端开发过程中你们觉得处理什么业务功能最烦人?

做前端已经有很长一段时间了,不知道大家是否和我有同样的感受,在一些 Web 应用中表单处理起来比其他功能模块都麻烦,很多体力活,往往在数据的校验会很费时间。

为了能够把这部分代码更有条理,我们把数据校验部分通过 Schema 预先定义一个数据模型,把数据扔进去,返回校验结果。

接下来我介绍一下这个工具,schema-typed 是一个数据建模及数据验证工具, 它可以非常方便的设计的表单数据结构,当然它不限于在表单使用。如果你在产品中使用了 React , 那配合 React Suite 的表单组件简直就是如虎添翼。

安装
npm install schema-typed --save
示例
import { SchemaModel, StringType, DateType, NumberType } from "schema-typed";

const userModel = SchemaModel({
  username: StringType().isRequired("用户名不能为空"),
  email: StringType().isEmail("请输入正确的邮箱"),
  age: NumberType("年龄应该是一个数字").range(18, 30, "年龄应该在 18 到 30 岁之间")
});

const checkResult = userModel.check({
  username: "foobar",
  email: "foo@bar.com",
  age: 40
});

console.log(checkResult);

checkResult 返回结构是:

{
    username: { hasError: false },
    email: { hasError: false },
    age: { hasError: true, errorMessage: "年龄应该在 18 到 30 岁之间" }
}
多重验证
StringType()
  .minLength(6, "不能少于 6 个字符")
  .maxLength(30, "不能大于 30 个字符")
  .isRequired("该字段不能为空");
自定义验证

通过 addRule 函数自定义一个规则。

如果是对一个字符串类型的数据进行验证,可以通过 pattern 方法设置一个正则表达式进行自定义验证。

const myModel = SchemaModel({
  field1: StringType().addRule((value, data) => {
    return /^[1-9][0-9]{3}s?[a-zA-Z]{2}$/.test(value);
  }, "请输入合法字符"),
  field2: StringType().pattern(/^[1-9][0-9]{3}s?[a-zA-Z]{2}$/, "请输入合法字符")
});

schema.check({ field1: "", field2: "" });

/**
{
  field1: {
    hasError: true,
    errorMessage: "请输入合法字符"
  },
  field2: {
    hasError: true,
    errorMessage: "请输入合法字符"
  }
};
**/
自定义验证 - 多字段交叉验证

例如,验证两次输入密码是否一致

const schema = SchemaModel({
  password1: StringType().isRequired("该字段不能为空"),
  password2: StringType().addRule((value, data) => {
    if (value !== data.password1) {
      return false;
    }
    return true;
  }, "两次密码不一致")
});

schema.check({ password1: "123456", password2: "root" });

/**
{
  password1: { hasError: false },
  password2: {
    hasError: true,
    errorMessage: "两次密码不一致"
  }
};
**/
嵌套对象

对于复杂的嵌套的 Object , 可以使用 ObjectType().shape 方法进行定义,比如:

const model = SchemaModel({
  id: NumberType().isRequired("该字段不能为空"),
  name: StringType().isRequired("用户名不能为空"),
  info: ObjectType().shape({
    email: StringType().isEmail("应该是一个 email"),
    age: numberType().min(18, "年龄应该大于18岁")
  });
});

另外,更推荐把对象扁平化设计

import { flaser } from "object-flaser";

const model = SchemaModel({
  id: NumberType().isRequired("该字段不能为空"),
  name: StringType().isRequired("用户名不能为空"),
  "info.email": StringType().isEmail("应该是一个 email"),
  "info.age": numberType().min(18, "年龄应该大于18岁")
});

const user = flaser({
  id: 1,
  name: "schema-type",
  info: {
    email: "schema-type@gmail.com",
    age: 17
  }
});

model.check(data);
API

StringType

NumberType

ArrayType

DateType

ObjectType

BooleanType

StringType

isRequired()

StringType().isRequired("该字段不能为空");

isEmail(errorMessage: string)

StringType().isEmail("请输入正确的邮箱地址");

isURL(errorMessage: string)

StringType().isURL("请输入正确的URL地址");

isOneOf(items: Array, errorMessage: string)

StringType().isOneOf(["Javascript", "CSS"], "只能输入 `Javascript`和 `CSS`");

containsLetter(errorMessage: string)

StringType().containsLetter("必须包含英文字符");

containsUppercaseLetter(errorMessage: string)

StringType().containsUppercaseLetter("必须包含大写的英文字符");

containsLowercaseLetter(errorMessage: string)

StringType().containsLowercaseLetter("必须包含小写的英文字符");

containsLetterOnly(errorMessage: string)

StringType().containsLetterOnly("只能包含的英文字符");

containsNumber(errorMessage: string)

StringType().containsNumber("必须包含数字");

pattern(regExp: RegExp, errorMessage: string)

StringType().pattern(/^[1-9][0-9]{3}s?[a-zA-Z]{2}$/, "请输入合法字符");

rangeLength(minLength: number, maxLength: number, errorMessage: string)

StringType().rangeLength(6, 30, "字符个数只能在 6 - 30 之间");

minLength(minLength: number, errorMessage: string)

StringType().minLength(6, "最小需要6个字符");

maxLength(maxLength: number, errorMessage: string)

StringType().minLength(30, "最大只能30个字符");

addRule(onValid: Function, errorMessage: string)

StringType().addRule((value, data) => {
  return /^[1-9][0-9]{3}s?[a-zA-Z]{2}$/.test(value);
}, "请输入合法字符");
NumberType

isRequired()

NumberType().isRequired("该字段必填");

isInteger(errorMessage: string)

NumberType().isInteger("只能是整型");

isOneOf(items: Array, errorMessage: string)

NumberType().isOneOf([5, 10, 15], "只能是`5`,`10`,`15`");

pattern(regExp: RegExp, errorMessage: string)

NumberType().pattern(/^[1-9][0-9]{3}$/, "请输入合法字符");

range(minLength: number, maxLength: number, errorMessage: string)

NumberType().range(18, 40, "请输入 18 - 40 之间的数字");

min(min: number, errorMessage: string)

NumberType().min(18, "最小值 18");

max(max: number, errorMessage: string)

NumberType().max(40, "最大值 40");

addRule(onValid: Function, errorMessage: string)

NumberType().addRule((value, data) => {
  return value % 5 === 0;
}, "请输入有效的数字");
ArrayType

isRequired()

ArrayType().isRequired("该字段必填");

rangeLength(minLength: number, maxLength: number, errorMessage: string)

ArrayType().rangeLength(1, 3, "至少选择1个,但不能超过3个");

minLength(minLength: number, errorMessage: string)

ArrayType().minLength(1, "至少选择1个");

maxLength(maxLength: number, errorMessage: string)

ArrayType().maxLength(3, "不能超过3个");

unrepeatable(errorMessage: string)

ArrayType().unrepeatable("不能出现重复选项");

of(type: Object, errorMessage: string)

ArrayType().of(StringType().isEmail(), "格式错误");

addRule(onValid: Function, errorMessage: string)

ArrayType().addRule((value, data) => {
  return value.length % 2 === 0;
}, "好事成双");
DateType

isRequired()

DateType().isRequired("日期不能为空");

range(min: Date, max: Date, errorMessage: string)

DateType().range(
  new Date("08/01/2017"),
  new Date("08/30/2017"),
  "时间应该在 08/01/2017 - 08/30/2017 之间"
);

min(min: Date, errorMessage: string)

DateType().min(new Date("08/01/2017"), "时间的最小值 08/01/2017");

max(max: Date, errorMessage: string)

DateType().max(new Date("08/30/2017"), "时间的最大值 08/30/2017");

addRule(onValid: Function, errorMessage: string)

DateType().addRule((value, data) => {
  return value.getDay() === 2;
}, "只能选择周二");
ObjectType

isRequired()

ObjectType().isRequired("该对象不能为空");

shape(type: Object)

ObjectType().shape({
  email: StringType().isEmail("应该是一个 email"),
  age: numberType().min(18, "年龄应该大于18岁")
});

addRule(onValid: Function, errorMessage: string)

ObjectType().addRule((value, data) => {
  if (value.id || value.email) {
    return true;
  }
  return false;
}, "id 与 email 必须有一个不能为空");
BooleanType

isRequired()

BooleanType().isRequired("该字段不能为空");

addRule(onValid: Function, errorMessage: string)

ObjectType().addRule((value, data) => {
  if (typeof value === "undefined" && A === 10) {
    return false;
  }
  return true;
}, "当 A 等于 10 的时候,该值必须为空");

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/97242.html

相关文章

  • 0开始构建一个属于你自己的PHP框架

    摘要:如何构建一个自己的框架为什么我们要去构建一个自己的框架可能绝大多数的人都会说市面上已经那么多的框架了,还造什么轮子。 showImg(https://segmentfault.com/img/bVNg9F?w=500&h=500); 如何构建一个自己的PHP框架 为什么我们要去构建一个自己的PHP框架?可能绝大多数的人都会说市面上已经那么多的框架了,还造什么轮子?。我的观点造轮子不是目...

    vpants 评论0 收藏0
  • 数据挖掘的TO-DO-LIST

    摘要:数据挖掘的流程与方法任务关联分析聚类分析分类分析异常分析特异组群分析演变分析方法统计在线处理分析情报检索机器学习分类实际应用应用分类趋势预测推荐关联类商品回归分析实际应用预测销售趋势聚类实际应用分类关联规则包括两个阶段从海量数据中找到高频项 数据挖掘的流程与方法 1.任务: 关联分析 聚类分析 分类分析 异常分析 特异组群分析 演变分析 2.方法: 统计 在线处理分析 情报检索 ...

    wangxinarhat 评论0 收藏0

发表评论

0条评论

最新活动
阅读需要支付1元查看
<