元式编程

---
title: 元式编程
date: 2018-06-09 16:29:00
updated: 2019-01-10 12:00:00
categories:
- web
tags:
- front end
---
目录

# 什么是它
# 如何实现
# 原始示例
# 逐步进阶
# ----》公共代码
# ----》函数注册
# ----》动态注册
# ----》数操分离
# ----》不会报错
# 参考文献
# 同级文章

正文

# 什么是它

“写一段自动写程序的程序”

元式编程就是将代码视作数据,直接用字符串 or AST or 其他任何形式去操纵代码,以此获得一些维护性、效率上的好处。

# 如何实现
Javascript 中,evalnew Function()便是两个可以用来进行元式编程的特性。
# 原始示例

class User {
    constructor(userID) {
        this.id = userID;
    }

    get_name() {
        return $.ajax(`/get_name?id=${this.id}`);
    }

    get_sex() {
        return $.ajax(`/get_sex?id=${this.id}`);
    }

    //下面是get_age、get_address......
}

代码问题:
①用户数据有多少个字段,我们就要定义多少个 get_something方法。
②这些方法里逻辑都是重复的,都是一个简单的 ajax。

# 公共代码

class User {
    constructor(userID) {
        this.id = userID;
    }
    
    __fetchData(key) {
        //这是一个private方法,直接调用类似__fetchData("age")是不被允许的
        return $.ajax(`/get_${key}?id=${this.id}`)
    }

    get_name() {
        return this.__fetchData('name');
    }

    get_sex() {
        return this.__fetchData("sex");
    }

    //下面是get_age、get_address......
}

可圈可点:
①将公共逻辑进行提取,封装__fetchData

# 函数注册

class User {
    constructor(userID) {
        this.id = userID;
        this.registerProperties(["name", "age", "sex", "address"]);
    }

    registerProperties(keyArray) {
        keyArray.forEach(key => {
            this[`get_${key}`] = () => this.__fetchData(key);
        })
    }

    __fetchData(key) {
        //这是一个private方法,直接调用类似__fetchData("age")是不被允许的
        return $.ajax(`/get_${key}?id=${this.id}`)
    }
}

可圈可点:
①将公共逻辑进行提取,封装__fetchData
②使用函数registerProperties给对象注册键值,避免了有多少个字段,我们就要定义多少个 get_something 方法。

# 动态注册

class User {
    constructor(userID) {
        this.id = userID;
        this.registerProperties(["name", "age", "sex", "address"]);
    }

    registerProperties(keyArray) {
        keyArray.forEach(key => {
            //注意这里的fnBody内部依然采用ES5的写法,因为babel目前不会编译函数字符串。
            var fnBody = `return this.__fetchData("/get_${key}?id=${this.id}")
                    .then(function(data){
                        return this.__handle_${key}?_this.handle_${key}(data):data;
                    })`;
            this[`get_${key}`] = new Function(fnBody);
        })
    }

    __handle_name(name) {
        //do somthing with name...
        return name;
    }

    __handle_age(age) {
        //do somthing with age...
        return age;
    }

    __fetchData(key) {
        //这是一个private方法,直接调用类似__fetchData("age")是不被允许的
        return $.ajax(`/get_${key}?id=${this.id}`)
    }
}

可圈可点:
①将公共逻辑进行提取,封装__fetchData
②使用函数registerProperties给对象注册键值,避免了有多少个字段,我们就要定义多少个get_something 方法。
③在拉取数据之后,我们要对部分数据进行一定的处理,比如对 name 我们要去掉首尾的空格,对 age 我们要加上一个“岁”字。具体的处理方法定义在 __handle_something 里面(以某种格式规范定义)。
④通过 new Function() ,动态生成函数。

# 控操分离

class User {
    constructor(userID, dataBase) {
        this.id = userID;
        this.__dataBase = dataBase;
        for (var method in dataBase) {
            //对每一个方法
            this.registerMethod(method);
        }
    }

    registerMethod(methodName) {
        //这里除去了前置的"get_"
        var propertyName = methodName.slice(4);
        
        //注意这里拉取数据的方法改为使用dataBase
        var fnBody = `return this.__dataBase.${methodName}()
                    .then(function(data){
                        return this.__handle_${propertyName}?_this.handle_${propertyName}(data):data;
                    })`;
        this[`get_${propertyName}`] = new Function(fnBody);
    }

    __handle_name(name) {
        //do somthing with name...
        return name;
    }

    __handle_age(age) {
        //do somthing with age...
        return age;
    }
}
var userDataBase = new UserDataBase();
var user = new User("123", userDataBase);

可圈可点:
①将公共逻辑进行提取,封装__dataBase__fetchData),并且语义化
②使用函数registerMethod(registerProperties)给对象注册方法(键值),避免了有多少个字段,我们就要定义多少个get_something 方法。
③在获取(拉取)数据之后,我们要对部分数据进行一定的处理,比如对 name 我们要去掉首尾的空格,对 age 我们要加上一个“岁”字。具体的处理方法定义在 __handle_something 里面(以某种格式规范定义)。
④通过 new Function() ,动态生成函数。
⑤通过一个别人封装好的 UserDataBase 里的方法来拉取传入的用户数据,促进松耦合。读取所有拉取字段的方法,然后通过元编程的方式来动态生成对应的方法,避免了冗余。

# 不会报错

function createUser(id, userDataBase) {
    return new Proxy(new User(id, userDataBase), {
        get: (target, property) => (typeof(target[property]) === "function" ? target[property] : () => false)
    })
}

var userDataBase = new UserDataBase();
var user = createUser("123", userDataBase);

user.get_name() => // fetch name data
user.get_wwwwww() // => false

①将公共逻辑进行提取,封装__dataBase__fetchData),并且语义化
②使用函数registerMethod(registerProperties)给对象注册方法(键值),避免了有多少个字段,我们就要定义多少个get_something 方法。
③在获取(拉取)数据之后,我们要对部分数据进行一定的处理,比如对 name 我们要去掉首尾的空格,对 age 我们要加上一个“岁”字。具体的处理方法定义在 __handle_something 里面(以某种格式规范定义)。
④通过 new Function() ,动态生成函数。
⑤通过一个别人封装好的 UserDataBase 里的方法来拉取传入的用户数据,促进松耦合。读取所有拉取字段的方法,然后通过元编程的方式来动态生成对应的方法,避免了冗余。
⑥用户数据中不存在www字段,即使这样执行user.get_wwwwww()也不会报错。
# 参考文献
[王伟嘉].Javascript元编程(一).2016-04-16.segmentfault
# 同级文章
函数编程
对象编程

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,793评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,567评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,342评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,825评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,814评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,680评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,033评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,687评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,175评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,668评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,775评论 1 332
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,419评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,020评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,978评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,206评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,092评论 2 351
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,510评论 2 343

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,453评论 25 707
  • 蒲公英的迁徙 文:莠子 我不是南飞的大雁 盘桓千里拥有两处家乡 我也不是被移植的草皮 成群结队覆盖新的地盘 我只是...
    莠子阅读 1,190评论 3 4
  • 这几天孩子做作业的时候,会有一点点的急躁,不够有耐心,所以让孩子静下心来画画,不但能够消除烦躁的情绪,还会...
    飞越高空阅读 134评论 0 3
  • 被老妈催婚已是非常熟悉,甚至习惯了。 但AQAL是个什么鬼? AQAL整合研究模型 AQAL是英文All Quot...
    逯晓风阅读 1,424评论 2 4