Android小项目——聊天室

本文介绍了如何在Android应用中实现更换头像功能,包括添加图标、处理网络权限和登录界面的设计。接着详细展示了选择头像功能,以及客户端的ChatRoom类,涉及网络通信和消息处理。服务器部分则展示了如何接收和发送消息,包括UserThread和ServerSocket的使用。
摘要由CSDN通过智能技术生成

简单介绍

Android的第二个小程序,初步实现了换头像功能、多人聊天功能。先放几张照片亮亮相吧:
在这里插入图片描述

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

更换图标

在drawale文件夹中加入.png的一张照片即可

android:icon="@drawable/icon"

网络权限

保险起见3条全加,当时只加了一条然后就连不上网。。。

    package="com.example.chartroom">
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
    <uses-permission android:name="android.permission.INTERNET" />
    <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
    <application

登录界面

注意实时查看你连的WIFI的IP地址,Win+R输入CMD,再输入ipconfig/all查看WIFI那一栏中的IPV4

activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="@drawable/bd1"
    android:orientation="vertical">
    
	//这可以把上面标题改掉
    <androidx.appcompat.widget.Toolbar
        android:id="@+id/toolbar"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:background="#2FF1EEEE">

        <TextView
            android:id="@+id/tv_room"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_gravity="center"
            android:gravity="center"
            android:text="欢迎登录聊天室"
            android:textColor="#93FFFFFF"
            android:textSize="19sp" />
            
    </androidx.appcompat.widget.Toolbar>
       

        <LinearLayout
            android:orientation="horizontal"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:gravity="center"
            android:layout_marginLeft="50sp"
            android:layout_marginRight="50sp"
            android:layout_marginTop="100dp">

            <TextView
                android:id="@+id/tv_name"
                android:layout_width="0dp"
                android:layout_height="wrap_content"
                android:layout_weight="1"
                android:gravity="center"
                android:text="用户名"
                android:textSize="20sp"
                android:textColor="#93FFFFFF"/>

            <EditText
                android:id="@+id/et_name"
                android:layout_width="0dp"
                android:layout_height="wrap_content"
                android:layout_weight="2"
                android:gravity="center"
                android:textColor="#93FFFFFF"
                android:theme="@style/CretateRoomEditTextTheme"/>

        </LinearLayout>

    <LinearLayout
        android:orientation="horizontal"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:gravity="center"
        android:layout_marginLeft="50sp"
        android:layout_marginRight="50sp"
        android:layout_marginTop="50dp">

        <TextView
            android:id="@+id/tv_ip"
            android:layout_width="0dp"
            android:layout_height="wrap_content"
            android:layout_weight="1"
            android:gravity="center"
            android:text="IP地址"
            android:textSize="20sp"
            android:textColor="#93FFFFFF"/>

        <EditText
            android:id="@+id/et_ip"
            android:layout_width="0dp"
            android:layout_height="wrap_content"
            android:layout_weight="2"
            android:gravity="center"       
            android:text="192.168.1.5" 
            android:textColor="#93FFFFFF"
            android:theme="@style/CretateRoomEditTextTheme" />


    </LinearLayout>

    <LinearLayout
        android:orientation="horizontal"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:gravity="center"
        android:layout_marginLeft="50sp"
        android:layout_marginRight="50sp"
        android:layout_marginTop="50dp">

        <TextView
            android:id="@+id/tv_port"
            android:layout_width="0dp"
            android:layout_height="wrap_content"
            android:layout_weight="1"
            android:gravity="center"
            android:text="端口"
            android:textSize="20sp"
            android:textColor="#93FFFFFF"/>

        <EditText
            android:id="@+id/et_port"
            android:layout_width="0dp"
            android:layout_height="wrap_content"
            android:layout_weight="2"
            android:gravity="center"
            android:textColor="#93FFFFFF"
            android:theme="@style/CretateRoomEditTextTheme"
            android:text="6666"/>

    </LinearLayout>

        <LinearLayout
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:gravity="center"
            android:layout_gravity="center"
            android:layout_marginLeft="60sp"
            android:layout_marginRight="50sp"
            android:layout_marginTop="50dp">

            <Button
                android:id="@+id/login"
                android:layout_width="0dp"
                android:layout_height="wrap_content"
                android:layout_weight = "1"
                android:layout_gravity="center"
                android:gravity="center"
                android:layout_margin="30dp"
                android:textColor="#93FFFFFF"
                android:textSize="16sp"
                android:background="@drawable/change"
                android:text="连接"/>

            <Button
                android:id="@+id/quit"
                android:layout_width="0dp"
                android:layout_height="wrap_content"
                android:layout_weight = "1"
                android:layout_gravity="center"
                android:gravity="center"
                android:layout_margin="30dp"
                android:textColor="#93FFFFFF"
                android:textSize="16sp"
                android:background="@drawable/change"
                android:text="退出"/>
        </LinearLayout>

</LinearLayout>

MainActivity.java

package com.example.chartroom;

import androidx.appcompat.app.AlertDialog;
import androidx.appcompat.app.AppCompatActivity;

import android.content.Context;
import android.content.DialogInterface;
import android.content.Intent;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.TextView;
import android.widget.Toast;

public class MainActivity extends AppCompatActivity {
    
    private Button login;
    private Button quit;
    private EditText et_name;
    private EditText et_ip;
    private EditText et_port;
    private TextView my_name;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        //获取id,连接控件
        login = (Button) findViewById(R.id.login);
        quit = (Button) findViewById(R.id.quit);
        et_name = findViewById(R.id.et_name);
        et_ip = (EditText) findViewById(R.id.et_ip);
        et_port = (EditText) findViewById(R.id.et_port);
        //login点击事件
        login.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                //获取一个视图View对象里的字符串
                String name = et_name.getText().toString();
                //如果没输入名字
                if("".equals(name)){
                    Toast.makeText(MainActivity.this, "请输入用户名!", Toast.LENGTH_SHORT).show();
                }else{
                    Intent intent = new Intent(MainActivity.this,ChoosePicture.class);
                    //向下一活动传递信息                                        intent.putExtra("name",et_name.getText().toString());
                    intent.putExtra("ip",et_ip.getText().toString());
                    intent.putExtra("port",et_port.getText().toString());                    
                    try{
                        startActivity(intent);
                    }catch(Exception e){
                        System.out.println("开启失败");
                        finish();
                    }                 
                }
            }
        });
        //quit点击事件
        quit.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
            	//用AlertDialog显示一个退出提示框
                AlertDialog.Builder dialog = new AlertDialog.Builder(MainActivity.this);
                dialog.setTitle("关闭提示");
                dialog.setMessage("确定退出登录?");
                dialog.setPositiveButton("确定", new DialogInterface.OnClickListener() {
                    @Override
                    public void onClick(DialogInterface dialogInterface, int i) {
                        System.exit(0);
                    }
                });
                dialog.setNegativeButton("取消", new DialogInterface.OnClickListener() {
                    @Override
                    public void onClick(DialogInterface dialogInterface, int i) {
                    }
                });
                dialog.show();
            }
        });
    }
}

选择头像

activity_choose_picture.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:background="@drawable/bd1"
    android:orientation="vertical">

    <androidx.appcompat.widget.Toolbar
        android:id="@+id/toolbar"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:background="#2FF1EEEE">

        <TextView
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_gravity="center"
            android:gravity="center"
            android:text="选择一个喜欢的头像"
            android:textColor="#93FFFFFF"
            android:textSize="19sp" />
    </androidx.appcompat.widget.Toolbar>

    <LinearLayout
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:orientation="horizontal">

        <Button
            android:id="@+id/one"
            android:layout_width="0dp"
            android:layout_height="115dp"
            android:layout_weight="1"
            android:background="@drawable/headofothers"
            android:layout_margin="10dp"/>

        <Button
            android:id="@+id/two"
            android:layout_width="0dp"
            android:layout_height="115dp"
            android:layout_weight="1"
            android:background="@drawable/headofmy"
            android:layout_margin="10dp"/>

        <Button
            android:id="@+id/three"
            android:layout_width="0dp"
            android:layout_height="115dp"
            android:layout_weight="1"
            android:background="@drawable/three"
            android:layout_margin="10dp"/>
            
    </LinearLayout>

    <LinearLayout
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:orientation="horizontal">


        <Button
            android:id="@+id/four"
            android:layout_width="0dp"
            android:layout_height="115dp"
            android:layout_weight="1"
            android:background="@drawable/six"
            android:layout_margin="10dp"/>

        <Button
            android:id="@+id/five"
            android:layout_width="0dp"
            android:layout_height="115dp"
            android:layout_weight="1"
            android:background="@drawable/four"
            android:layout_margin="10dp"/>

        <Button
            android:id="@+id/six"
            android:layout_width="0dp"
            android:layout_height="115dp"
            android:layout_weight="1"
            android:background="@drawable/five"
            android:layout_margin="10dp"/>
    </LinearLayout>

</LinearLayout>

ChoosePicture.java

package com.example.chartroom;

import android.content.Context;
import android.content.Intent;
import android.os.Bundle;
import android.os.PersistableBundle;
import android.view.View;
import android.widget.Button;
import android.widget.ImageView;

import androidx.annotation.Nullable;
import androidx.appcompat.app.AppCompatActivity;

public class ChoosePicture extends AppCompatActivity implements View.OnClickListener{

    private String et_name;
    private String et_ip;
    private String et_port;

    @Override
    public void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_choose_picture);
        System.out.println("启动2");
        //启动活动,并获取传递过来的信息
        Intent intent = getIntent();
        //传入键值得到数据
        et_name = intent.getStringExtra("name");
        et_ip = intent.getStringExtra("ip");
        et_port = intent.getStringExtra("port");
        Button button1 = (Button)findViewById(R.id.one);
        Button button2 = (Button)findViewById(R.id.two);
        Button button3 = (Button)findViewById(R.id.three);
        Button button4 = (Button)findViewById(R.id.four);
        Button button5 = (Button)findViewById(R.id.five);
        Button button6 = (Button)findViewById(R.id.six);
        button1.setOnClickListener(this);
        button2.setOnClickListener(this);
        button3.setOnClickListener(this);
        button4.setOnClickListener(this);
        button5.setOnClickListener(this);
        button6.setOnClickListener(this);
    }

    @Override
    public void onClick(View view) {
        //向下一活动传递信息
        Intent intent = new Intent(ChoosePicture.this,ChatRoom.class);
        intent.putExtra("name",et_name.toString());
        intent.putExtra("ip",et_ip.toString());
        intent.putExtra("port",et_port.toString());
        switch(view.getId()){
            case R.id.one:
                intent.putExtra("imageId",R.drawable.headofothers);
                break;
            case R.id.two:
                intent.putExtra("imageId",R.drawable.headofmy);
                break;
            case R.id.three:
                intent.putExtra("imageId",R.drawable.three);
                break;
            case R.id.four:
                intent.putExtra("imageId",R.drawable.six);
                break;
            case R.id.five:
                intent.putExtra("imageId",R.drawable.four);
                break;
            case R.id.six:
                intent.putExtra("imageId",R.drawable.five);
                break;
        }
        startActivity(intent);
    }

}

客户端

这是将信息传出和接收传入信息的重要板块,其中涉及到IO流、多线程、网络编程等知识。大家可以先复习一遍再来写这一块。

ChatRoom.java

package com.example.chartroom;

import androidx.annotation.NonNull;
import androidx.appcompat.app.AppCompatActivity;
import androidx.recyclerview.widget.LinearLayoutManager;
import androidx.recyclerview.widget.RecyclerView;

import android.annotation.SuppressLint;
import android.content.Intent;
import android.os.Bundle;
import android.os.Handler;
import android.os.Looper;
import android.os.Message;
import android.text.TextUtils;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.TextView;
import android.widget.Toast;

import java.io.DataInputStream;
import java.io.DataOutputStream;
import java.io.IOException;
import java.net.Socket;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.Date;
import java.util.List;

public class ChatRoom extends AppCompatActivity implements View.OnClickListener{
    private List<Msg> msgList = new ArrayList<>();
    private MsgAdapter adapter;
    private Button back;                //退回键
    private RecyclerView recyclerView;  //对话显示框
    private EditText input_text;        //输入框
    private int imageId ;               //获取自己的头像
    private int his_imageId;            //获取对方发头像
    private String time;                //获取时间
    private Button send;                //发送按钮
    private String name;                //从主活动获取的名字
    private String hisName;             //对方昵称,从自定义的receive线程中获取,便于和自己的昵称区分
    private String content;             //获取对话内容
    private String ip ;                 //获取ip地址
    private String port;                //获取端口号
    private Socket socketSend;          //套接字,用于绑定ip号和端口号便于计算机之间的传输消息
    private DataInputStream dis;        //码头
    private DataOutputStream dos;       //集装箱
    private String recMsg;
    boolean isRunning = false;          //判断线程是否运行
    boolean isSend = false;             //判断是否发送
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_chat_room);
        //启动活动,并获取传递过来的信息
        Intent intent = getIntent();
        //传入键值得到数据
        name = intent.getStringExtra("name");
        ip = intent.getStringExtra("ip");
        port = intent.getStringExtra("port");
        //取出int要指定key,还要设置默认值,当intent中没有该key对应的value时,返回设置的默认值
        imageId = intent.getIntExtra("imageId",0);
        //获取实例
        input_text = (EditText) findViewById(R.id.input_text);
        back = (Button) findViewById(R.id.back);
        send = (Button) findViewById(R.id.send);
        //注册监听器
        back.setOnClickListener(this);
        send.setOnClickListener(this);

        //将RecyclerView和list建立联系(建立与适配器关系啥的)
        LinearLayoutManager layoutManager = new LinearLayoutManager(ChatRoom.this);
        recyclerView = (RecyclerView) findViewById(R.id.msg_recycler_view);
        recyclerView.setLayoutManager(layoutManager);
        adapter = new MsgAdapter(msgList);
        recyclerView.setAdapter(adapter);

        //如果要连网的话就不能在主线程上操作,所以要另外开启一条线程
        new Thread(new Runnable() {
            @Override
            public void run() {
                try {
                    System.out.println("号码");
                    socketSend = new Socket(ip, Integer.parseInt(port));
                    isRunning = true;
                    dis = new DataInputStream(socketSend.getInputStream());
                    dos = new DataOutputStream(socketSend.getOutputStream());
                    System.out.println("打开dos");
                    //开一条线程接收服务器传来的信息
                    new Thread(new receive(), "接收线程").start();
                    System.out.println("打开线程");
                } catch (Exception e) {
                    Log.e("TAG",e.toString());
                    e.printStackTrace();
                    //为当前线程准备消息队列
                    Looper.prepare();
                    //Toast只有在主线程中能显示出来
                    Toast.makeText(ChatRoom.this, "连接服务器失败", Toast.LENGTH_SHORT).show();
                    //开启循环取消息
                    Looper.loop();
                }
            }
        }).start();

    }

    //获取当前时间
    public String getCurrentTime(){
        Date d = new Date();
        //设置显示的时间格式
        SimpleDateFormat sdf = new SimpleDateFormat("yyyy年MM月dd日HH:mm:ss");
        //以这种格式显示
        return sdf.format(d);
    }

    @Override
    public void onClick(View view) {
        switch(view.getId()){
            case R.id.back:
                finish();
                break;
            case R.id.send:
                //显示时间
                time = getCurrentTime();
                String content = input_text.getText().toString();
                //显示信息
                StringBuilder sb = new StringBuilder();
                sb.append(content);
                if(!"".equals(content)){  //发送信息不为空
                    isSend = true;
                    new Thread(new Runnable() {
                        @Override
                        public void run() {
                            String content = input_text.getText().toString();
                            Log.d("ttw","发了一条消息");
                            System.out.println("content = "+content);
                            System.out.println("传入发送信息");
                            if(!"".equals(content) && isSend){
                                String date = getCurrentTime();
                                try {
                                    if(!"".equals(date) && !"".equals(name)){
                                        dos.writeUTF(date);
                                        dos.writeUTF(content);
                                        dos.writeUTF(name);
                                        dos.writeUTF(String.valueOf(imageId));
                                        System.out.println("发送完了");
                                    }
                                } catch (IOException e) {
                                    e.printStackTrace();
                                }
                                isSend = false;
                            }
                        }
                    }).start();
                    Msg msg = new Msg(content,Msg.TYPE_SENT,time,name,imageId);   //发送消息
                    System.out.println("imageId = "+imageId);
                    msgList.add(msg);
                    //当有新消息时,刷新ListView中的消息
                    adapter.notifyItemInserted(msgList.size()-1);
                    //将LestView定位到最后一行
                    recyclerView.scrollToPosition(msgList.size()-1);
                    input_text.setText("");
                }else{
                    Toast.makeText(ChatRoom.this, "不可发送空信息!", Toast.LENGTH_SHORT).show();
                }
                sb.delete(0,sb.length());
                break;
            default:
                break;
        }
    }

    //子线程与主线程通过Handler来进行通信。子线程可以通过Handler来通知主线程进行UI更新。
    //Handler有两个主要的用途:(1)安排消息和可运行对象在将来的某个时间点执行;(2)将一个要在不同的线程上执行的动作编入队列。
    private Handler handler = new Handler(Looper.myLooper()){
        @Override
        public void handleMessage(@NonNull Message msg) {
            if(!recMsg.isEmpty()){
                System.out.println("更新消息");
                addNewMessage(content,Msg.TYPE_RECEIVED,time,hisName,his_imageId);  //刷新接收的消息
            }
        }
    };

    public void addNewMessage(String msg,int type,String time,String name,int his_imageId){
        Msg message = new Msg(msg,type,time,name,his_imageId);
        msgList.add(message);
        //当有消息时,通知列表有新的数据插入,刷新recyclerview中消息
        adapter.notifyItemInserted(msgList.size()-1);
        //将消息一直放在显示屏的底部不随意跑上去
        recyclerView.scrollToPosition(msgList.size()-1);
    }

    //接收线程
    class receive implements Runnable{
        @Override
        public void run() {
            recMsg = "";
            while(isRunning){
                System.out.println("开始接收线程receive");
                Msg msg = null;
                try {
                    time = dis.readUTF();
                    content = dis.readUTF();
                    hisName = dis.readUTF();
                    his_imageId = Integer.parseInt(dis.readUTF());
                    System.out.println("接收信息 = "+content);
                    recMsg = hisName + time + content;
                    msg = new Msg(content,Msg.TYPE_RECEIVED,time,hisName,his_imageId);
                } catch (Exception e) {
                    System.out.println("接受失败");
                    e.printStackTrace();
                }
                //判断是否为空字符串
                if(!TextUtils.isEmpty(recMsg)){
                    System.out.println("查看名字"+msg.getName());
                    Message message = new Message();
                    message.obj = msg;
                    handler.sendMessage(message);
                }
            }
        }
    }

}

activity_chat_room.xml

要显示<这个标识符的话,得写成:"&#060,"

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="@drawable/bd2"
    android:orientation="vertical">

    <androidx.appcompat.widget.Toolbar
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:background="#2FF1EEEE">

        <Button
            android:id="@+id/back"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:textColor="#FFFFFF"
            android:textSize="30sp"
            android:background="#00FFFFFF"
            android:text="&#060;"/>


        <TextView
            android:id="@+id/tv_room"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_gravity="center"
            android:text="ChatRoom"
            android:textColor="#CDFFFFFF"
            android:textSize="23sp" />



    </androidx.appcompat.widget.Toolbar>


    <androidx.recyclerview.widget.RecyclerView
    android:id="@+id/msg_recycler_view"
    android:layout_width="match_parent"
    android:layout_height="0dp"
    android:layout_weight="1" />


    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_marginRight="10dp"
        android:layout_marginLeft="10dp">

        <EditText
            android:id="@+id/input_text"
            android:layout_width="0dp"
            android:layout_height="wrap_content"
            android:layout_weight="1"
            android:layout_margin="10dp"
            android:hint="Type Something"
            android:maxLines="2" />

        <Button
            android:id="@+id/send"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_margin="10dp"
            android:text="Send"
            android:textColor="#AAFFFFFF"/>

    </LinearLayout>

</LinearLayout>

msg_item.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:orientation="vertical">
<LinearLayout
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:orientation="vertical"
    android:layout_marginTop="20dp">

    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:gravity="center">

        <TextView
            android:id="@+id/time"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_gravity="center"
            android:gravity="center"
            android:textSize="15dp" />

    </LinearLayout>

    </LinearLayout>

            <LinearLayout
                android:id="@+id/left_layout"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:layout_gravity="left"
                android:layout_marginTop="5dp">

                <ImageView
                    android:id="@+id/head_others"
                    android:layout_marginRight="5dp"
                    android:layout_width="60dp"
                    android:layout_height="60dp" />

                <LinearLayout
                    android:layout_width="wrap_content"
                    android:layout_height="wrap_content"
                    android:orientation="vertical"
                    android:layout_marginLeft="5dp">

                    <TextView
                        android:id="@+id/others_name"
                        android:layout_gravity="left"
                        android:layout_width="wrap_content"
                        android:layout_height="0dp"
                        android:layout_weight="1"/>

                    <LinearLayout
                        android:layout_width="wrap_content"
                        android:layout_height="0dp"
                        android:layout_weight="2"
                        android:background="@drawable/his_bubble">

                        <TextView
                            android:id="@+id/left_msg"
                            android:layout_width="wrap_content"
                            android:layout_height="wrap_content"
                            android:layout_gravity="center"
                            android:gravity="center"
                            android:textColor="@color/black"
                            android:layout_margin="8dp"
                            android:textSize="20dp" />
                    </LinearLayout>


                </LinearLayout>

    </LinearLayout>


    <LinearLayout
        android:id="@+id/right_layout"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="right"
        android:layout_marginTop="5dp">

        <LinearLayout
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_marginRight="5dp"
            android:orientation="vertical">

            <TextView
                android:id="@+id/my_name"
                android:layout_width="wrap_content"
                android:layout_height="0dp"
                android:layout_gravity="right"
                android:layout_weight="1" />

            <LinearLayout
                android:layout_width="wrap_content"
                android:layout_height="0dp"
                android:layout_weight="2"
                android:background="@drawable/my_bubble">

                <TextView
                    android:id="@+id/right_msg"
                    android:layout_width="wrap_content"
                    android:layout_height="wrap_content"
                    android:layout_gravity="center"
                    android:layout_margin="8dp"
                    android:gravity="center"
                    android:textColor="@color/black"
                    android:textSize="20dp" />
            </LinearLayout>
        </LinearLayout>

        <ImageView
            android:id="@+id/head_my"
            android:layout_marginLeft="5dp"
            android:layout_width="60dp"
            android:layout_height="60dp" />

        </LinearLayout>

</LinearLayout>

Msg.java

package com.example.chartroom;

public class Msg {
    public static final int TYPE_RECEIVED = 0;	//标记用于判断是输出消息还是接收消息
    public static final int TYPE_SENT = 1;
    private String content;		//存储消息信息
    private int type;	//存储判断信息
    private String time;
    private String name;
    private int imageId;

    public Msg(String content, int type, String time,String name,int imageId) {
        this.content = content;
        this.type = type;
        this.time = time;
        this.name = name;
        this.imageId = imageId;
    }

    public String getContent() {
        return content;
    }

    public int getType() {
        return type;
    }

    public String getTime(){
        return time;
    }

    public String getName(){
        return name;
    }

    public int getImageId(){
        return imageId;
    }
}

MsgAdapter.java

package com.example.chartroom;

import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ArrayAdapter;
import android.widget.EditText;
import android.widget.ImageView;
import android.widget.LinearLayout;
import android.widget.TextView;

import androidx.annotation.NonNull;
import androidx.recyclerview.widget.RecyclerView;

import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.List;
import java.util.zip.Inflater;

public class MsgAdapter extends RecyclerView.Adapter<MsgAdapter.ViewHolder> {
    List<Msg> mMsgList;

    //把显示的数据源传进来
    public MsgAdapter(List<Msg> msgList) {
        super();
        mMsgList = msgList;
    }

    //创建ViewHolder实例
    @Override
    public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
        View view = LayoutInflater.from(parent.getContext()).inflate(R.layout.msg_item,parent,false);
        return new ViewHolder(view);
    }

    //对RecyclerView子项数据进行赋值
    @Override
    public void onBindViewHolder(ViewHolder holder, int position) {
        Msg msg = mMsgList.get(position);
        holder.time.setText((CharSequence)msg.getTime());
        if(msg.getType() == Msg.TYPE_RECEIVED){  //接收端
            holder.his_head.setImageResource(msg.getImageId());
            holder.leftLayout.setVisibility(View.VISIBLE);  //左可见
            holder.rightLayout.setVisibility(View.GONE);    //右不可见
            holder.hisName.setText(msg.getName());
            holder.leftMsg.setText(msg.getContent());       //左边显示消息
        }else if(msg.getType() == Msg.TYPE_SENT){
			holder.my_head.setImageResource(msg.getImageId());
            holder.rightLayout.setVisibility(View.VISIBLE);
            holder.leftLayout.setVisibility(View.GONE);
            holder.name.setText(msg.getName());
            holder.rightMsg.setText(msg.getContent());
        }
    }

    //告诉RecycleView一共多少子项,返回数据源长度
    @Override
    public int getItemCount() {
        return mMsgList.size();
    }

    public static class ViewHolder extends RecyclerView.ViewHolder {
        LinearLayout leftLayout ;
        LinearLayout rightLayout;
        TextView leftMsg;
        TextView rightMsg;
        TextView time;
        TextView name;
        TextView hisName;
        ImageView my_head;
        ImageView his_head;

        public ViewHolder(View view) {
            super(view);
            leftLayout = view.findViewById(R.id.left_layout);
            rightLayout = view.findViewById(R.id.right_layout);
            leftMsg = view.findViewById(R.id.left_msg);
            rightMsg = view.findViewById(R.id.right_msg);
            time = view.findViewById(R.id.time);              //显示时间
            name = view.findViewById(R.id.my_name);
            hisName = view.findViewById(R.id.others_name);
            my_head = view.findViewById(R.id.head_my);
            his_head = view.findViewById(R.id.head_others);
        }
    }
}

服务器

任意Java编译器即可,我用的IDEA

import java.io.*;
import java.net.ServerSocket;
import java.net.Socket;
import java.util.ArrayList;
import java.util.ConcurrentModificationException;

public class Server {
    public static void main(String[] args) {
        new Thread(new startServer()).start();
    }
    private static int port = 6666;
    public static ArrayList<UserThread> socketList = new ArrayList<UserThread>();
    private static class startServer extends Thread{
        public void run(){
            try {
                //绑定服务器要监听的端口
                ServerSocket serverSocket = new ServerSocket(port);
                while(true){
                    //从队列中取出连接请求,使得队列能及时腾出空位,以容纳新的连接请求
                    Socket socket = serverSocket.accept();
                    System.out.println(""+socket);
                    UserThread userThread = new UserThread(socket);
                    socketList.add(userThread);
                    new Thread(userThread).start();
                }
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }

    private static class UserThread implements Runnable {
        private Socket skt;
        private DataInputStream dis;
        private DataOutputStream dos;

        public DataOutputStream getDos() {
            return dos;
        }

        public UserThread(Socket socket) {
            skt = socket;
        }

        //接收send过来的线程
        @Override
        public void run() {
            try {
                dos = new DataOutputStream(skt.getOutputStream());
                dis = new DataInputStream(skt.getInputStream());
                while (true) {
                    String r_content ;
                    String name ;
                    String time ;
                    String imageId ;
                    time = dis.readUTF();
                    r_content = dis.readUTF();
                    name = dis.readUTF();
                    imageId = dis.readUTF();
                    System.out.println("content = "+r_content);
                    if(time == null){
                        continue;
                    }
                    //发送出去
                    for(UserThread ut : socketList){
                        if(ut.equals(this)){  //是自己的消息就不发出
                            System.out.println("自己的");
                            continue;
                        }
                        try{
                            System.out.println(time);
                            //writeUTF在写入数据流的时候会加上两个字节以表示字节的长度
                            ut.getDos().writeUTF(time);
                            ut.getDos().writeUTF(r_content);
                            ut.getDos().writeUTF(name);
                            ut.getDos().writeUTF(imageId);
                            System.out.println("写出去");
                            //刷新
                            ut.getDos().flush();
                        }catch(Exception e){
                            socketList.remove(ut);
                            e.printStackTrace();
                        }
                    }
                }
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    }
}

  • 13
    点赞
  • 109
    收藏
    觉得还不错? 一键收藏
  • 4
    评论
### 回答1: 新闻app是一款基于Android平台的小型项目应用程序,它主要用于展示各类新闻内容,为用户提供便捷的阅读体验。该项目的源码包含了应用程序的基本框架和功能实现,方便开发者进行二次开发和定制。 新闻app的源码主要包含以下几个方面的内容: 1. 用户界面设计:源码中包含了新闻app的界面布局和样式,开发者可以根据自己的需要进行修改和美化。用户界面通常包括新闻列表、新闻详情页、分类标签等,开发者可以自由设计并添加其他功能模块。 2. 数据获取与展示:源码中实现了与服务器进行数据交互的功能,通过网络请求获取新闻数据,并在界面上展示出来。开发者可以根据需要修改数据请求接口和解析方式,实现与自己的服务器交互。 3. 新闻分类与搜索:源码中提供了新闻分类和搜索功能的实现,用户可以根据自己的兴趣和需求选择不同的新闻分类进行浏览,也可以通过搜索关键词进行精确定位。 4. 用户交互与分享:源码中包含了用户的登录注册功能和新闻内容的分享功能,用户可以通过登录账号进行个性化设置和收藏喜欢的新闻内容,也可以将新闻分享到社交媒体上与他人交流。 总之,新闻app源码是一个基础框架,开发者可以在此基础上进行二次开发和定制,根据自己的需求添加功能模块和美化界面,实现自己独特的新闻应用。 ### 回答2: Android新闻App是一个基于Android平台开发的小型项目,它的主要功能是提供最新的新闻内容给用户,并且用户可以进行浏览、搜索和分享等操作。下面是这个项目的一些关键特点和所需的源码组成部分: 1. 特点: - 用户界面友好,交互性强,提供舒适的浏览体验; - 支持实时更新,提供最新的新闻内容; - 具备搜索功能,方便用户查找感兴趣的新闻; - 支持新闻分享功能,方便用户将新闻分享给朋友; - 具备图文混排的能力,可以展示新闻的文字和图片。 2. 源码组成部分: - 主界面布局代码:定义了App的整体布局结构,包括顶部导航栏、底部工具栏和新闻显示区域等。 - 数据源代码:负责获取新闻数据,可以通过API接口获取最新的新闻内容,也可以从本地数据库获取已缓存的新闻数据。 - 新闻列表适配器代码:用于将新闻数据展示在界面上,包括标题、描述和图片等。 - 新闻详情界面代码:用于显示单篇新闻的详细内容,包括标题、正文和相关图片等。 - 搜索功能代码:实现了按关键字搜索新闻的功能,可以在已有的新闻数据中进行筛选。 - 分享功能代码:集成了社交媒体的分享SDK,方便用户将新闻内容分享给朋友。 - 图片加载和缓存代码:处理了新闻中的图片加载和本地缓存,提高了图片加载速度和用户体验。 通过以上的源码组成部分,可以完成一个基本的新闻App,用户可以在界面上浏览最新的新闻内容,进行搜索和分享操作。这个小项目可以帮助开发者理解Android开发框架和开发方式,提高编码能力和UI设计能力。 ### 回答3: 新闻app是基于Android平台开发的一个小型应用程序,可以提供用户各种最新的新闻资讯。以下是关于这个项目的源码介绍。 该项目源码主要由Java语言编写,使用了Android Studio作为开发工具。代码结构清晰,包含了主要的几个模块。 1. 用户界面模块:这个模块负责显示新闻列表和新闻详情等信息,主要包含布局文件和相应的逻辑代码。列表界面使用RecyclerView控件展示新闻列表,详情界面使用WebView展示新闻内容。 2. 网络请求模块:这个模块负责与后台服务器进行数据交互,使用了Android的HttpURLConnection类来发送请求和接收响应。请求参数可以根据实际需要进行修改,例如可以根据新闻类别进行请求。 3. 数据解析模块:这个模块负责解析从服务器返回的JSON格式的数据,转换成Java对象供应用程序使用。可以使用Android提供的JSON解析库,如Gson。 4. 数据存储模块:这个模块负责缓存新闻数据,以提高应用程序的响应速度。可以使用SharedPreferences或SQLite数据库来存储新闻数据。同时也可以使用图片缓存库,如Glide或Picasso来缓存新闻图片。 5. 用户交互模块:这个模块负责处理用户的交互行为,例如点击新闻列表项跳转到新闻详情界面,下拉刷新获取最新数据等。可以使用Android提供的相关控件和事件监听器来实现用户交互。 除了以上几个主要的模块,还可以根据需要添加其他功能,例如搜索栏、分享按钮等。 总体来说,这个新闻app的源码提供了一个完整的开发框架,初学者可以通过阅读和理解源码来学习Android应用程序的开发流程和一些常用技术。同时,也可以根据实际需求进行二次开发,添加新的功能和改进用户体验。
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值