IO/NIO/AIO 基本概念

1.阻塞IO—Socket

Socket又称“套接字”,应用程序通常通过“套接字”向网络发出请求或应答网络请求。

Socket和ServerSocket类库位于java.net包中。ServerSocket用于服务器端,Socket是建立网络连接时使用的。在连接成功时,应用程序两端都会产生一个Socket实例,操作这个实例,完成所需的会话。对于一个网络连接来说,套接字是平等的,不会因为在服务器端或在客户端而产生不同的级别。不管是ServerSocket还是Socket,它们的工作都是通过SocketImpl类及其子类完成的。

套接字的连接过程可以分为四个步骤:服务器监听、客户端请求服务器、服务器端连接确认、客户端连接确认并进行通信。

(1)服务器监听:服务器端套接字并不定位具体的客户端套接字,而是出于等待连接的状态,实时监控网络状态。

(2)客户端请求:客户端的套接字提出连接请求,要连接的目标是服务器端的套接字。为此,客户端的套接字必须首先描述要连接的服务器端的套接字,指出服务器端的套接字的地址和端口号,然后向服务器端套接字提出连接请求。

(3)服务器端连接确认:当服务器端的套接字监听到或者说接收到客户端套接字的连接请求,它就响应客户端套接字的请求,建立一个新的线程,把服务器端套接字的描述发送给客户端。

(4)客户端连接确认:一旦客户端确认了此描述,连接就建立好了,双方开始通信。而服务器端套接字继续处于监听状态,继续接收其他客户端套接字的连接请求。

借用一下网上的Socket通信模型图片:

Socket通信步骤:

①创建ServerSocket和Socket

②打开连接到Socket的输入/输出流

③按照协议对Socket进行读写操作

④关闭输入输出流、关闭Socket

服务器端:

①创建ServerSocket对象,绑定监听端口

②通过accept()方法监听客户端请求

③建立连接后,通过输入流读取客户端发送的请求信息

④通过输出流向客户端发送响应信息

⑤关闭相关资源

客户端:

①创建Socket对象,指明需要连接的服务器的地址和端口号

②连接建立后,通过输出流向服务器端发送请求信息

③通过输入流获取服务器响应信息

④关闭响应资源

下面看一个简单的小例子:

服务器端响应工具类:

 

 
  1. public class ServerHandler implements Runnable {

  2.  
  3. private Socket socket;

  4.  
  5. public ServerHandler(Socket socket) {

  6. this.socket = socket;

  7. }

  8.  
  9. @Override

  10. public void run() {

  11. BufferedReader bufferedReader = null;

  12. PrintWriter printWriter = null;

  13. try {

  14. bufferedReader = new BufferedReader(new InputStreamReader(socket.getInputStream()));

  15. printWriter = new PrintWriter(socket.getOutputStream(), true);

  16.  
  17. while (true) {

  18. String info = bufferedReader.readLine();

  19. if(info == null)

  20. break;

  21. System.out.println("客户端发送的消息:" + info);

  22. printWriter.println("服务器端响应了客户端请求....");

  23. }

  24. } catch (Exception e) {

  25. e.printStackTrace();

  26. } finally {

  27. if(bufferedReader != null){

  28. try {

  29. bufferedReader.close();

  30. } catch (IOException e) {

  31. e.printStackTrace();

  32. }

  33. }

  34. if(printWriter != null){

  35. try {

  36. printWriter.close();

  37. } catch (Exception e) {

  38. e.printStackTrace();

  39. }

  40. }

  41. if(socket != null){

  42. try {

  43. socket.close();

  44. } catch (IOException e) {

  45. e.printStackTrace();

  46. }

  47. }

  48. socket = null;

  49. }

  50. }

  51. }

服务器端:

 

 

 
  1. public class Server {

  2. private static int PORT = 8379;

  3. public static void main(String[] args) {

  4. ServerSocket serverSocket = null;

  5. try {

  6. serverSocket = new ServerSocket(PORT);

  7. System.out.println("服务器端启动了....");

  8. //进行阻塞

  9. Socket socket = serverSocket.accept();

  10. //启动一个线程来处理客户端请求

  11. new Thread(new ServerHandler(socket)).start();

  12. } catch (Exception e) {

  13. e.printStackTrace();

  14. } finally {

  15. if(serverSocket != null){

  16. try {

  17. serverSocket.close();

  18. } catch (IOException e) {

  19. e.printStackTrace();

  20. }

  21. }

  22. serverSocket = null;

  23. }

  24. }

  25. }

客户端:

 

 

 
  1. public class Client {

  2. private static int PORT = 8379;

  3. private static String IP = "127.0.0.1";

  4.  
  5. public static void main(String[] args) {

  6. BufferedReader bufferedReader = null;

  7. PrintWriter printWriter = null;

  8. Socket socket = null;

  9. try {

  10. socket = new Socket(IP, PORT);

  11. printWriter = new PrintWriter(socket.getOutputStream(), true);

  12. bufferedReader = new BufferedReader(new InputStreamReader(socket.getInputStream()));

  13.  
  14. printWriter.println("客户端请求了服务器....");

  15. String response = bufferedReader.readLine();

  16. System.out.println("Client:" + response);

  17. } catch (Exception e) {

  18. e.printStackTrace();

  19. } finally {

  20. if(bufferedReader != null){

  21. try {

  22. bufferedReader.close();

  23. } catch (IOException e) {

  24. e.printStackTrace();

  25. }

  26. }

  27. if(printWriter != null){

  28. try {

  29. printWriter.close();

  30. } catch (Exception e) {

  31. e.printStackTrace();

  32. }

  33. }

  34. if(socket != null){

  35. try {

  36. socket.close();

  37. } catch (IOException e) {

  38. e.printStackTrace();

  39. }

  40. } else {

  41. socket = null;

  42. }

  43. }

  44. }

  45. }

运行结果:

 


以上的代码有个问题,就是每次有客户端请求服务器端都会创建一个线程,当线程过多时,服务器端可能会宕机。解决这个问题,可以使用JDK提供的线程池(伪异步)。其它地方都不变,将服务器端的代码修改成如下即可:

 

 
  1. public class Server {

  2. private static int PORT = 8379;

  3. public static void main(String[] args) {

  4. ServerSocket serverSocket = null;

  5. try {

  6. serverSocket = new ServerSocket(PORT);

  7. System.out.println("服务器端启动了....");

  8. //进行阻塞

  9. Socket socket = null;

  10. //启动一个线程来处理客户端请求

  11. //new Thread(new ServerHandler(socket)).start();

  12. HandlerExecutorPool pool = new HandlerExecutorPool(50, 1000);

  13. while (true) {

  14. socket = serverSocket.accept();

  15. pool.execute(new ServerHandler(socket));

  16. }

  17. } catch (Exception e) {

  18. e.printStackTrace();

  19. } finally {

  20. if(serverSocket != null){

  21. try {

  22. serverSocket.close();

  23. } catch (IOException e) {

  24. e.printStackTrace();

  25. }

  26. }

  27. serverSocket = null;

  28. }

  29. }

  30. }

其中HandlerExecutorPool为自定义的线程池,代码如下:

 
  1. public class HandlerExecutorPool {

  2.  
  3. private ExecutorService executor;

  4.  
  5. public HandlerExecutorPool(int maxSize, int queueSize) {

  6. this.executor = new ThreadPoolExecutor(Runtime.getRuntime().availableProcessors(), maxSize, 120L, TimeUnit.SECONDS,

 
  1. new ArrayBlockingQueue<>(queueSize));

  2. }

  3.  
  4. public void execute(Runnable task) {

  5. executor.execute(task);

  6. }

  7. }

2.IO(BIO)与NIO的区别

其本质就是阻塞和非阻塞的区别。

阻塞概念:应用程序在获取网络数据的时候,如果网络传输数据很慢,那么程序就一直等着,直到传输完毕为止。

非阻塞概念:应用程序直接可以获取已经准备就绪的数据,无需等待。

IO为同步阻塞形式,NIO为同步非阻塞形式。NIO没有实现异步,在JDK1.7之后,升级了NIO库包,支持异步非阻塞通信模型,即NIO2.0(AIO)。

同步和异步:同步和异步一般是面向操作系统与应用程序对IO操作的层面上来区别的。①同步时,应用程序会直接参与IO读写操作,并且应用程序会直接阻塞到某一个方法上,直到数据准备就绪(BIO);或者采用轮询的策略实时检查数据的就绪状态,如果就绪则获取数据(NIO)。②异步时,则所有的IO读写操作都交给操作系统处理,与应用程序没有直接关系,应用程序并不关心IO读写,当操作系统完成IO读写操作时,会向应用程序发出通知,应用程序直接获取数据即可。

同步说的是Server服务端的执行方式,阻塞说的是具体的技术,接收数据的方式、状态(io、nio)。

3.NIO编程介绍

学习NIO编程,首先需要了解几个概念:

(1)Buffer(缓冲区)

Buffer是一个对象,它包含一些需要写入或者读取的数据。在NIO类库中加入Buffer对象,体现了新类库与原IO的一个重要区别。在面向流的IO中,可以直接将数据写入或读取到Stream对象中。在NIO类库中,所有的数据都是用缓冲区处理的(读写)。 缓冲区实质上是一个数组,通常它是一个字节数组(ByteBuffer),也可以使用其他类型的数组。这个数组为缓冲区提供了访问数据的读写等操作属性,如位置、容量、上限等概念,具体的可以参考API文档。

Buffer类型:最常使用的是ByteBuffer,实际上每一种java基本类型都对应了一种缓存区(除了Boolean类型)。

①ByteBuffer②CharBuffer③ShortBuffer④IntBuffer⑤LongBuffer⑥FloatBuffer⑦DoubleBuffer

(2)Channel(管道、通道)

Channel就像自来水管道一样,网络数据通过Channel读取和写入,通道与流的不同之处在于通道是双向的,而流只能在一个方向上移动(一个流必须是InputStream或者OutputStream的子类),而通道可以用于读、写或者二者同时进行,最关键的是可以和多路复用器集合起来,有多种的状态位,方便多路复用器去识别。通道分为两大类:一类是用于网络读写的SelectableChannel,另一类是用于文件操作的FileChannel,我们使用的SocketChannel和ServerSocketChannel都是SelectableChannel的子类。

 

(3)Selector(选择器、多路复用器)

是NIO编程的基础,非常重要。多路复用器提供选择已经就绪的任务的能力。简单说,就是Selector会不断的轮询注册在其上的通道(Channel),如果某个通道发生了读写操作,这个通道就处于就绪状态,会被Selector轮询出来,然后通过SelectionKey可以取得就绪的Channel集合,从而进行后续的IO操作。一个多路复用器(Selector)可以负责成千上万的通道(Channel),没有上限。这也是JDK使用了epoll代替传统的select实现,获得连接句柄(客户端)没有限制。那也就意味着我们只要一个线程负责Selector的轮询,就可以接入成千上万个客户端,这是JDK NIO库的巨大进步。

Selector线程类似一个管理者(Master),管理了成千上万个管道,然后轮询哪个管道的数据已经准备好了,通知CPU执行IO的读取或写入操作。

Selector模式:当IO事件(管道)注册到选择器以后,Selector会分配给每个管道一个key值,相当于标签。Selector选择器是以轮询的方式进行查找注册的所有IO事件(管道),当IO事件(管道)准备就绪后,Selector就会识别,会通过key值来找到相应的管道,进行相关的数据处理操作(从管道中读取或写入数据,写到缓冲区中)。每个管道都会对选择器进行注册不同的事件状态,以便选择器查找。

事件状态:

SelectionKey.OP_CONNECT

SelectionKey.OP_ACCEPT

SelectionKey.OP_READ

SelectionKey.OP_WRITE

NIO通信模型图解:

(虚线表示不直接相关联)

下面用代码来演示一下Buffer、Channel、Selector的使用。

以IntBuffer为例,讲解一下Buffer的常用API:

 

 
  1. public class IntBufferTest {

  2. public static void main(String[] args) {

  3. //1、基本操作

  4. //创建指定长度的缓冲区

  5. /*IntBuffer buffer = IntBuffer.allocate(10);

  6. buffer.put(11); //position位置:0->1

  7. buffer.put(5); //position位置:1->2

  8. buffer.put(32); //position位置:2->3

  9. System.out.println("未调用flip复位方法前的buffer:" + buffer);

  10. //把位置复位为0,也就是position位置由3->0

  11. buffer.flip();

  12. //比较未调用flip方法和调用之后buffer的limit可以发现,不进行复位操作的话,position的值为3,limit的值为10

  13. // 因为缓冲区中已有11、5、32三个元素,也就意味着put()方法会使position向后递增1

  14. System.out.println("调用flip复位方法后的buffer:" + buffer);

  15. System.out.println("buffer容量为:" + buffer.capacity());

  16. System.out.println("buffer限制为:" + buffer.limit());

  17. System.out.println("获取下标为1的元素:" + buffer.get(1));

  18. System.out.println("调用get(index)方法后的buffer:" + buffer); //调用get(index)方法,不会改变position的值

  19. buffer.put(1, 4); //将buffer位置为1的值替换为4,调用put(index,value)不会改变position的值

  20. System.out.println("调用put(index, value)方法后的buffer:" + buffer);

  21.  
  22. for(int i=0; i<buffer.limit(); i++) {

  23. //调用get方法会使缓冲区的位置(position)向后递增一位

  24. System.out.print(buffer.get() + "\t");

  25. }

  26. System.out.println("\nbuffer对象遍历之后buffer为:" + buffer);*/

  27.  
  28.  
  29. //2、wrap方法的使用

  30. /*int[] arr = new int[]{1, 2, 3};

  31. IntBuffer buffer = IntBuffer.wrap(arr);

  32. System.out.println("wrap(arr)方法:" + buffer);

  33. //IntBuffer.wrap(array, postion, length)表示容量为array的长度,但是可操作的元素为位置postion到length的数组元素

  34. buffer = IntBuffer.wrap(arr, 0, 2);

  35. System.out.println("wrap(arr, 0, 2):" + buffer);*/

  36.  
  37. //3、其他方法

  38. IntBuffer buffer = IntBuffer.allocate(10);

  39. int[] arr = new int[]{1, 2, 3};

  40. buffer.put(arr);

  41. System.out.println("调用put(arr)方法后的buffer:" + buffer);

  42. //一种复制方法,buffer1的pos、lim、cap与buffer的一样

  43. IntBuffer buffer1 = buffer.duplicate();

  44. System.out.println("buffer1:" + buffer1);

  45.  
  46. buffer.position(1); //将buffer的position设置为1,不建议使用。功能相当于flip()方法,但是从运行结果可以看出,lim依然等于10

  47. System.out.println("调用position()方法后的buffer:" + buffer);

  48. System.out.println("buffer的可读数据量:" + buffer.remaining()); //计算出从pos到lim的长度

  49. int[] arr1 = new int[buffer.remaining()];

  50. //将缓冲区的数据放入arr1中

  51. buffer.get(arr1);

  52. for(Integer i : arr1) {

  53. System.out.print(Integer.toString(i) + ",");

  54. }

  55. System.out.println();

  56.  
  57. //比较flip()方法和position(index)方法的区别

  58. buffer1.flip();

  59. System.out.println("buffer1的可读数量:" + buffer1.remaining());

  60. arr1 = new int[buffer1.remaining()];

  61. buffer1.get(arr1);

  62. for(Integer i : arr1) {

  63. System.out.print(Integer.toString(i) + ",");

  64. }

  65. }

  66. }

运行结果:

 

接下来是Buffer、Channel、Selector的一个入门的小例子:

Server端:

 

 
  1. public class Server implements Runnable {

  2.  
  3. private Selector selector;

  4. private ByteBuffer buffer = ByteBuffer.allocate(1024);

  5.  
  6. public Server(int port) {

  7. try {

  8. //1 打开多复用器

  9. selector = Selector.open();

  10. //2 打开服务器通道

  11. ServerSocketChannel ssc = ServerSocketChannel.open();

  12. //3 设置服务器通道为非阻塞方式

  13. ssc.configureBlocking(false);

  14. //4 绑定地址

  15. ssc.bind(new InetSocketAddress(port));

  16. //5 把服务器通道注册到多路复用选择器上,并监听阻塞状态

  17. ssc.register(selector, SelectionKey.OP_ACCEPT);

  18. System.out.println("Server start, port:" + port);

  19. } catch (IOException e) {

  20. e.printStackTrace();

  21. }

  22. }

  23.  
  24. @Override

  25. public void run() {

  26. while (true) {

  27. try {

  28. //1 必须让多路复用选择器开始监听

  29. selector.select();

  30. //2 返回所有已经注册到多路复用选择器上的通道的SelectionKey

  31. Iterator<SelectionKey> keys = selector.selectedKeys().iterator();

  32. //3 遍历keys

  33. while (keys.hasNext()) {

  34. SelectionKey key = keys.next();

  35. keys.remove();

  36. if(key.isValid()) { //如果key的状态是有效的

  37. if(key.isAcceptable()) { //如果key是阻塞状态,则调用accept()方法

  38. accept(key);

  39. }

  40. if(key.isReadable()) { //如果key是可读状态,则调用read()方法

  41. read(key);

  42. }

  43. }

  44. }

  45. } catch (IOException e) {

  46. e.printStackTrace();

  47. }

  48. }

  49. }

  50.  
  51. private void accept(SelectionKey key) {

  52. try {

  53. //1 获取服务器通道

  54. ServerSocketChannel ssc = (ServerSocketChannel) key.channel();

  55. //2 执行阻塞方法

  56. SocketChannel sc = ssc.accept();

  57. //3 设置阻塞模式为非阻塞

  58. sc.configureBlocking(false);

  59. //4 注册到多路复用选择器上,并设置读取标识

  60. sc.register(selector, SelectionKey.OP_READ);

  61. } catch (Exception e) {

  62. e.printStackTrace();

  63. }

  64. }

  65.  
  66. private void read(SelectionKey key) {

  67. try {

  68. //1 清空缓冲区中的旧数据

  69. buffer.clear();

  70. //2 获取之前注册的SocketChannel通道

  71. SocketChannel sc = (SocketChannel) key.channel();

  72. //3 将sc中的数据放入buffer中

  73. int count = sc.read(buffer);

  74. if(count == -1) { // == -1表示通道中没有数据

  75. key.channel().close();

  76. key.cancel();

  77. return;

  78. }

  79. //读取到了数据,将buffer的position复位到0

  80. buffer.flip();

  81. byte[] bytes = new byte[buffer.remaining()];

  82. //将buffer中的数据写入byte[]中

  83. buffer.get(bytes);

  84. String body = new String(bytes).trim();

  85. System.out.println("Server:" + body);

  86. } catch (Exception e) {

  87. e.printStackTrace();

  88. }

  89. }

  90.  
  91. public static void main(String[] args) {

  92. new Thread(new Server(8379)).start();

  93. }

  94. }

客户端:

 

 

 
  1. public class Client {

  2. public static void main(String[] args) {

  3. InetSocketAddress address = new InetSocketAddress("127.0.0.1", 8379);

  4. SocketChannel sc = null;

  5. ByteBuffer buffer = ByteBuffer.allocate(1024);

  6. try {

  7. //打开通道

  8. sc = SocketChannel.open();

  9. //建立连接

  10. sc.connect(address);

  11. while (true) {

  12. byte[] bytes = new byte[1024];

  13. System.in.read(bytes);

  14. //把输入的数据放入buffer缓冲区

  15. buffer.put(bytes);

  16. //复位操作

  17. buffer.flip();

  18. //将buffer的数据写入通道

  19. sc.write(buffer);

  20. //清空缓冲区中的数据

  21. buffer.clear();

  22. }

  23. } catch (Exception e) {

  24. e.printStackTrace();

  25. } finally {

  26. if(sc != null) {

  27. try {

  28. sc.close();

  29. } catch (IOException e) {

  30. e.printStackTrace();

  31. }

  32. }

  33. }

  34. }

  35. }

运行结果:

 



3、AIO

在NIO的基础上引入了异步通道的概念,并提供了异步文件和异步套接字通道的实现,从而在真正意义上实现了异步非阻塞,之前的NIO只是非阻塞而并非异步。AIO不需要通过对多路复用器对注册的通道进行轮询操作即可实现异步读写,从而简化NIO编程模型。

①AsynchronousServerSocketChannel

②AsynchronousSocketChannel

下面看代码:

Server端:

 

 
  1. public class Server {

  2. //线程池

  3. private ExecutorService executorService;

  4. //线程组

  5. private AsynchronousChannelGroup channelGroup;

  6. //服务器通道

  7. public AsynchronousServerSocketChannel channel;

  8.  
  9. public Server(int port) {

  10. try {

  11. //创建线程池

  12. executorService = Executors.newCachedThreadPool();

  13. //创建线程组

  14. channelGroup = AsynchronousChannelGroup.withCachedThreadPool(executorService, 1);

  15. //创建服务器通道

  16. channel = AsynchronousServerSocketChannel.open(channelGroup);

  17. //绑定地址

  18. channel.bind(new InetSocketAddress(port));

  19. System.out.println("server start, port:" + port);

  20. channel.accept(this, new ServerCompletionHandler());

  21. Thread.sleep(Integer.MAX_VALUE);

  22. } catch (Exception e) {

  23. e.printStackTrace();

  24. }

  25. }

  26.  
  27. public static void main(String[] args) {

  28. Server server = new Server(8379);

  29. }

  30. }

ServerCompletionHandler类:

 
  1. public class ServerCompletionHandler implements CompletionHandler<AsynchronousSocketChannel, Server> {

  2. @Override

  3. public void completed(AsynchronousSocketChannel channel, Server attachment) {

  4. //当有下一个客户端接入的时候,直接调用Server的accept方法,这样反复执行下去,保证多个客户端都可以阻塞

  5. attachment.channel.accept(attachment, this);

  6. read(channel);

  7. }

  8.  
  9. private void read(AsynchronousSocketChannel channel) {

  10. //读取数据

  11. ByteBuffer buffer = ByteBuffer.allocate(1024);

  12. channel.read(buffer, buffer, new CompletionHandler<Integer, ByteBuffer>() {

  13. @Override

  14. public void completed(Integer resultSize, ByteBuffer attachment) {

  15. attachment.flip();

  16. System.out.println("Server->" + "收到客户端发送的数据长度为:" + resultSize);

  17. String data = new String(buffer.array()).trim();

  18. System.out.println("Server->" + "收到客户端发送的数据为:" + data);

  19. String response = "服务器端响应了客户端。。。。。。";

  20. write(channel, response);

  21. }

  22.  
  23. @Override

  24. public void failed(Throwable exc, ByteBuffer attachment) {

  25. exc.printStackTrace();

  26. }

  27. });

  28. }

  29.  
  30. private void write(AsynchronousSocketChannel channel, String response) {

  31. try {

  32. ByteBuffer buffer = ByteBuffer.allocate(1024);

  33. buffer.put(response.getBytes());

  34. buffer.flip();

  35. channel.write(buffer).get();

  36. } catch (Exception e) {

  37. e.printStackTrace();

  38. }

  39. }

  40.  
  41. @Override

  42. public void failed(Throwable exc, Server attachment) {

  43. exc.printStackTrace();

  44. }

  45. }

客户端:

 
  1. public class Client implements Runnable {

  2.  
  3. private AsynchronousSocketChannel channel;

  4.  
  5. public Client() throws IOException {

  6. channel = AsynchronousSocketChannel.open();

  7. }

  8.  
  9. public void connect() {

  10. channel.connect(new InetSocketAddress("127.0.0.1", 8379));

  11. }

  12.  
  13. public void write(String data) {

  14. try {

  15. channel.write(ByteBuffer.wrap(data.getBytes())).get();

  16. read();

  17. } catch (Exception e) {

  18. e.printStackTrace();

  19. }

  20. }

  21.  
  22. public void read() {

  23. ByteBuffer buffer = ByteBuffer.allocate(1024);

  24. try {

  25. channel.read(buffer).get();

  26. buffer.flip();

  27. byte[] bytes = new byte[buffer.remaining()];

  28. buffer.get(bytes);

  29. String data = new String(bytes, "UTF-8").trim();

  30. System.out.println(data);

  31. } catch (Exception e) {

  32. e.printStackTrace();

  33. }

  34. }

  35.  
  36. @Override

  37. public void run() {

  38. while (true) {

  39.  
  40. }

  41. }

  42.  
  43. public static void main(String[] args) {

  44. try {

  45. Client c1 = new Client();

  46. Client c2 = new Client();

  47. Client c3 = new Client();

  48.  
  49. c1.connect();

  50. c2.connect();

  51. c3.connect();

  52.  
  53. new Thread(c1).start();

  54. new Thread(c2).start();

  55. new Thread(c3).start();

  56.  
  57. Thread.sleep(1000);

  58.  
  59. c1.write("c1 aaa");

  60. c2.write("c2 bbbb");

  61. c3.write("c3 ccccc");

  62. } catch (Exception e) {

  63. e.printStackTrace();

  64. }

  65. }

  66. }

运行结果:

 

  • 4
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值