如何应用DataGram进行用户-服务器编程

简介:


Java编程艺术》章节选登。作者:高永强 清华大学出版社 (即将出版)


23.2.5  Datagram编程 (1)

数据报表 Datagram ,或称数报式数据传输技术,利用 UDP 通讯协议( User Datagram Protocol ),进行用户 - 服务器间的数据传递,但 Java 虚拟机将 UDP  底层通讯细节隐藏,编程人员不必顾及其通讯协议和过程,只需利用 java.net 包中提供的 API DatagramSocket DatagramPacket 进行程序设计,调用适当的方法,实现用户 - 服务器编程。其中 DatagrameScoket 用来创建端口间的通讯,而 DatagramPacket 用来获取通过网络地址和端口以邮包方式( Packet )发送来的信息。表 23.4 列出了 java.net 包中 DataramSocket DataramPacket 的常用构造器以及方法。
23.4  DatagramSocket DatagramPacket 类的常用构造器以及方法
构造器 / 方法
    
DatagramSocket(int port,InetAdrress  address)
按指定端口和互联网地址创建对象。
  close()
关闭插座连接。
  connect(InetAddress address, int port)
按指定互联网地址和端口连接。
  disconnect()
断开当前的连接。
  InetAddress getInetAddress()
返回当前数报插座的互联网地址。
  InetAddress getLocalAddress()
返回当前数报插座的本机地址
  int getPort()
返回当前数报插座的连接端口。
  int getLocalPort()
返回当前数报插座的本机连接端口。
  receive(DatagramPacket packet)
接收当前数报插座的邮件。
  send(DatagramPacket packet)
发送当前数报插座的邮件。
DatagramPacket(byte[] buf, int length)
按指定缓冲数组合长度创建获取邮包的对象。
  InetAddress getAddress()
返回当前进行邮包传送的互联网地址。
  Byte[] getData()
返回当前发送或接收数据缓冲数组。
  int getLength()
返回当前发送或接收数据的长度。
  int getPort()
返回当前发送或接收数据的端口。
 
注意   DatagramSocket DatagramPacket 抛出检查性异常,程序中必须提供处理这些异常的代码。具体实例见下面的讨论。
 
下面的例子利用 DatagramSocket DatagramPacket ,模拟用户 - 服务器通讯,将用户        的英文输入,通过邮包发送到服务器端程序,转换为大写字母,并将结果传回到用户屏       幕。其功能类似于在 Socket ServerSocket 讨论过的程序,但增加了统计并返回邮包长度的操作。图 23.7 显示了这个例子的一个典型运行结果。图上方为服务器端程序运行后的 截图。(注:截图未能显示。请参考原书)
23.7   利用 Datagram 的典型运行结果
如下是利用 Datagram 编写的服务器端程序的代码:
 
// 这个程序存在本书配套资源目录 Ch23 名为 DatagramServerTest.java
import java.io. * ;
import java.net. * ;
public class DatagramServerTest {
    public static void main(String[] args) {
        System.out.println(\"Welcome! The server is running....\");
        String line = \"Datagram packet from server: I love Java programming.\\n\";
        String promptString = line.toUpperCase() + \"Enter quit to STOP\";
        try {
             DatagramSocket socket = new DatagramSocket(1688);
                                                // 创建指定端口的 Datagram
             DatagramPacket receivePacket;      // 声明接收邮包
             byte[] buf = new byte[256];        // 缓冲器
             receivePacket = new DatagramPacket(buf, buf.length);
                                                // 创建接收邮包
             socket.receive(receivePacket);     // 接收邮包
             buf = promptString.getBytes();     // 内容至缓冲
             InetAddress address = receivePacket.getAddress();
                                                // 得到接收地址
             int port = receivePacket.getPort();// 得到接收端口
             sending(socket, buf, buf.length, address, port);
                                                // 调用发送邮包方法
             while (true) {
                 buf = new byte[256];        // 清除缓冲
                 receivePacket = new DatagramPacket(buf, buf.length);
                                                // 创建新邮包
                 socket.receive(receivePacket); // 接收
                 String receive = new String(receivePacket.getData());                                                  // 得到邮包内容
                 buf = receive.toUpperCase().getBytes();
                                                // 内容转成大写并送往缓冲
                 sending(socket, buf, buf.length, address, port);// 发送
                 buf = new byte[256];           // 清除缓冲
                 String wordCount = \"(Converting from server and packet length: \" + receive.trim().length() + \")\";
                 receivePacket = new DatagramPacket(buf, buf.length);                                                   // 创建新邮包
                 socket.receive(receivePacket); // 接收
                 buf = wordCount.getBytes();    // 发件内容送往缓冲
                 sending(socket, buf, buf.length, address, port);
                                                // 调用发送方法
             }
        }
        catch (IOException e) {
             e.printStackTrace();
        }
     }
    // 发送邮件方法
    public static void sending(DatagramSocket socket, byte[] buf, int length, InetAddress address, int port) {
            DatagramPacket sendPacket = new DatagramPacket(buf, length,                 address, port);
            try {
                socket.send(sendPacket);        // 发送
            }catch (IOException e) {
                e.printStackTrace();
            }
    }
 }
 
代码中首先接收用户端发送过来的一个空邮包,并利用这个邮包发送慰问和提示信息到用户。在循环中,接收用户发来的邮包内容,并将其转换成大写字母、统计字符串即邮包长度,调用自定义静态方法 sending() 将结果邮包发还给发来的用户。代码中在重新利用缓冲器发送新内容时,利用重新定义缓冲器来清除其原有内容。(待续



















本文转自高永强51CTO博客,原文链接: http://blog.51cto.com/yqgao/157391 ,如需转载请自行联系原作者
相关文章
|
Java
CompletableFuture在异常处理方面的一些常见问题和解决方案,建议牢记!
CompletableFuture在异常处理方面的一些常见问题和解决方案,建议牢记!
1146 0
CompletableFuture在异常处理方面的一些常见问题和解决方案,建议牢记!
|
存储 安全 Java
ArrayBlockingQueue 和 LinkedBlockingQueue 有什么区别?
ArrayBlockingQueue 和 LinkedBlockingQueue 有什么区别?
|
10天前
|
人工智能 运维 自然语言处理
阿里云Agent Infra香港首秀:170+机构到场,1700+人次互动
2025年5月21日,阿里云应邀出席由香港生产力促进局(HKPC)主办的"Innovating Public Services, Driving the AI Transformation" 大型人工智能主题盛会。本次活动汇聚香港政产学研各界力量,聚焦人工智能技术在政府治理、社会服务及产业转型中的创新应用。阿里云作为亚太地区领先的云计算与人工智能全栈服务商,深度参与本次活动,通过展台展示、技术分享及高层对话等多种形式,全面呈现阿里云在Agent Infra及智能体应用领域的最新成果,进一步夯实了阿里云在香港及粤港澳大湾区市场的影响力。
171 2
|
2月前
|
人工智能 运维 安全
别让“龙虾”裸奔!企业规模化“养虾”亟需新一代云网架构护航
本文深入分析了千级 AI Agent 规模化部署时面临的网络架构挑战,包括单 VPC 带宽天花板、安全组规则爆炸、混合云互通复杂等三大硬伤。基于阿里云 ACS+VPC+TR+CEN 的分层隔离架构,提供按业务域划分 VPC、TR 统一路由枢纽、CEN 全球互联的完整解决方案,实现性能隔离、故障隔离、安全合规、弹性扩展和成本优化五大核心价值。适用于 Agent 数量>500、多地域部署、强合规行业及混合云架构的企业场景。
457 6
别让“龙虾”裸奔!企业规模化“养虾”亟需新一代云网架构护航
|
存储 Linux 调度
【看表情包学Linux】系统下的文件操作 | 文件系统接口 | 系统调用与封装 | open,write,close 接口 | 系统传递标记位 O_RDWR,O_RDONLY,O_WRONLY...
【看表情包学Linux】系统下的文件操作 | 文件系统接口 | 系统调用与封装 | open,write,close 接口 | 系统传递标记位 O_RDWR,O_RDONLY,O_WRONLY...
491 1
|
人工智能 算法 网络安全
基于PAI+专属网关+私网连接:构建全链路Deepseek云上私有化部署与模型调用架构
本文介绍了阿里云通过PAI+专属网关+私网连接方案,帮助企业实现DeepSeek-R1模型的私有化部署。方案解决了算力成本高、资源紧张、部署复杂和数据安全等问题,支持全链路零公网暴露及全球低延迟算力网络,最终实现技术可控、成本优化与安全可靠的AI部署路径,满足企业全球化业务需求。
|
9月前
|
存储 运维 数据可视化
短剧为什么比长剧更依赖云和网络?
微短剧对云与网络的依赖远高于传统影视,因其制作周期短、投流驱动强、数据量大、IT团队轻量化及出海需求迫切。短剧的成功离不开高速传输、实时数据反馈、多云互联与跨境合规传输。一套理想的云网方案应具备高速文件传输、多云互联、跨境加速与合规、实时数据回传及可视化管理能力,助力短剧企业快速响应市场、提升投放效率、实现全球化分发。
|
容灾 关系型数据库 数据库
阿里云RDS服务巴黎奥运会赛事系统,助力云上奥运稳定运行
2024年巴黎奥运会,阿里云作为官方云服务合作伙伴,提供了稳定的技术支持。云数据库RDS通过备份恢复、实时监控、容灾切换等产品能力,确保了赛事系统的平稳运行。
 阿里云RDS服务巴黎奥运会赛事系统,助力云上奥运稳定运行
|
Java
java do while 的语法怎么用?
java do while 的语法怎么用?
516 3
|
算法
《黑神话:悟空》的关卡设计与优化技巧
【8月更文第26天】《黑神话:悟空》作为一款备受期待的动作冒险游戏,其关卡设计不仅需要吸引玩家的注意力,还要提供流畅且引人入胜的游戏体验。本文将探讨关卡设计的原则,并介绍一些优化技巧,以帮助提高玩家的沉浸感和游戏的整体质量。
728 0