java网络编程笔记(JAVA网络编程笔记)

2023-11-24 19:44:37 数码极客 bianji01

 

Java网络编程从入门到精通(18):Socket类的getter和setter方法(2)

二 用于获得和设置Socket选项的getter和setter方法Socket选择可以指定Socket类发送和接受数据的方式 在JDK *** 有 个Socket选择可以设置 这 个选项都定义在 SocketOptions接口中 定义如下public?final?static?int?TCP_NODELAY?=? x ;public?final?static?int?SO_REUSEADDR?=? x ;public?final?static?int?SO_LINGER?=? x ;public?final?static?int?SO_TIMEOUT?=? x ;public?final?static?int?SO_SNDBUF?=? x ;public?final?static?int?SO_RCVBUF?=? x ;public?final?static?int?SO_KEEPALIVE?=? x ;public?final?static?int?SO_OOBINLINE?=? x ;有趣的是 这 个选项除了第一个没在SO前缀外 其他 个选项都以SO作为前缀 其实这个SO就是Socket Option的缩写 因此 在Java中约定所有以SO为前缀的常量都表示Socket选项 当然 也有例外 如TCP_NODELAY 在Socket类中为每一个选项提供了一对get和set方法 分别用来获得和设置这些选项TCP_NODELAYpublic?boolean?getTcpNoDelay()?throws?SocketExceptionpublic?void?setTcpNoDelay(boolean?on)?throws?SocketException在默认情况下 客户端向服务器发送数据时 会根据数据包的大小决定是否立即发送 当数据包中的数据很少时 如只有 个字节 而数据包的头却有几十个字节(IP头+TCP头)时 系统会在发送之前先将较小的包合并到软大的包后 一起将数据发送出去 在发送下一个数据包时 系统会等待服务器对前一个数据包的响应 当收到服务器的响应后 再发送下一个数据包 这就是所谓的Nagle算法 在默认情况下 Nagle算法是开启的这种算法虽然可以有效地改善网络传输的效率 但对于网络速度比较慢 而且对实现性的要求比较高的情况下(如游戏 Telnet等) 使用这种方式传输数据会使得客户端有明显的停顿现象 因此 最好的解决方案就是需要Nagle算法时就使用它 不需要时就关闭它 而使用setTcpToDelay正好可以满足这个需求 当使用setTcpNoDelay(true)将Nagle算法关闭后 客户端每发送一次数据 无论数据包的大小都会将这些数据发送出去SO_REUSEADDRpublic?boolean?getReuseAddress()?throws?SocketExceptionpublic?void?setReuseAddress(boolean?on)?throws?SocketException通过这个选项 可以使多个Socket对象绑定在同一个端口上 其实这样做并没有多大意义 但当使用close方法关闭Socket连接后 Socket对象所绑定的端口并不一定马上释放 系统有时在Socket连接关闭才会再确认一下是否有因为延迟面未到达的数据包 这完全是在底层处理的 也就是说对用户是透明的 因此 在使用Socket类时完全不会感觉到这种处理机制对于随机绑定端口的Socket对象没有什么影响 但对于绑定在固定端口的Socket对象就可能会抛出 Address already in use JVM_Bind 例外 因此 使用这个选项可以避免个例外的发生package mynet;import? *;import?java io *;public?class Test{public?static?void?main(String[]?args){Socket?socket ?=?new?Socket();Socket?socket ?=?new?Socket();try{socket setReuseAddress(true);socket bind(new?InetSocketAddress());System out println( socket getReuseAddress():+?socket getReuseAddress());socket bind(new?InetSocketAddress());}catch?(Exception?e){System out println( error: ?+?e getMessage());try{socket setReuseAddress(true);socket bind(new?InetSocketAddress());System out println( socket getReuseAddress():+?socket getReuseAddress());System out println( 端口 第二次绑定成功! );}catch?(Exception?e ){System out println(e getMessage());}}}}上面的代码的运行结果如下socket getReuseAddress():trueerror:Address?already?in?use:?JVM_Bindsocket getReuseAddress():true端口 第二次绑定成功!使用SO_REUSEADDR选项时有两点需要注意必须在调用bind方法之前使用setReuseAddress方法来打开SO_REUSEADDR选项 因此 要想使用SO_REUSEADDR选项 就不能通过Socket类的构造方法来绑定端口必须将绑定同一个端口的所有的Socket对象的SO_REUSEADDR选项都打开才能起作用 如在例程 中 socket 和socket 都使用了setReuseAddress方法打开了各自的SO_REUSEADDR选项SO_LINGERpublic?int?getSoLinger()?throws?SocketExceptionpublic?void?setSoLinger(boolean?on ?int?linger)?throws?SocketException这个Socket选项可以影响close方法的行为 在默认情况下 当调用close方法后 将立即返回 如果这时仍然有未被送出的数据包 那么这些数据包将被丢弃 如果将linger参数设为一个正整数n时(n的值最大是 ) 在调用close方法后 将最多被阻塞n秒 在这n秒内 系统将尽量将未送出的数据包发送出去 如果超过了n秒 如果还有未发送的数据包 这些数据包将全部被丢弃 而close方法会立即返回 如果将linger设为 和关闭SO_LINGER选项的作用是一样的如果底层的Socket实现不支持SO_LINGER都会抛出SocketException例外 当给linger参数传递负数值时 setSoLinger还会抛出一个IllegalArgumentException例外 可以通过getSoLinger方法得到延迟关闭的时间 如果返回 则表明SO_LINGER是关闭的 例如 下面的代码将延迟关闭的时间设为 分钟if(socket getSoLinger()?==? )?socket setSoLinger(true);SO_TIMEOUTpublic?int?getSoTimeout()?throws?SocketExceptionpublic?void?setSoTimeout(int?timeout)?throws?SocketException这个Socket选项在前面已经讨论过 可以通过这个选项来设置读取数据超时 当输入流的read方法被阻塞时 如果设置timeout(timeout的单位是毫秒) 那么系统在等待了timeout毫秒后会抛出一个InterruptedIOException例外 在抛出例外后 输入流并未关闭 你可以继续通过read方法读取数据如果将timeout设为 就意味着read将会无限等待下去 直到服务端程序关闭这个Socket 这也是timeout的默认值 如下面的语句将读取数据超时设为 秒socket setSoTimeout( ?*? );当底层的Socket实现不支持SO_TIMEOUT选项时 这两个方法将抛出SocketException例外 不能将timeout设为负数 否则setSoTimeout方法将抛出IllegalArgumentException例外SO_SNDBUFpublic?int?getSendBufferSize()?throws?SocketExceptionpublic?void?setSendBufferSize(int?size)?throws?SocketException在默认情况下 输出流的发送缓冲区是 个字节( K) 这个值是Java所建议的输出缓冲区的大小 如果这个默认值不能满足要求 可以用setSendBufferSize方法来重新设置缓冲区的大小 但最好不要将输出缓冲区设得太小 否则会导致传输数据过于频繁 从而降低网络传输的效率如果底层的Socket实现不支持SO_SENDBUF选项 这两个方法将会抛出SocketException例外 必须将size设为正整数 否则setSendBufferedSize方法将抛出IllegalArgumentException例外SO_RCVBUFpublic?int?getReceiveBufferSize()?throws?SocketExceptionpublic?void?setReceiveBufferSize(int?size)?throws?SocketException在默认情况下 输入流的接收缓冲区是 个字节( K) 这个值是Java所建议的输入缓冲区的大小 如果这个默认值不能满足要求 可以用setReceiveBufferSize方法来重新设置缓冲区的大小 但最好不要将输入缓冲区设得太小 否则会导致传输数据过于频繁 从而降低网络传输的效率如果底层的Socket实现不支持SO_RCVBUF选项 这两个方法将会抛出SocketException例外 必须将size设为正整数 否则setReceiveBufferSize方法将抛出IllegalArgumentException例外SO_KEEPALIVEpublic?boolean?getKeepAlive()?throws?SocketExceptionpublic?void?setKeepAlive(boolean?on)?throws?SocketException如果将这个Socket选项打开 客户端Socket每隔段的时间(大约两个小时)就会利用空闲的连接向服务器发送一个数据包 这个数据包并没有其它的作用 只是为了检测一下服务器是否仍处于活动状态 如果服务器未响应这个数据包 在大约 分钟后 客户端Socket再发送一个数据包 如果在 分钟内 服务器还没响应 那么客户端Socket将关闭 如果将Socket选项关闭 客户端Socket在服务器无效的情况下可能会长时间不会关闭 SO_KEEPALIVE选项在默认情况下是关闭的 可以使用如下的语句将这个SO_KEEPALIVE选项打开socket setKeepAlive(true);SO_OOBINLINEpublic?boolean?getOOBInline()?throws?SocketException?public?void?setOOBInline(boolean?on)?throws?SocketException如果这个Socket选项打开 可以通过Socket类的sendUrgentData方法向服务器发送一个单字节的数据 这个单字节数据并不经过输出缓冲区 而是立即发出 虽然在客户端并不是使用OutputStream向服务器发送数据 但在服务端程序中这个单字节的数据是和其它的普通数据混在一起的 因此 在服务端程序中并不知道由客户端发过来的数据是由OutputStream还是由sendUrgentData发过来的 下面是sendUrgentData方法的声明public?void?sendUrgentData(int?data)?throws?IOException虽然sendUrgentData的参数data是int类型 但只有这个int类型的低字节被发送 其它的三个字节被忽略 下面的代码演示了如何使用SO_OOBINLINE选项来发送单字节数据package mynet;import? *;import?java io *;class Server{public?static?void?main(String[]?args)?throws?Exception{ServerSocket?serverSocket?=?new?ServerSocket( );System out println( 服务器已经启动 端口号 );while?(true){Socket?socket?=?serverSocket accept();socket setOOBInline(true);InputStream?in?=?socket getInputStream();InputStreamReader?inReader?=?new?InputStreamReader(in);BufferedReader?bReader?=?new?BufferedReader(inReader);System out println(bReader readLine());System out println(bReader readLine());socket close();}}}public?class Client{public?static?void?main(String[]?args)?throws?Exception{Socket?socket?=?new?Socket();socket setOOBInline(true);OutputStream?out?=?socket getOutputStream();OutputStreamWriter?outWriter?=?new?OutputStreamWriter(out);outWriter write( );//?向服务器发送字符 CoutWriter write( hello?world\r\n );socket sendUrgentData( );//?向服务器发送字符 Asocket sendUrgentData( );//?向服务器发送字符 BoutWriter flush();socket sendUrgentData( );//?向服务器发送汉字 中socket sendUrgentData( );socket sendUrgentData( );//?向服务器发送汉字 国socket sendUrgentData( );socket close();}}由于运行上面的代码需要一个服务器类 因此 在加了一个类名为Server的服务器类 关于服务端套接字的使用方法将会在后面的文章中详细讨论 在类Server类中只使用了ServerSocket类的accept方法接收客户端的请求 并从客户端传来的数据中读取两行字符串 并显示在控制台上由于本例使用了 因Server和Client类必须在同一台机器上运行运行Serverjava?mynet Server运行Clientjava?mynet Client在服务端控制台的输出结果服务器已经启动 端口号 ABChello?world中国在ClienT类中使用了sendUrgentData方法向服务器发送了字符 A ( )和 B ( ) 但发送 B 时实际发送的是 由于sendUrgentData只发送整型数的低字节 因此 实际发送的是 十进制整型 的二进制形式如图 所示图十进制整型 的二进制形式从图 可以看出 虽然 分布在了两个字节上 但它的低字节仍然是在Client类中使用flush将缓冲区中的数据发送到服务器 我们可以从输出结果发现一个问题 在Client类中先后向服务器发送了 C hello world r n A B 而在服务端程序的控制台上显示的却是ABChello world 这种现象说明使用sendUrgentData方法发送数据后 系统会立即将这些数据发送出去 而使用write发送数据 必须要使用flush方法才会真正发送数据在Client类中向服务器发送 中国 字符串 由于 中 是由 和 两个字节组成的 而 国 是由 和 两个字节组成的 因此 可分别发送这四个字节来传送 中国 字符串lishixinzhi/Article/program/Java/hx/201311/26387

JAVA网络编程

java的大方向就是j2ee

j2ee不仅仅是socket编程,具体包括13中核心技术。

J2EE的核心API与组件

J2EE平台由一整套服务(Services)、应用程序接口(APIs)和协议构成,它对开发基于Web的多层应用提供了功能支持,下面对J2EE中的13种技术规范进行简单的描述(限于篇幅,这里只能进行简单的描述):

1.JDBC(JavaDatabaseConnectivity):

JDBCAPI为访问不同的数据库提供了一种统一的途径,象ODBC一样,JDBC对开发者屏蔽了一些细节问题,另外,JDCB对数据库的访问也具有平台无关性。

2.JNDI(JavaNameandDirectoryInterface):

JNDIAPI被用于执行名字和目录服务。它提供了一致的模型来存取和操作企业级的资源如DNS和LDAP,本地文件系统,或应用服务器中的对象。

3.EJB(EnterpriseJavaBean):

J2EE技术之所以赢得媒体广泛重视的原因之一就是EJB。它们提供了一个框架来开发和实施分布式商务逻辑,由此很显著地简化了具有可伸缩性和高度复杂的企业级应用的开发。EJB规范定义了EJB组件在何时如何与它们的容器进行交互作用。容器负责提供公用的服务,例如目录服务、事务管理、安全性、资源缓冲池以及容错性。但这里值得注意的是,EJB并不是实现J2EE的唯一途径。正是由于J2EE的开放性,使得有的厂商能够以一种和EJB平行的方式来达到同样的目的。

4.RMI(RemoteMethodInvoke):

正如其名字所表示的那样,RMI协议调用远程对象上方法。它使用了序列化方式在客户端和服务器端传递数据。RMI是一种被EJB使用的更底层的协议。

5.JavaIDL/CORBA:

在JavaIDL的支持下,开发人员可以将Java和CORBA集成在一起。他们可以创建Java对象并使之可在CORBAORB中展开,或者他们还可以创建Java类并作为和其它ORB一起展开的CORBA对象的客户。后一种方法提供了另外一种途径,通过它Java可以被用于将你的新的应用和旧的系统相集成。

6.JSP(JavaServerPages):

JSP页面由HTML代码和嵌入其中的Java代码所组成。服务器在页面被客户端所请求以后对这些Java代码进行处理,然后将生成的HTML页面返回给客户端的浏览器。

7.JavaServlet:

Servlet是一种小型的Java程序,它扩展了Web服务器的功能。作为一种服务器端的应用,当被请求时开始执行,这和CGIPerl脚本很相似。Servlet提供的功能大多与JSP类似,不过实现的方式不同。JSP通常是大多数HTML代码中嵌入少量的Java代码,而servlets全部由Java写成并且生成HTML。

8.XML(ExtensibleMarkupLanguage):

XML是一种可以用来定义其它标记语言的语言。它被用来在不同的商务过程中共享数据。

XML的发展和Java是相互独立的,但是,它和Java具有的相同目标正是平台独立性。通过将Java和XML的组合,您可以得到一个完美的具有平台独立性的解决方案。

9.JMS(JavaMessageService):

MS是用于和面向消息的中间件相互通信的应用程序接口(API)。它既支持点对点的域,有支持发布/订阅(publish/subscribe)类型的域,并且提供对下列类型的支持:经认可的消息传递,事务型消息的传递,一致性消息和具有持久性的订阅者支持。JMS还提供了另

一种方式来对您的应用与旧的后台系统相集成。

10.JTA(JavaTransactionArchitecture):

JTA定义了一种标准的API,应用系统由此可以访问各种事务监控。

11.JTS(JavaTransactionService):

JTS是CORBAOTS事务监控的基本的实现。JTS规定了事务管理器的实现方式。该事务管理器是在高层支持JavaTransactionAPI(JTA)规范,并且在较底层实现OMGOTSspecification的Java映像。JTS事务管理器为应用服务器、资源管理器、独立的应用以及通信资源管理器提供了事务服务。

12.JavaMail:

JavaMail是用于存取邮件服务器的API,它提供了一套邮件服务器的抽象类。不仅支持SMTP服务器,也支持IMAP服务器。

13.JAF(JavaBeansActivationFramework):

JavaMail利用JAF来处理MIME编码的邮件附件。MIME的字节流可以被转换成Java对象,或者转换自Java对象。大多数应用都可以不需要直接使用JAF。

第一个阶段(java基础阶段)

1.java语法

2.面向对象

3.常用的api

4.界面编程

5.多线程

6.文件io

7.java网络编程..

看看张孝祥老师的java视频(不过张孝

祥老师普通话不是很好,而且语速很慢,不过技术是不用说的啦!感谢张老师的无私奉献了,呵呵)

第二个阶段(数据库阶段)

1.oracle

2.mysql

3.sqlserver

目前中国软件公司用的最多的三大主流数据库是sqlserver,mysql,oracle.目前看来oracle数据库越来越流行了。一般情况下PHP对应MySQL数据库;ASP对应SQL数据库;JSP对应ORACLE数据库。所以大家会经常看

到市场上JAVA培训都是用的ORACLE数据库,其实你学会了ORACLE数据库,再去学习MySQL、SQL数据库,你会觉得有很多相似的之处,学起来也容易上手了。oracle我推荐

韩顺平老师oracle视频(韩老师授课很有趣,特适合基础不太好的人。讲课有一套)

第三个阶段(web开发阶段)

1.html

2.css

3.javascript

这三个部分是进行web开发的必须技术.一定要好好学习.不过我当时没有发现有讲的特别好的。好像很多老师都讲框架和后台,界面这块不多,所以就没有什么好推荐的了,大家可以

在网上找找看看,<<别具光芒>>这本书讲的挺好,不过不太适合初学者,可以看看孙鑫老师视频的《HTML语言速成》。

第四个阶段(j2ee中级部分)

1.servlet

2.jsp

3.mvc

这个阶段是接近企业的需求了,所以学习起来,难度偏大,如果前面没有学习扎实,会比较麻烦,我就是这样的,前面没有吃透,就学习j2ee中级,搞得我消化不良。所以小弟我中心的告诫大家,把前面的学好,再学习第四个阶段,这里的mvc思想不太好理解,是学习的重点。同时要学习tomcat/jboss/wl这些服务器所以,你可能感到要学习的东西一下多了很多,其实我就在这里动摇过,有那么一段时间,又想放弃了。可是一想到房东那种不屑的表情,我就想,一定要好好学习,混出个人样来。我家是农村的,回去就是种地,当公务员我没有关系,再说会老家我学习的专业也用不上,要不就是当个网管什么的,不说了,反正当时心情很复杂吧。这个阶段我推荐

韩顺平老师servlet视频jsp视频(这里也特别谢谢韩顺平老师,我是比较喜欢他讲课的风格,因为我当时基础不好,所以不敢看讲的太理论的课。)

第五个阶段(j2ee高级部分)

1.struts

2.hibernate

3.spring

学习完Servlet、JSP然后再去学习框架Struts、Hibernate、Spring等最前沿的最流行的网络编程必备的软件技能。

推荐的参考书籍是美河图书提供《Servlet与JSP核心编程》

框架的基础就是Servlet、JSP。首先大家应该学习的是Struts框架,典型的MVC模型。推荐学习视频是张小静的Struts视频。(网上有孙鑫的视频嘛,主要就是讲的Struts、Hibernate、Spring这三个框架,但是这个好像是偷录的效果不好,听起来比较吃力,衔接的不好)

推荐书籍孙卫琴编著《精通Struts基于MVC的Java.Web设计与开发》

然后就是Hibernate框架,推荐李兴华老师的Hibernate视频,

参考书籍孙卫琴《精通Java对象持久化技术详解》

最后关于Spring框架的视频,在网上你可以搜索到很多,但是系统讲解的真的没有发现,本着对大家负责的态度,不敢乱推荐,大家到网上搜一下,也许现在有了也可能。

学完上述三个阶段的内容,就在看看xml,ajax,ejb这些知识!尤其是的Ajax用得非常火,我们公司就用,ajax的框架流行的是(jquery,dw),我们公司用的是jquery.这里提一下,学习Ajax之前一定要有JavaScript的基础,推荐视频张孝祥JavaScript网页开发,这套视频有相应的配套书籍《JavaScript网页开发》

本回答由提问者推荐

java的网络编程有哪些方面?

Servlet JSP MVC...

java的网络编程有哪些方面?

Servlet JSP MVC...

声明:易趣百科所有作品(图文、音视频)均由用户自行上传分享,仅供网友学习交流。若您的权利被侵害,请联系315127732@qq.com
广告位招租
横幅广告