【达内课程】利用Socket实现群聊

2024-05-12 09:58
文章标签 实现 socket 课程 群聊

本文主要是介绍【达内课程】利用Socket实现群聊,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 思路
  • 实现

思路

【完成群聊天室】
服务端:
每当接收到客户端发来的消息后,需要给每一个在线的客户都输出一遍

List<Socket> sockets;main{socket= ss.accept();sockets.add(socket);new Thread(){run(){}}.start();
}

客户端:
建立连接、发送消息、接收并显示消息

注意:
网络编程注意事项:
1、所有网络操作都需要在工作线程中执行
2、网络操作需要添加权限

实现

按照以上思路完成服务端 ChatServer

public class ChatServer {private List<Socket> sockets = new ArrayList<Socket>();/*** 开启服务** @throws IOException*/public void startServer() throws IOException {System.out.println("服务已启动...");//启动服务ServerSocket ss = new ServerSocket(8888);//不断接收从客户端发来的消息while (true) {//阻塞执行,接收客户端连接Socket socket = ss.accept();sockets.add(socket);//想要并发处理的话,需要开启线程new WorkThread(socket).start();}}/*** 工作线程* 接收客户端的消息,并给每个线程发一遍*/class WorkThread extends Thread {private Socket socket;public WorkThread(Socket socket) {this.socket = socket;}public void run() {try {DataInputStream dis = new DataInputStream(socket.getInputStream());while (true) {String message = dis.readUTF();//给每一个客户端都发一遍for (int i = 0; i < sockets.size(); i++) {Socket s = sockets.get(i);DataOutputStream dos = new DataOutputStream(s.getOutputStream());dos.writeUTF(message);dos.flush();}}} catch (IOException e) {e.printStackTrace();//当开启一条线程后,线程会等待客户端发消息//在等待的过程中,可能出现异常,比如连接中断,就会抛出connection reset的异常//就会走到这里System.out.println("连接中断");//把当前socket对象从集合中移除sockets.remove(socket);}}}public static void main(String[] args) {ChatServer server = new ChatServer();try {server.startServer();} catch (IOException e) {// TODO Auto-generated catch blocke.printStackTrace();}}
}

布局文件activity_main

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"android:layout_width="match_parent"android:layout_height="match_parent"android:orientation="vertical"><EditTextandroid:id="@+id/et_ip"android:layout_width="match_parent"android:layout_height="wrap_content"android:layout_toLeftOf="@+id/btn_ip" /><Buttonandroid:id="@+id/btn_ip"android:layout_width="wrap_content"android:layout_height="wrap_content"android:layout_alignParentRight="true"android:text="确定" /><ListViewandroid:id="@+id/listview"android:layout_width="match_parent"android:layout_height="match_parent"android:layout_above="@+id/btn_message"android:layout_below="@+id/btn_ip" /><Buttonandroid:id="@+id/btn_message"android:layout_width="wrap_content"android:layout_height="wrap_content"android:layout_alignParentRight="true"android:layout_alignParentBottom="true"android:text="确定" /><EditTextandroid:id="@+id/et_message"android:layout_width="match_parent"android:layout_height="wrap_content"android:layout_alignParentBottom="true"android:layout_toLeftOf="@+id/btn_message" />
</RelativeLayout>

MainActivity

public class MainActivity extends AppCompatActivity implements View.OnClickListener {private EditText et_ip;private EditText et_message;private Button btn_ip;private Button btn_message;private ListView listview;private Socket socket;private DataInputStream dis;private DataOutputStream dos;private List<String> messages = new ArrayList<>();private ArrayAdapter<String> adapter;private Handler handler = new Handler() {@Overridepublic void handleMessage(Message msg) {switch (msg.what) {case HANDLER_CONNECT_SUCCESS://连接成功Toast.makeText(MainActivity.this, "连接成功", Toast.LENGTH_SHORT).show();et_ip.setEnabled(false);btn_ip.setEnabled(false);break;case HANDLER_MESSAGE_RECEIVED://接受到消息String message = (String) msg.obj;messages.add("小明:" + message);//更新adapteradapter.notifyDataSetChanged();//显示最后一条listview.setSelection(messages.size() - 1);break;}}};private static final int HANDLER_CONNECT_SUCCESS = 1;private static final int HANDLER_MESSAGE_RECEIVED = 2;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);setViews();//设置适配器setAdapter();}private void setAdapter() {adapter = new ArrayAdapter<String>(this, android.R.layout.simple_list_item_1, messages);listview.setAdapter(adapter);}private void setViews() {et_ip = findViewById(R.id.et_ip);et_message = findViewById(R.id.et_message);btn_ip = findViewById(R.id.btn_ip);btn_message = findViewById(R.id.btn_message);listview = findViewById(R.id.listview);btn_ip.setOnClickListener(this);btn_message.setOnClickListener(this);}@Overridepublic void onClick(View view) {switch (view.getId()) {case R.id.btn_ip://建立连接new Thread() {@Overridepublic void run() {try {connect();} catch (IOException e) {e.printStackTrace();//连接建立失败}}}.start();break;case R.id.btn_message://发送消息new Thread() {@Overridepublic void run() {try {sendMessage();} catch (IOException e) {e.printStackTrace();}}}.start();break;}}private void sendMessage() throws IOException {String text = et_message.getText().toString();dos.writeUTF(text);dos.flush();}/*** 读取服务端发回的消息的工作线程*/class ReadThread extends Thread {//run方法执行完毕后,线程也就销毁了//垃圾回收机制会回收@Overridepublic void run() {/*while (true){try {String message = dis.readUTF();} catch (IOException e) {e.printStackTrace();//当断开连接,并不会跳出while循环,所以改成以下写法}}*/try {while (true) {String message = dis.readUTF();Log.i("info", "message:" + message);//更新ListView,给handler发消息Message msg = new Message();msg.what = HANDLER_MESSAGE_RECEIVED;msg.obj = message;handler.sendMessage(msg);}} catch (IOException e) {e.printStackTrace();//连接异常,断开Log.e("error", "连接出错,读不了数据了,看着办吧");}}}private void connect() throws IOException {String ip = et_ip.getText().toString();socket = new Socket(ip, 8888);dis = new DataInputStream(socket.getInputStream());dos = new DataOutputStream(socket.getOutputStream());//连接成功,发消息给主线程Handler 更新UIhandler.sendEmptyMessage(HANDLER_CONNECT_SUCCESS);//启动读取消息的工作线程new ReadThread().start();}@Overrideprotected void onDestroy() {super.onDestroy();if (socket != null) {try {//socket关掉后,流也就关掉了//而客户端还有线程正等着读取数据,所以也需要处理下socket.close();} catch (IOException e) {e.printStackTrace();}}}
}

AndroidManifest增加权限

    <uses-permission android:name="android.permission.INTERNET"/>

运行步骤
1、用 IntelliJ IDEA 运行ChatServer
在这里插入图片描述

2、查看自己电脑 ip
cmd 中执行ipconfig
在这里插入图片描述

3、修改MainActivity中,以下代码,把 message 前增加一个称呼,例如“小明”,运行在一台模拟机中

private void sendMessage() throws IOException {String text = et_message.getText().toString();dos.writeUTF("小明:\n"+text);dos.flush();}

4、修改 MainActivity 中 3 步骤代码中的称呼,改为“小红”,运行在另一台模拟机中

5、都输入 ip 地址确保连接成功

5、开始尬聊

在这里插入图片描述

这篇关于【达内课程】利用Socket实现群聊的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/982310

相关文章

hdu1043(八数码问题,广搜 + hash(实现状态压缩) )

利用康拓展开将一个排列映射成一个自然数,然后就变成了普通的广搜题。 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#include<stdlib.h>#include<ctype.h>#inclu

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

【Prometheus】PromQL向量匹配实现不同标签的向量数据进行运算

✨✨ 欢迎大家来到景天科技苑✨✨ 🎈🎈 养成好习惯,先赞后看哦~🎈🎈 🏆 作者简介:景天科技苑 🏆《头衔》:大厂架构师,华为云开发者社区专家博主,阿里云开发者社区专家博主,CSDN全栈领域优质创作者,掘金优秀博主,51CTO博客专家等。 🏆《博客》:Python全栈,前后端开发,小程序开发,人工智能,js逆向,App逆向,网络系统安全,数据分析,Django,fastapi

让树莓派智能语音助手实现定时提醒功能

最初的时候是想直接在rasa 的chatbot上实现,因为rasa本身是带有remindschedule模块的。不过经过一番折腾后,忽然发现,chatbot上实现的定时,语音助手不一定会有响应。因为,我目前语音助手的代码设置了长时间无应答会结束对话,这样一来,chatbot定时提醒的触发就不会被语音助手获悉。那怎么让语音助手也具有定时提醒功能呢? 我最后选择的方法是用threading.Time

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo

C#实战|大乐透选号器[6]:实现实时显示已选择的红蓝球数量

哈喽,你好啊,我是雷工。 关于大乐透选号器在前面已经记录了5篇笔记,这是第6篇; 接下来实现实时显示当前选中红球数量,蓝球数量; 以下为练习笔记。 01 效果演示 当选择和取消选择红球或蓝球时,在对应的位置显示实时已选择的红球、蓝球的数量; 02 标签名称 分别设置Label标签名称为:lblRedCount、lblBlueCount

Kubernetes PodSecurityPolicy:PSP能实现的5种主要安全策略

Kubernetes PodSecurityPolicy:PSP能实现的5种主要安全策略 1. 特权模式限制2. 宿主机资源隔离3. 用户和组管理4. 权限提升控制5. SELinux配置 💖The Begin💖点点关注,收藏不迷路💖 Kubernetes的PodSecurityPolicy(PSP)是一个关键的安全特性,它在Pod创建之前实施安全策略,确保P

工厂ERP管理系统实现源码(JAVA)

工厂进销存管理系统是一个集采购管理、仓库管理、生产管理和销售管理于一体的综合解决方案。该系统旨在帮助企业优化流程、提高效率、降低成本,并实时掌握各环节的运营状况。 在采购管理方面,系统能够处理采购订单、供应商管理和采购入库等流程,确保采购过程的透明和高效。仓库管理方面,实现库存的精准管理,包括入库、出库、盘点等操作,确保库存数据的准确性和实时性。 生产管理模块则涵盖了生产计划制定、物料需求计划、

C++——stack、queue的实现及deque的介绍

目录 1.stack与queue的实现 1.1stack的实现  1.2 queue的实现 2.重温vector、list、stack、queue的介绍 2.1 STL标准库中stack和queue的底层结构  3.deque的简单介绍 3.1为什么选择deque作为stack和queue的底层默认容器  3.2 STL中对stack与queue的模拟实现 ①stack模拟实现

基于51单片机的自动转向修复系统的设计与实现

文章目录 前言资料获取设计介绍功能介绍设计清单具体实现截图参考文献设计获取 前言 💗博主介绍:✌全网粉丝10W+,CSDN特邀作者、博客专家、CSDN新星计划导师,一名热衷于单片机技术探索与分享的博主、专注于 精通51/STM32/MSP430/AVR等单片机设计 主要对象是咱们电子相关专业的大学生,希望您们都共创辉煌!✌💗 👇🏻 精彩专栏 推荐订阅👇🏻 单片机