# JAVA之旅(二十六)——装饰设计模式,继承和装饰的区别,LineNumberReader,自定义LineNumberReader,字节流读取操作,I/O复制图片
* * *
## 一.装饰设计模式
> 其实我们自定义readLine就是一种装饰模式
* 当想要对已有的对象进行功能增强时,可以定义一个类,将已有对象传入,并且提供加强功能,那么自定义的该类就称为装饰类
~~~
package com.lgl.hellojava;
public class HelloJJAVA {
public static void main(String[] args) {
Person p = new Person();
p.eat();
// 开始进行增强
superPerson p1 = new superPerson(p);
p1.superEat();
}
}
class Person {
public void eat() {
System.out.println("吃饭");
}
}
class superPerson {
private Person p;
public superPerson(Person p) {
this.p = p;
}
public void superEat() {
System.out.println("小菜+吃饭");
}
}
~~~
> 这里的逻辑就是当我们吃饭这个功能需要增强的时候,我们应该装饰他
* 装饰类通常会通过构造方法接收被装饰的对象,并基于被装饰的对象的功能提供更强的功能
## 二.继承和装饰的区别
> 你现在知道了装饰模式,那你一定会疑问,和继承的道理类似,对吧,我们现在来说下他们的区别
>
> 这里我们就不写代码了,我们看注释
~~~
package com.lgl.hellojava;
public class HelloJJAVA {
public static void main(String[] args) {
/**
* MyReader:专门用于读取数据的类
* MyTextReader:专门读取文本 两个向上抽取,形成继承体系
*/
/**
* 想实现更多的功能
* MyBufferReader
* myBufferTestReader
*/
/**
*谁需要加强就传谁进来
* class MyBufferReader{
* }
*/
}
}
~~~
> 这个逻辑大概是这样的,我们有两个功能,一个读取文件,一个读取文本,他们其实是有共性的,你就把他们共性部分抽取出来,可是我现在在读取文本的时候我顺便想读取图片呢?其实,我们就是这样才产生的装饰者模式
* 装饰者模式比继承要灵活,避免了继承体系的臃肿,而且降低了类与类之间的关系
* 装饰类因为增强已有对象,具备功能和已有的想相同,只不过提供了更强的功能,所以装饰类和被装饰类通常属于一个体系中的
## 三.LineNumberReader
> 这也是一个子类
![这里写图片描述](http://img.blog.csdn.net/20160707214409225)
> 他也是一个包装类,我们看例子
~~~
package com.lgl.hellojava;
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.IOException;
import java.io.LineNumberReader;
public class HelloJJAVA {
public static void main(String[] args) {
FileReader fr;
try {
fr = new FileReader("test.txt");
LineNumberReader lnr = new LineNumberReader(fr);
String line = null;
while((line = lnr.readLine()) != null){
System.out.println(lnr.getLineNumber()+":"+line);
}
lnr.close();
} catch (FileNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
~~~
> 他输出的结果
![这里写图片描述](http://img.blog.csdn.net/20160707214921086)
> 他可以获取和设置行号
## 四.自定义LineNumberReader
> 我们可以根据他的原理自己也来实现一个,仔细看注释
~~~
package com.lgl.hellojava;
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.IOException;
import java.io.Reader;
public class HelloJJAVA {
public static void main(String[] args) {
try {
FileReader fr = new FileReader("test.txt");
MyLineNumberReader my = new MyLineNumberReader(fr);
String line = null;
while ((line = my.MyReadLine()) != null) {
System.out.println(my.getLineReader() + ":" + line);
}
my.MyClose();
} catch (FileNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
class MyLineNumberReader {
// 读取
private Reader r;
// 行号
private int lineReader;
// 构造方法
public MyLineNumberReader(Reader r) {
this.r = r;
}
// 提供对外方法
public String MyReadLine() {
// 行号自增
lineReader++;
StringBuilder sb = new StringBuilder();
int ch = 0;
try {
while ((ch = r.read()) != -1) {
if (ch == '\r')
continue;
if (ch == '\n')
return sb.toString();
else
sb.append((char) ch);
}
if (sb.length() != 0)
return sb.toString();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
return null;
}
public int getLineReader() {
return lineReader;
}
public void setLineReader(int lineReader) {
this.lineReader = lineReader;
}
public void MyClose() {
try {
r.close();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
~~~
> 这个思路是不是很清晰,实际上和LineNumberReader是类似的
## 五.字节流读取操作
> 字符流我们讲的差不多了,我们接着说字节,其实他们类似的,知识他操作的是字节而已
* inputStream:读
* outputStream:写
> 我们还是从例子开始
~~~
package com.lgl.hellojava;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
public class HelloJJAVA {
public static void main(String[] args) {
writeFile();
}
// 写文件
public static void writeFile() {
try {
FileOutputStream fo = new FileOutputStream("demo.txt");
fo.write("test".getBytes());
fo.close();
} catch (FileNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
~~~
> 这里我们可以看到,他写入数据不需要刷新,现在还没有涉及到缓存区,我们继续看,写已经写好了,现在我们开始读,对于读取数据,我们开头用到的两种方法
~~~
// 字符读数据
public static void readFile() {
try {
FileInputStream fs = new FileInputStream("demo.txt");
int ch = 0;
while ((ch = fs.read()) != -1) {
System.out.println((char) ch);
}
fs.close();
} catch (FileNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
// 字节读取
public static void readFile1() {
try {
FileInputStream fs = new FileInputStream("demo.txt");
byte[] buf = new byte[1024];
int len = 0;
while ((len = fs.read(buf)) != -1) {
System.out.println(new String(buf, 0, len));
}
fs.close();
} catch (FileNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
~~~
> 现在我们有了专门处理的字节流,我们可以这样做
~~~
public static void readFile2() {
try {
FileInputStream fs = new FileInputStream("demo.txt");
int num = fs.available();
System.out.println(num);
fs.close();
} catch (FileNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
~~~
> 我们发现直接用available就可以拿到字节了,原理其实是这段代码
~~~
public static void readFile2() {
try {
FileInputStream fs = new FileInputStream("demo.txt");
byte[] buf = new byte[fs.available()];
fs.read(buf);
System.out.println(new String(buf));
fs.close();
} catch (FileNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
~~~
## 六.I/O复制图片
> ok,这里算是一个小练习,复制一张图片,我们理顺下思路
* 1.用字节读取流和图片关联
* 2.用字节流写入流对象创建一个图片文件,存储数据
* 3.通过循环读写,完成数据存储
* 4.关闭流
> OK,我们用代码说话
~~~
package com.lgl.hellojava;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
public class HelloJJAVA {
public static void main(String[] args) {
FileOutputStream fos = null;
FileInputStream fis = null;
try {
// 复制
fos = new FileOutputStream("copy_img.png");
// 原图
fis = new FileInputStream("img.png");
byte[] buf = new byte[1024];
int len = 0;
while ((len = fis.read(buf)) != -1) {
fos.write(buf, 0, len);
}
fis.close();
fos.close();
} catch (FileNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
~~~
> 这样。我们图片就拷贝过来了
![这里写图片描述](http://img.blog.csdn.net/20160707234857950)
> 好的,知识点今天就到这里
## 有兴趣的可以加群:555974449,咱们一起学习,一起进步!
版权声明:本文为博主原创文章,博客地址:http://blog.csdn.net/qq_26787115,未经博主允许不得转载。
- 0-发现
- AndroidInterview-Q-A
- Android能让你少走弯路的干货整理
- LearningNotes
- temp
- temp11
- 部分地址
- 0-待办任务
- 待补充列表
- 0-未分类
- AndroidView事件分发与滑动冲突处理
- Spannable
- 事件分发机制详解
- 1-Java
- 1-Java-01基础
- 未归档
- 你应该知道的JDK知识
- 集合框架
- 1-Java-04合集
- Java之旅0
- Java之旅
- JAVA之旅01
- JAVA之旅02
- JAVA之旅03
- JAVA之旅04
- JAVA之旅05
- JAVA之旅06
- JAVA之旅07
- JAVA之旅08
- JAVA之旅09
- java之旅1
- JAVA之旅10
- JAVA之旅11
- JAVA之旅12
- JAVA之旅13
- JAVA之旅14
- JAVA之旅15
- JAVA之旅16
- JAVA之旅17
- JAVA之旅18
- JAVA之旅19
- java之旅2
- JAVA之旅20
- JAVA之旅21
- JAVA之旅22
- JAVA之旅23
- JAVA之旅24
- JAVA之旅25
- JAVA之旅26
- JAVA之旅27
- JAVA之旅28
- JAVA之旅29
- java之旅3
- JAVA之旅30
- JAVA之旅31
- JAVA之旅32
- JAVA之旅33
- JAVA之旅34
- JAVA之旅35
- 1-Java-05辨析
- HashMapArrayMap
- Java8新特性
- Java8接口默认方法
- 图解HashMap(1)
- 图解HashMap(2)
- 2-Android
- 2-Android-1-基础
- View绘制流程
- 事件分发
- AndroidView的事件分发机制和滑动冲突解决
- 自定义View基础
- 1-安卓自定义View基础-坐标系
- 2-安卓自定义View基础-角度弧度
- 3-安卓自定义View基础-颜色
- 自定义View进阶
- 1-安卓自定义View进阶-分类和流程
- 10-安卓自定义View进阶-Matrix详解
- 11-安卓自定义View进阶-MatrixCamera
- 12-安卓自定义View进阶-事件分发机制原理
- 13-安卓自定义View进阶-事件分发机制详解
- 14-安卓自定义View进阶-MotionEvent详解
- 15-安卓自定义View进阶-特殊形状控件事件处理方案
- 16-安卓自定义View进阶-多点触控详解
- 17-安卓自定义View进阶-手势检测GestureDetector
- 2-安卓自定义View进阶-绘制基本图形
- 3-安卓自定义View进阶-画布操作
- 4-安卓自定义View进阶-图片文字
- 5-安卓自定义View进阶-Path基本操作
- 6-安卓自定义View进阶-贝塞尔曲线
- 7-安卓自定义View进阶-Path完结篇伪
- 8-安卓自定义View进阶-Path玩出花样PathMeasure
- 9-安卓自定义View进阶-Matrix原理
- 通用类介绍
- Application
- 2-Android-2-使用
- 2-Android-02控件
- ViewGroup
- ConstraintLayout
- CoordinatorLayout
- 2-Android-03三方使用
- Dagger2
- Dagger2图文完全教程
- Dagger2最清晰的使用教程
- Dagger2让你爱不释手-终结篇
- Dagger2让你爱不释手-重点概念讲解、融合篇
- dagger2让你爱不释手:基础依赖注入框架篇
- 阅读笔记
- Glide
- Google推荐的图片加载库Glide:最新版使用指南(含新特性)
- rxjava
- 这可能是最好的RxJava2.x入门教程完结版
- 这可能是最好的RxJava2.x入门教程(一)
- 这可能是最好的RxJava2.x入门教程(三)
- 这可能是最好的RxJava2.x入门教程(二)
- 这可能是最好的RxJava2.x入门教程(五)
- 这可能是最好的RxJava2.x入门教程(四)
- 2-Android-3-优化
- 优化概况
- 各种优化
- Android端秒开优化
- apk大小优化
- 内存分析
- 混淆
- 2-Android-4-工具
- adb命令
- 一键分析Android的BugReport
- 版本控制
- git
- git章节简述
- 2-Android-5-源码
- HandlerThread 源码分析
- IntentService的使用和源码分析
- 2-Android-9-辨析
- LRU算法
- 什么是Bitmap
- 常见图片压缩方式
- 3-Kotlin
- Kotlin使用笔记1-草稿
- Kotlin使用笔记2
- kotlin特性草稿
- Kotlin草稿-Delegation
- Kotlin草稿-Field
- Kotlin草稿-object
- 4-JavaScript
- 5-Python
- 6-Other
- Git
- Gradle
- Android中ProGuard配置和总结
- gradle使用笔记
- Nexus私服搭建
- 编译提速最佳实践
- 7-设计模式与架构
- 组件化
- 组件化探索(OKR)
- 1-参考列表
- 2-1-组件化概述
- 2-2-gradle配置
- 2-3-代码编写
- 2-4-常见问题
- 2-9-值得一读
- 8-数据结构与算法
- 0临时文件
- 汉诺塔
- 8-数据-1数据结构
- HashMap
- HashMap、Hashtable、HashSet 和 ConcurrentHashMap 的比较
- 迟到一年HashMap解读
- 8-数据-2算法
- 1个就够了
- Java常用排序算法(必须掌握的8大排序算法)
- 常用排序算法总结(性能+代码)
- 必须知道的八大种排序算法(java实现)
- 9-职业
- 阅读
- 书单
- 面试
- 面试-01-java
- Java面试题全集骆昊(上)
- Java面试题全集骆昊(下)
- Java面试题全集骆昊(中)
- 面试-02-android
- 40道Android面试题
- 面试-03-开源源码
- Android图片加载框架最全解析(二),从源码的角度理解Glide的执行流程
- 面试-07-设计模式
- 面试-08-算法
- 面试-09-其他
- SUMMARY
- 版权说明
- temp111