关于Java回调方法的理解

什么是回调方法

个人理解的回调是指在解决某个问题时需要两步操作,此时第一步操作可以看作是为了解决问题,第二步操作是在第一步操作结果的基础上完善或补充操作。此处的第二步操作一般就是回调方法。为什么这么说呢,常规来说为了解决一个问题,我们可能需要调用某些耗时不确定的方法,若让主线程一直等待调用结果很容易造成程序无响应而卡死,影响用户体验。此时就可以用到回调方法来解决问题,将耗时不确定的方法设计为带有回调参数的方法,在耗时方法执行结束时候,继续做后续操作。这里被封装起来的这些操作就是回调方法。

怎么使用回调方法

此处以spring封装的jdbctemplate在查询数据库后可以得到不同类型的数据为例。众所周知,从数据库中查询数据得到的肯定是一个resultset,但这在开发过程中并不能满足实际的需求。我们一般希望得到的是实体对象,或是一个集合等。那么此时有两种选择,一是先进行数据查询,得到result,然后再通过第二个方法对result进行封装。第二种则是在进行数据查询的同时,调用者告诉spring,查询结束后得到result,顺便去执行封装的方法。然后再把封装结果给我。两种方法其实都可行,但是常用的确是第二种。这是为什么呢?一般来说使用回调都是为了避免耗时较长的方法。但更重要的原因是面向对象设计的封装性,模块间要解耦,模块内要内聚。使用回调可以降低模块间的耦合性。另外,在上例中,因为常规关系型数据库并不是面向对象的,所以我们得到的结果并不是能够满足oop的要求,故通过封装,直接得到对应实体或实体集合更符合面向对象的思想。而在平时使用的过程中,通过sql查询并直接返回封装对象更方便,也更符合面向对象的思想。
通过如下代码再来理解回调的使用:
假设数据库中有如下数据:


image.png

Person实体类如下

package com.af.study.spring.bean;

/**
 * person实体
 * Created by zyb on 2017/06/05.
 */
public class Person {

    String id;
    String name;
    Integer sex;

    public String getId() {
        return id;
    }

    public void setId(String id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public Integer getSex() {
        return sex;
    }

    public void setSex(Integer sex) {
        this.sex = sex;
    }
}

** PersonDao,在封装数据时使用了回调。**

package com.af.study.spring.jdbc;

import com.af.study.spring.bean.Person;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.jdbc.core.BeanPropertyRowMapper;
import org.springframework.jdbc.core.JdbcTemplate;
import org.springframework.jdbc.core.namedparam.NamedParameterJdbcDaoSupport;
import org.springframework.stereotype.Component;

import java.util.List;

/**
 * personDao
 * Created by zyb on 2017/07/17.
 */
@Component
public class PersonDao extends NamedParameterJdbcDaoSupport {

    @Autowired
    public PersonDao(JdbcTemplate jdbcTemplate) {
        setJdbcTemplate(jdbcTemplate);
    }

    List<Person> getBeans(){
        return this.getJdbcTemplate().query("select * from t_person", BeanPropertyRowMapper.newInstance(Person.class));
    }
}

测试类

package com.af.study.spring.jdbc;

import com.af.study.spring.bean.Person;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

import java.util.List;

/**
 * Created by zyb on 2017/07/17.
 */
public class PersonDaoTest {

    public static void main(String[] args) {
        ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
        PersonDao personDao = ctx.getBean("personDao", PersonDao.class);
        List<Person> personList = personDao.getBeans();
        for (Person person : personList) {
            System.out.println(person.getId() + " --- " + person.getName() + " --- " + person.getSex());
        }
    }
}

运行后结果是:

1 --- lx --- 0
2 --- yh --- 1

从上面简单示例可以看到,查询数据库之后直接得到了Person对象集合,而并不是常规的resultSet。spring通过传递一个row mapper参数进去,自动实现一个回掉方法,在得到result set之后进行对象的封装。
通过查看spring源码可以看到如下代码:

public <T> T query(final String sql, final ResultSetExtractor<T> rse) throws DataAccessException {
        Assert.notNull(sql, "SQL must not be null");
        Assert.notNull(rse, "ResultSetExtractor must not be null");
        if (logger.isDebugEnabled()) {
            logger.debug("Executing SQL query [" + sql + "]");
        }
        class QueryStatementCallback implements StatementCallback<T>, SqlProvider {
            @Override
            public T doInStatement(Statement stmt) throws SQLException {
                ResultSet rs = null;
                try {
                    rs = stmt.executeQuery(sql);
                    ResultSet rsToUse = rs;
                    if (nativeJdbcExtractor != null) {
                        rsToUse = nativeJdbcExtractor.getNativeResultSet(rs);
                    }
                    // 此处调用了回调方法,前面的代码得到result set,通过如下方法,实现封装并返回
                    return rse.extractData(rsToUse);
                }
                finally {
                    JdbcUtils.closeResultSet(rs);
                }
            }
            @Override
            public String getSql() {
                return sql;
            }
        }
        return execute(new QueryStatementCallback());
    }

此处用了内部类来封装操作,将耗时操作(即数据库查询)和后续操作(即封装对象)组合成了新的方法,再将内部类作为参数传递给execute方法,在execute方法中可以看到,直接调用了组合后的方法,将两步操作结果合并并返回。

public <T> T execute(StatementCallback<T> action) throws DataAccessException {
        Assert.notNull(action, "Callback object must not be null");

        Connection con = DataSourceUtils.getConnection(getDataSource());
        Statement stmt = null;
        try {
            Connection conToUse = con;
            if (this.nativeJdbcExtractor != null &&
                    this.nativeJdbcExtractor.isNativeConnectionNecessaryForNativeStatements()) {
                conToUse = this.nativeJdbcExtractor.getNativeConnection(con);
            }
            stmt = conToUse.createStatement();
            applyStatementSettings(stmt);
            Statement stmtToUse = stmt;
            if (this.nativeJdbcExtractor != null) {
                stmtToUse = this.nativeJdbcExtractor.getNativeStatement(stmt);
            }
            T result = action.doInStatement(stmtToUse);
            handleWarnings(stmt);
            return result;
        }
        catch (SQLException ex) {
            // Release Connection early, to avoid potential connection pool deadlock
            // in the case when the exception translator hasn't been initialized yet.
            JdbcUtils.closeStatement(stmt);
            stmt = null;
            DataSourceUtils.releaseConnection(con, getDataSource());
            con = null;
            throw getExceptionTranslator().translate("StatementCallback", getSql(action), ex);
        }
        finally {
            JdbcUtils.closeStatement(stmt);
            DataSourceUtils.releaseConnection(con, getDataSource());
        }
    }

继续跟进extractData方法,可以看到实际上就是遍历了result set,将每一行记录进行封装。

    public List<T> extractData(ResultSet rs) throws SQLException {
        List<T> results = (this.rowsExpected > 0 ? new ArrayList<T>(this.rowsExpected) : new ArrayList<T>());
        int rowNum = 0;
        while (rs.next()) {
            results.add(this.rowMapper.mapRow(rs, rowNum++));
        }
        return results;
    }

到这里,对于回调方法的演示基本就结束了。下面来说说如何设计、使用回调。

如何设计自己的回调方法

如文章开头所说,回调方法实际上是将耗时操作之后的行为封装,作为参数传递给耗时的方法。通常是使用接口配合内部类的方法来实现,之所以使用内部类+接口,我的理解是结合了代理的思想来实现,通过接口来限定需要代理的方法,在通过内部类来实现接口,不会产生额外的类,也不会对其他地方造成影响。总之,理解了回调就是将操作组合成新的方法作为参数传递,相信在代码中运用回调并不是难事。

写在最后

并不是一定要使用回调,回调并不会对功能造成影响,只是提供了一种提升效率,减少因为等待时间等因素造成bug等问题的解决方案。所有回调都可以将两步操作分开。要做到深入理解回调使用的场景,不要为了使用而强行使用,反而造成了效率问题等。同样也适用于设计模式的学习中。
本文也仅仅是个人的理解,在表述不清或理解不到位的地方,欢迎各位大神指正,也求各位轻拍,刚开始写博客,可能有些地方也表述的不够清晰,望大家能多多包涵。

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

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,363评论 25 707
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,556评论 18 399
  • 国家电网公司企业标准(Q/GDW)- 面向对象的用电信息数据交换协议 - 报批稿:20170802 前言: 排版 ...
    庭说阅读 10,846评论 6 13
  • 文/陈皮朵娃 老唐头从学校政教处长的位置退休半年,闲得没事干,怪难受的。他很怀念上班的日子,那时候三天两头有个把歪...
    陈皮朵娃阅读 255评论 2 1
  • 挑剔 老妈远途旅行归来,一回家就进厨房忙活,说是用了从南京带回来了食材做梅菜扣肉,要给无肉不欢的老爸一个惊喜。锅上...
    希瑞爱夏天阅读 110评论 0 0