使用控制抽象函数(贷出模式loan pattern)

2019-08-28  本文已影响0人  no0one

(Scala编程P.173)打开某个资源,对它进行操作,然后关闭这个资源。可以用类似如下的方法,将这个模式捕获成一个控制抽象,withPrintWriter打开某个资源并将这个资源“贷出”给函数op

def withPrintWriter(file: File, op: PrintWriter => Unit): Unit = {
  val writer = new PrintWriter(file)
  try {
    op(writer)
  } finally {
    writer.close()
  }
}

withPrintWriter(
  new File("date.txt"),
  writer => writer.println(new java.util.Date)
)

实践:

读取配置文件,都是打开文件,读取配置文件内容,可以考虑贷出模式

ReadConfigFile.scala

import com.typesafe.config.{Config, ConfigFactory}

import scala.reflect.ClassTag

class ReadConfigFile[T:ClassTag](clz: Class[T]){
  def read(fileName: String)(fromConfig: Config => T): T = {  //柯里化
    val file = new File(fileName)
    if (!file.exists())  clz.newInstance()//不能直接new T(),scala不能识别,需要通过ClassTag,增加Class[T]入参用,newInstance来构造一个新的对象,而且这个对象不能带入参。
    else {
      println(s"path = $fileName")
      try {
        val conf:Config = ConfigFactory.parseFile(file).withFallback(ConfigFactory.load())
        fromConfig(conf)
      } catch {
        case ex: Exception => clz.newInstance()
      }
    }
  }
}

这里用到了泛型类,可以传入对于任何需要读取配置参数类,将读取文件的方法通过fromConfig函数值传进来

PersonConfig.scala

class PersonConfig{
  var name: String = ""
  var age: Int = 0

  override def toString: String = {
    "name=" + name + " age=" + age
  }
}

object PersonConfig{
  def createPersonConfig(config: Config): PersonConfig= {
    val PersonConfig= new PersonConfig
    PersonConfig.name = config.getString("name ")
    PersonConfig.age= config.getInt("age")
    PersonConfig
  }
}

ITDepartmentConfig.scala

class ITDepartmentConfigextends ReadConfigFile(classOf[PersonConfig]){
  private val personConfig: PersonConfig= {
    read("person.conf") { config =>
      PersonConfig.createServerConfig(config.getConfig("person_config"))
    }//传入读取文件的方法并生成相关的对象
  }

  def person: PersonConfig= personConfig
}

**Test.scala**
object Test {
  def main(args: Array[String]): Unit = {
    val itDepartmentConfig= new ITDepartmentConfig
    println(itDepartmentConfig.person.toString)
  }
}

上这里提到了不能直接new T()的问题,可以通过多加一个传名参数来解决,就不需要使用ClassTag,使用ClassTag导致构造函数中不可以带入参。方法如下:

class ReadConfigFile[T]{
  def read(fileName: String)(fromConfig: Config => T)(emptyObject: => T): T = {
    val file = new File(fileName)
    if (!file.exists())  emptyObject
    else {
      println(s" path = $fileName ")
      try {
        val conf:Config = ConfigFactory.parseFile(file).withFallback(ConfigFactory.load())
        fromConfig(conf)
      } catch {
        case ex: Exception => emptyObject
      }
    }
  }
}

调用时

class RESTServerConfig extends ReadConfigFile[ServerConfig] with Subject {
  private val serverConfig: ServerConfig = {
    read(FileUtil.getConfigPath + "rest-server.conf") { config =>
      ServerConfig.createServerConfig(config.getConfig("rest_server_config"))
    }(new ServerConfig())
  }

  def server: ServerConfig = serverConfig
}
上一篇 下一篇

猜你喜欢

热点阅读