实现带参数、返回值类型声明的js函数:

类型定义:
window.Str = Type.Str = Type.define('STRING', Type.isStr);
var Per = Type.define('PERSON', function(p){
    return p && p.type === 'person' && p.name;
});

定义函数:
var addStr = Str(function(a, b){
  return a + b;
}, Str, Str);
运行:
addStr('1', 1);
则报错:
TypeError: Unexpected argument, expecting STRING (arg[1])

定义函数:
var addToStr = Str(function(a, b){
  return a + b;
});
运行:
addToStr(1, 1);
则报错:
TypeError: Unexpected result, expecting STRING
1. [代码]define Type

/*
 * The main class: Type
 */
 
var Type = function(){
    this.author = {
        name: 'nighca',
        email: 'nighca@live.cn'
    };
};
2. [代码]type mehods

/*
 * func define/destroy to define/destroy a type
 */
 
Type.__defined__ = {};
 
Type.define = function(name, check, override){
    var Type = this;
    if(!Type.isStr(name) || !Type.isFunc(check)){
        throw new TypeError('Param error');
    }else if(!override && this.__defined__[name]){
        throw new Error('Type ' + name + ' already exists');
    }else{
        var funcCreator = function(func){
            var argumentTypes = Array.prototype.slice.call(arguments, 1);
            var wrapper = function(){
                for (var i = 0, l = argumentTypes.length; i < l; i++) {
                    if(argumentTypes[i] && !argumentTypes[i].check(arguments[i])){
                        throw new TypeError(
                            'Unexpected argument, expecting ' +
                            argumentTypes[i].typeName +
                            ' (arg[' + i + '])'
                        );
                    }
                };
                var ret = func.apply(this, arguments);
                if(!check(ret)){
                    throw new TypeError('Unexpected result, expecting ' + name);
                }
                return ret;
            };
            wrapper.toString = function(){
                return func.toString();
            };
            return wrapper;
        };
        funcCreator.check = check;
        funcCreator.typeName = name;
        funcCreator.constructor = Type;
        funcCreator.__proto__ = Type.prototype;
 
        return this.__defined__[name] = funcCreator;
    }
};
 
Type.isType = function(t){
    return t && t.constructor === Type;
}
 
Type.mix = function(name){
    var types, type;
    if(!Type.isStr(name)){
        types = Array.prototype.slice.call(arguments, 0);
        name = null;
    }else{
        types = Array.prototype.slice.call(arguments, 1);
    }
 
    if(types.length < 2){
        throw new Error('Params error');
    }
    var i, l;
    for (i = 0, l = types.length; i < l; i++) {
        if(!Type.isType(types[i])){
            throw new TypeError('Param not a type');
        }
    }
 
    if(!name){
        name = 'MIX_OF_' + types[0].typeName;
        for (i = 1, l = types.length; i < l; i++) {
            name += '_' + types[i].typeName;
        }
        name += '@' + Date.now();
    }
 
    var checker = function(obj){
        for (var i = 0, l = types.length; i < l; i++) {
            if(!types[i].check(obj)){
                return false;
            }
        }
        return true;
    };
 
    return Type.define(name, checker);
};http://www.huiyi8.com/jianbihua/
简笔画大全 
Type.any = function(name){
    var types, type;
    if(!Type.isStr(name)){
        types = Array.prototype.slice.call(arguments, 0);
        name = null;
    }else{
        types = Array.prototype.slice.call(arguments, 1);
    }
 
    if(types.length < 2){
        throw new Error('Params error');
    }
    var i, l;
    for (i = 0, l = types.length; i < l; i++) {
        if(!Type.isType(types[i])){
            throw new TypeError('Param not a type');
        }
    }
 
    if(!name){
        name = 'ANY_OF_' + types[0].typeName;
        for (i = 1, l = types.length; i < l; i++) {
            name += '_' + types[i].typeName;
        }
        name += '@' + Date.now();
    }
 
    var checker = function(obj){
        for (var i = 0, l = types.length; i < l; i++) {
            if(types[i].check(obj)){
                return true;
            }
        }
        return false;
    };
 
    return Type.define(name, checker);
};
 
Type.isDefined = function(name){
    return !!this.__defined__[name];
};
 
Type.destroy = function(name){
    if(!Type.isStr(name)){
        throw new TypeError('Param error');
    }
 
    var type = this.__defined__[name];
    if(type){
        type = null;
        delete this.__defined__[name];
    }
};
 
Type.prototype.suicide = function(){
    this.constructor.destroy(this.typeName);
};​
3. [代码]some help funcs

/*
 * Some funcs to judge obj type
 */
 
Type.toString = function(){
    return JSON ? JSON.stringify(new this()) : '>_<';
};
 
Type.format = function(obj){
    return Object.prototype.toString.call(obj);
};
 
Type.isNum = function(obj){
    return typeof obj === 'number' || Type.format(obj) === '[object Number]';
};
 
Type.isStr = function(obj){
    return typeof obj === 'string' || Type.format(obj) === '[object String]';
};
 
Type.isFunc = function(obj){
    return typeof obj === 'function' || Type.format(obj) === '[object Function]';
};
 
Type.isBool = function(obj){
    return typeof obj === 'boolean' || Type.format(obj) === '[object Boolean]';
};
 
Type.isArr = function(obj){
    return Type.format(obj) === '[object Array]';
};
 
Type.isReg = function(obj){
    return Type.format(obj) === '[object RegExp]';
};
 
Type.isObj = function(obj){
    return Type.format(obj) === '[object Object]';
};
 
Type.isUndef = function(obj){
    return typeof obj === 'undefined';
};
 
Type.isNul = function(obj){
    return Type.format(obj) === '[object Null]';
};
 
Type.isVoid = function(obj){
    return true;
};
4. [代码]Some default types    
/*
 * Some default types binded to window & Type
 */
 
window.Num = Type.Num = Type.define('NUMBER', Type.isNum);
 
window.Str = Type.Str = Type.define('STRING', Type.isStr);
 
window.Func = Type.Func = Type.define('FUNCTION', Type.isFunc);
 
window.Bool = Type.Bool = Type.define('BOOLEAN', Type.isBool);
 
window.Arr = Type.Arr = Type.define('ARRAY', Type.isArr);
 
window.Reg = Type.Reg = Type.define('REGEXP', Type.isReg);
 
window.Obj = Type.Obj = Type.define('OBJECT', Type.isObj);
 
window.Undef = Type.Undef = Type.define('UNDEFINED', Type.isUndef);
 
window.Nul = Type.Nul = Type.define('NULL', Type.isNul);
 
window.Void = Type.Void = Type.define('VOID', Type.isVoid);

javascript函数参数、返回值类型检查的更多相关文章

  1. JAVA函数的返回值类型详解以及生成随机数的例题

    函数的四要素:函数名.输入.输出(返回).加工. 函数分为两种:一种是有返回值得函数,一种是没有返回值的函数. 1. 定义:没有返回值的函数:(当我不需要函数的计算结果再拿出来进行运算的时候,我就不需 ...

  2. Swift 定义函数 参数 返回值

    定义多参数函数 - 用func声明函数  func name(parameters) -> return type { function body } func halfOpenRangeLen ...

  3. Swift2.0语言教程之函数的返回值与函数类型

    Swift2.0语言教程之函数的返回值与函数类型 Swift2.0中函数的返回值 根据是否具有返回值,函数可以分为无返回值函数和有返回值函数.以下将会对这两种函数类型进行讲解. Swift2.0中具有 ...

  4. MATLAB importdata函数返回值类型

    importdata函数是MATLAB中I/O文件操作的一个重要函数.需要注意的是,针对不同的文件内容,importdata函数的返回值类型也有所不同. MATLAB帮助文档中的详细说明如下: Bas ...

  5. 请为main函数提供返回值

    很多人甚至市面上的一些书籍,都使用了void main( ) ,其实这是错误的.C/C++ 中从来没有定义过void main( ).C++ 之父 Bjarne Stroustrup 在他的主页上的 ...

  6. php7函数,声明,返回值等新特性介绍

    使用 ... 运算符定义变长参数函数 (PHP 5 >= 5.6.0, PHP 7) 现在可以不依赖 func_get_args(), 使用 ... 运算符 来实现 变长参数函数. functi ...

  7. c++ 模板类,方法返回值类型是typedef出来的,或者是auto,那么此方法在类外面如何定义?

    c++ 模板类,方法返回值类型是typedef出来的,或者是auto,那么此方法在类外面如何定义? 比如方法max1的返回值是用typedef定义出来的mint,那么在类外如何定义这个方法呢? tem ...

  8. javascript . 03 函数定义、函数参数(形参、实参)、函数的返回值、冒泡函数、函数的加载、局部变量与全局变量、隐式全局变量、JS预解析、是否是质数、斐波那契数列

    1.1 知识点 函数:就是可以重复执行的代码块 2.  组成:参数,功能,返回值 为什么要用函数,因为一部分代码使用次数会很多,所以封装起来, 需要的时候调用 函数不调用,自己不会执行 同名函数会覆盖 ...

  9. c++特性:指向类成员的指针和非类型类模板参数和函数指针返回值 参数推导机制和关联型别

    一.c++允许定义指向类成员的指针,包括类函数成员指针和类数据成员指针 格式如下: class A { public: void func(){printf("This is a funct ...

随机推荐

  1. Java.lang.NoSuchMethodError: 后带 V/Z等字母的

    知道 Java.lang.NoSuchMethodError: 后带 V/Z等字母的 错误,一般都是 jar包冲突引起的,找到冲突的jar包,去掉一个就好

  2. HDU 5076 Memory

    Memory Time Limit: 4000ms Memory Limit: 262144KB This problem will be judged on HDU. Original ID: 50 ...

  3. var声明的成员变量和函数内声明的变量区别

    1.函数内部,有var声明的是局部变量,没var的,声明的全局变量. 2.在全局作用域内声明变量时,有var 和没var声明的都是全局变量,是window的属性.通过变量var声明全局对象的属性无法通 ...

  4. 调用BOS服务保存一个单据的简化示例

    IMetaDataService metadataService = ServiceHelper.GetService<IMetaDataService>(); // 加载元数据 Form ...

  5. 2016 Multi-University Training Contest 2 solutions BY zimpha

    Acperience 展开式子, \(\left\| W-\alpha B \right\|^2=\displaystyle\alpha^2\sum_{i=1}^{n}b_i^2-2\alpha\su ...

  6. 尽量写出大家都能看懂的ReactJS入门教程

    个人感觉ReactJS相比于传统的JS框架还是挺有意思的,主要是它将JS代码和HTML代码完美的结合在了一起,有点jsp把java代码和html混在一起写的意思?但是它通过组件的形式实现了代码可复用, ...

  7. 进程Queue、线程Queue、堆栈、生产者消费者模型

    没学队列之前,可以用文件实现进程之间通信 但是有2个问题: 1. 速度慢:文件是保存在硬盘空间 2. 为了数据安全要加锁(处理锁是一件很麻烦的事,容易死锁,建议自己轻易不要处理锁) 队列:队列是基于管 ...

  8. P1003 铺地毯(noip 2011)

    洛谷——P1003 铺地毯 题目描述 为了准备一个独特的颁奖典礼,组织者在会场的一片矩形区域(可看做是平面直角坐标系的第一象限)铺上一些矩形地毯.一共有 n 张地毯,编号从 1 到n .现在将这些地毯 ...

  9. hihocoder 1579(排列组合)

    题意 给出一个长度为n的字符串的sa数组,n<=1e5,问有多少种不同的字符串的sa数组正好是输入的sa数组(字符串每个位置都是小写字母) 分析 sa数组描述的是字符的大小关系,而不是确切的字符 ...

  10. easyUI排序问题

    使用easyUI时,需要在点击页面的某一列进行desc或asc排序,那么在jsp中可以把该列js的sortable 设置true. 加在某字段上时,该字段点击时页面会出现一小三角图案 ,此时easyU ...