/src/java/opentss/ChannelReader.java

http://open-tss.googlecode.com/ · Java · 175 lines · 126 code · 28 blank · 21 comment · 22 complexity · dda5c495c484b69006521ff615dca6a5 MD5 · raw file

  1. /**
  2. * Copyright 2002-2006 the original author or authors.
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License");
  5. * you may not use this file except in compliance with the License.
  6. * You may obtain a copy of the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS,
  12. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. * See the License for the specific language governing permissions and
  14. * limitations under the License.
  15. */
  16. package opentss;
  17. import java.io.ByteArrayInputStream;
  18. import java.io.EOFException;
  19. import java.io.IOException;
  20. import java.net.SocketException;
  21. import java.nio.ByteBuffer;
  22. import java.nio.channels.SocketChannel;
  23. import org.apache.commons.lang.StringUtils;
  24. /**
  25. * Reader on top of channel.
  26. *
  27. * @author <a href="mailto:max.h.chen@hotmail.com">Max Chen</a>
  28. */
  29. public class ChannelReader implements Reader {
  30. ReadLock lock = new ReadLock();
  31. private SocketChannel channel;
  32. private ByteBuffer readBuffer;
  33. private int soTimeout = 180000; // 3??
  34. public ChannelReader(SocketChannel channel, int readBufSize) {
  35. this.channel = channel;
  36. try {
  37. if (channel.socket().getSoTimeout() > 0)
  38. this.soTimeout = channel.socket().getSoTimeout();
  39. } catch (SocketException e) {
  40. // ignore;
  41. }
  42. this.readBuffer = ByteBuffer.allocate(readBufSize);
  43. }
  44. public byte readByte() throws IOException {
  45. byte[] bytes = new byte[1];
  46. read(bytes, 0, bytes.length);
  47. ByteArrayInputStream bis = new ByteArrayInputStream(bytes);
  48. return StreamUtil.readByte(bis);
  49. }
  50. public boolean readBoolean() throws IOException {
  51. byte[] bytes = new byte[1];
  52. read(bytes, 0, bytes.length);
  53. ByteArrayInputStream bis = new ByteArrayInputStream(bytes);
  54. return StreamUtil.readBoolean(bis);
  55. }
  56. public char readChar() throws IOException {
  57. byte[] bytes = new byte[2];
  58. read(bytes, 0, bytes.length);
  59. ByteArrayInputStream bis = new ByteArrayInputStream(bytes);
  60. return StreamUtil.readChar(bis);
  61. }
  62. public short readShort() throws IOException {
  63. byte[] bytes = new byte[2];
  64. read(bytes, 0, bytes.length);
  65. ByteArrayInputStream bis = new ByteArrayInputStream(bytes);
  66. return StreamUtil.readShort(bis);
  67. }
  68. public int readInt() throws IOException {
  69. byte[] bytes = new byte[4];
  70. read(bytes, 0, bytes.length);
  71. ByteArrayInputStream bis = new ByteArrayInputStream(bytes);
  72. return StreamUtil.readInt(bis);
  73. }
  74. public long readLong() throws IOException {
  75. byte[] bytes = new byte[8];
  76. read(bytes, 0, bytes.length);
  77. ByteArrayInputStream bis = new ByteArrayInputStream(bytes);
  78. return StreamUtil.readLong(bis);
  79. }
  80. public float readFloat() throws IOException {
  81. return Float.intBitsToFloat(readInt());
  82. }
  83. public double readDouble() throws IOException {
  84. return Double.longBitsToDouble(readLong());
  85. }
  86. public String readUTF() throws IOException {
  87. int strlen = readInt();
  88. if (strlen == -1)
  89. return null;
  90. else if (strlen == 0)
  91. return StringUtils.EMPTY;
  92. int utflen = readInt();
  93. StringBuffer sb = new StringBuffer(strlen);
  94. byte[] bytes = new byte[utflen];
  95. read(bytes, 0, utflen);
  96. StreamUtil.toString(bytes, utflen, sb);
  97. return sb.toString();
  98. }
  99. public int read(byte[] bytes, int offset, int length) throws IOException {
  100. if (bytes == null) {
  101. throw new NullPointerException();
  102. } else if (offset < 0 || length < 0 || length > bytes.length - offset) {
  103. throw new IndexOutOfBoundsException();
  104. } else if (length == 0) {
  105. return 0;
  106. }
  107. int read = 0;
  108. int capacity = readBuffer.capacity();
  109. long startTime = System.currentTimeMillis();
  110. while (length > 0 && (System.currentTimeMillis() - startTime) < soTimeout) {
  111. lock.readyToReadData(); // ??channel?????????????????????
  112. try {
  113. while (length > capacity) {
  114. while (readBuffer.hasRemaining())
  115. channel.read(readBuffer);
  116. readBuffer.rewind();
  117. readBuffer.get(bytes, offset, capacity);
  118. offset += capacity;
  119. read += capacity;
  120. length -= capacity;
  121. readBuffer.clear();
  122. }
  123. readBuffer.limit(length);
  124. while (readBuffer.hasRemaining())
  125. channel.read(readBuffer);
  126. readBuffer.flip();
  127. readBuffer.get(bytes, offset, length);
  128. read += length;
  129. length = 0;
  130. readBuffer.clear();
  131. }
  132. catch (EOFException e) {
  133. lock.blockedReading();
  134. }
  135. }
  136. if (length != 0)
  137. throw new SocketException("Expect " + (read + length) + " bytes in channel, but was " + read + " bytes.");
  138. return read;
  139. }
  140. public void close() {
  141. readBuffer = null;
  142. try {
  143. channel.close();
  144. } catch (Throwable t) {
  145. }
  146. }
  147. }