主頁 > 軟體設計 > java仿QQ微信聊天室

java仿QQ微信聊天室

2021-04-26 10:57:36 軟體設計

話不多說,先上圖

即時通信系統:可以完成群聊、私人聊天,實時發送和顯示聊天資訊,完成好友串列,進一步可以帶表情,附件發送,聊天資訊的傳送,等 系統的主要技術socket通信技術,多執行緒技術,資料庫技術,

本系統實作的功能有:1.用戶名字登錄 2.用戶之間的群聊和查看群聊記錄3.用戶之間的私聊和私聊記錄4.動態重繪并顯示好友串列5.顯示在線人數 6.服務器可以顯示用戶在線狀態 7.能夠退出處理,

該程式可以實作局域網的聊天,只要把ip改為局域網ip即可,該系統ui中有許多沒實作的功能,比如添加好友(通過驗證成為好友),發布說說到空間,然后可以進行點贊和評論等,

有興趣的可進行拓展,,系統視頻效果觀看鏈接http://m.v.qq.com/play.html?cid=&vid=f3241h0cgfo&vuid24=7Dp8aai1Y0YPJR%2FiSwf3dQ%3D%3D&url_from=share&second_share=0&share_from=copy&pgid=page_detail&mod_id=mod_toolbar_new

下載鏈接https://download.csdn.net/download/qq_44716544/12352588

gitee下載鏈接https://gitee.com/c-xiaobai-c/java-chat-system.git

注冊,下面是資料庫的關鍵代碼,主要用于驗證賬號的唯一性和昵稱的唯一性,主要用到了二重判斷,先是判斷賬號的唯一,然后再判斷昵稱的唯一,當有一條件不足則注冊失敗,下邊代碼主要是資料庫的查詢功能,先是查詢資料庫中有沒有該賬號,然后查資料庫中有沒該昵稱,最后把注冊資訊存盤進資料庫,不過下面只展示了查詢的代碼,儲存資訊代碼類似這里不做展示,

Connectionlain c = new Connectionlain();
		          Connection conn = c.getConnect();
		          PreparedStatement ps = null;
		          ResultSet rs = null;
		          try {
	                ps = conn.prepareStatement("select *from userlist where  name=?");
		          ps.setString(1, textloginName.getText());
		          rs = ps.executeQuery();
		         if (rs.next()) {	   
			  JOptionPane.showMessageDialog(this, "該昵稱已存在,請重新填寫昵稱");
		             } else {	
                         save();
	                   	}
	} catch (SQLException e1) {
	              	// TODO Auto-generated catch block
	              	//e1.printStackTrace();
	}
	 finally{  
                         try {     
                         rs.close();    
                        ps.close();  
                        //    ct.close(); 
                       } catch (SQLException e1) {  
                      e1.printStackTrace();  
                       }  
       }	

聯系人頁面,主要用到了資料庫的讀取功能,對已登錄的用戶賬號進行讀取,還用到了jlist串列,把已登錄的用戶賬號顯示在好友列,

下面展示關鍵代碼:
        list = new JList(listModel);
	   	list.setSelectionMode(ListSelectionModel.SINGLE_SELECTION);// 設定單 一選擇模式(每次只能有一個元素被選中)
		list.setCellRenderer(new MyCellRenderer(icons));//使用自己的CellRenderer,使用圖片功能
		listScroller = new JScrollPane(list);
		listScroller.setBounds(20,300,465,350);//添加帶滾動條的list
		//list.setBounds(20,300,450,350);
		listScroller.setBackground(new Color(30, 144, 255));
		listScroller.setOpaque(false);
		listScroller.setBorder(null);	
		//this.add(listScroller,BorderLayout.CENTER);//添加帶滾動條的list
		//this.add(list);
		this.add(listScroller);
		
  listModel = new DefaultListModel();  
	    try {  
           Class.forName("com.mysql.jdbc.Driver");
   		//(2)獲取資料庫連接
   		Connection ct=DriverManager.getConnection("jdbc:mysql://127.0.0.1:3306/check?useUn   icode=true&characterEncoding=utf_8","root", "123");
   		//(3)創建SQL陳述句物件
   		Statement ps=ct.createStatement();
   		//(4)執行查詢,回傳結果集
   		 ResultSet rs=ps.executeQuery("SELECT * FROM userlist2");
   	  	String []data1=new String [1];	
           while(rs.next()){  
   		//if(rs.next()){
               //rowData可以存放多行  
     	     String st1=rs.getString(1);
       	    data1[0] =rs.getString(1);
           for(int i=0;i<data1.length;i++){
		   listModel.add(i, data1[i]);//添加登錄的用戶到串列框
          	 }
      	   System.out.println(data1[0]);	 
           }} catch (Exception e1) {  
             e1.printStackTrace();  
          } finally{       
               try {  
                   if(rs!=null){  
                   rs.close();  
                   }  
                   if(ps!=null){  
                       ps.close();  
                   }  
                   if(ct!=null){  
                       ct.close();  
                   }  
                 } catch (SQLException e1) {  
                   e1.printStackTrace();  
                }  

因為群聊和私聊的代碼相似,只展示私聊關鍵代碼;下面展示關鍵代碼:

	String serverIp="127.0.0.1";
			int serverPort=8000;
			//連接服務器,獲取套接字io流
			socket=new Socket(serverIp,serverPort);
			dis=new DataInputStream(socket.getInputStream());
			dos=new DataOutputStream(socket.getOutputStream());
			//獲取用戶名,構建,發送登陸報文
			String   id2="好友"+id1;
			String msgLogin="LOGIN#"+id2;
			dos.writeUTF(msgLogin);
			dos.flush();
			//讀取服務器回傳的資訊,判斷是否登錄成功
		String response=dis.readUTF();
		//登陸失敗
		if(response.equals("FAIL")){
			addMsg("登陸失敗");
			System.out.println("登陸失敗");
			socket.close();
			return;
		}
		//登陸成功
		if(response.equals("SUCCESS")){
			addMsg("登陸成功");
			isLogged=true;
					btnSend.setEnabled(true);
		}	
		}
		public void run(){
			//連接服務器并登錄
			try{
				login();
			}catch(IOException e){
				//addMsg("連接服務器時出現例外");
				e.printStackTrace();
				System.out.println("連接服務器時出現例外");
				return;
			}
			while(isLogged){
				try{
					String msg=dis.readUTF();
					String[] parts=msg.split("#");
					//處理服務器發送過來的報文
					if(parts[0].equals("USERLIST")){
						for(int i=1;i<parts.length;i++){
						}
					}
					
					if(parts[0].equals("LOGIN")){
						addMsg(parts[1]+"上線了");
				
					}
					if(parts[0].equals("LOGOUT")){ 						
					}
					if(parts[0].equals("TALKTO_ALL")){
						addMsg(parts[1]+"跟所有人說:"+parts[2]);
					}
					if(parts[0].equals("TALKTO")){
							addMsg(parts[1]+"跟我說:"+parts[2]);			
						}						
				}catch(IOException e){
					isLogged=false;
					e.printStackTrace();
				}
			}
		}	
	}
	//啟動執行緒
private void start(){
	clientThread=new ClientThread();
	clientThread.start();
}
private void addMsg(String msg)
	textAreaRecord.append(msg+"\n");
	//自動滾動到文本區的最后一行
	textAreaRecord.setCaretPosition(textAreaRecord.getText().length());
	new save();	
}
private void addMsg2(String msg){
	textAreaRecord.append("\t"+"\t"+"\t"+"\t"+msg+":"+"用戶"+id1+"\n");
	//自動滾動到文本區的最后一行
	textAreaRecord.setCaretPosition(textAreaRecord.getText().length());
	new save();	
}
	class save{	
		public save(){
			  try {
				   // 注冊mysql驅動程式
				   Class.forName("com.mysql.jdbc.Driver");//driver
				   System.out.println("找到Mysql資料庫驅動程式");

			    } catch (Exception e1) {
				   System.out.println("在類路徑上找不到Mysql驅動程式," + "請檢查類路徑上是否加載mysql的jar包!");
			   }
			   // (3)獲取資料庫連接
			    Connection conn = null;// 同時按下CTRL+SHIFT+O			  
			   try {			
				    conn = DriverManager.getConnection(
						   "jdbc:mysql://127.0.0.1:3306/check?charcterEncoding=utf-8", "root", "123");//(url,username,password)
				    System.out.println("建立資料庫連接成功");
				

			     } catch (Exception e1) {
			      	e1.printStackTrace();
			      	System.out.println("創建資料庫連接失敗!");
			      }

			//     (4)創建一個SQL陳述句執行(需要在Java執行SQL陳述句)
			      Statement stmt = null;
			       try {
			     	// 通過conn物件創建SQL陳述句物件
			     	stmt = conn.createStatement();
			     } catch (Exception e1) {
			     	e1.printStackTrace();
			   }


			    String sql = "INSERT INTO usermessage2 (message2) VALUES('"+textAreaRecord.getText()+"')";
			
			
			    try {
			  	// 執行SQL陳述句
			    	stmt.executeUpdate(sql);
			  	//  JOptionPane op7=new JOptionPane();
			   //  op7.showMessageDialog(null, "注冊成功");
			   	System.out.println("資料插入成功");
		//	   	JOptionPane.showMessageDialog(this, "注冊成功");
				
			    } catch (Exception e1) {
			    	e1.printStackTrace();
			  	System.out.println("插入失敗");
			   }
			
		  	//(6)關閉資源
			   try {
				  stmt.close();
				  conn.close();
			      } catch (Exception e1) {
				    e1.printStackTrace();

服務器 主要用到了網路通信和多執行緒的功能

	try{
			//獲取serverip和serverPoet
			String serverIp=textServerIP.getText();
			int serverPort=Integer.parseInt(textPort.getText());
			//創建套接字地址
			SocketAddress socketAddress=new InetSocketAddress(serverIp,serverPort);
			//創建ServerSocket,系結套接字地址
			server=new ServerSocket();
			server.bind(socketAddress);
			//修改判斷服務器是否運行的標識變數
			isRunning=true;
			
			btnStart.setEnabled(false);
			btnStop.setEnabled(true);
			addMsg("服務器啟動成功");
			System.out.println("服務器啟動成功");				
		}catch(IOException e){
			System.out.println("服務器啟動失敗");
			e.printStackTrace();
			isRunning=false;
		}
	}
	public void run(){
		startServer();
		//當服務器處于運行狀態時,回圈監聽客戶端的連接請求
		while(isRunning){
			try{
				Socket socket=server.accept();
				//創建與客戶端互動的執行緒
				Thread thread=new Thread(new ClientHandler(socket));
				thread.start();
			}catch(IOException e){
				System.out.println("還沒連接");
			}
		}
	}	
}

class ClientHandler implements Runnable{
	private Socket socket;
	private DataInputStream dis;
	private DataOutputStream dos;
	private boolean isConnected;
	private String username;
	public ClientHandler(Socket socket){
		this.socket=socket;
		try{
			this.dis=new DataInputStream(socket.getInputStream());
			this.dos=new DataOutputStream(socket.getOutputStream());
			isConnected=true;
		}catch(IOException e){
			isConnected=false;
			e.printStackTrace();
		}
	}
	public void run(){
		while(isRunning&&isConnected){
			try{
				//讀取客戶端發送的報文
				String msg=dis.readUTF();
				String[] parts=msg.split("#");
				if(parts[0].equals("LOGIN")){
					String loginUsername=parts[1];		
					Date date3=new Date();	
					SimpleDateFormat f2=new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
					String d3=f2.format(date3);
					String q="用戶:"+parts[1]+""+"已登錄"+"\t"+d3;
				   addMsg(q);	
					//如果該用戶已登錄,則回傳失敗報文,否則回傳成功報文
					if(clientHandlerMap.containsKey(loginUsername)){
						dos.writeUTF("FAIL");
					}else{
						dos.writeUTF("SUCCESS");
						//將此客戶端處理執行緒的資訊添加到clientHandlerMap中
						clientHandlerMap.put(loginUsername, this);
						//將現有用戶的資訊發給新用戶
						StringBuffer msgUserList=new StringBuffer();
						msgUserList.append("USERLIST");
					
						for(String username : clientHandlerMap.keySet()){
							msgUserList.append(username +"#");
						}
						dos.writeUTF(msgUserList.toString());
						//將新登錄的用戶資訊發給其他用戶
						String msgLogin="LOGIN#"+loginUsername;
						broadcastMsg(loginUsername,msgLogin);
						//存盤登錄的用戶名
						this.username=loginUsername;
					}	
				}
				if(parts[0].equals("LOGOUT")){
					clientHandlerMap.remove(username);
					String msgLogout="LOGOUT#"+username;
					//broadcastMsg(username,msgLogout);
					isConnected=false;
					socket.close();
				}
				
				if(parts[0].equals("TALKTO_ALL")){
			String msgTalkToAll="TALKTO_ALL#"+username+"#"+parts[1];	 Date  date3=new Date();	
					SimpleDateFormat f2=new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
					String d3=f2.format(date3);
					String p="用戶 "+username+"對所有人說"+parts[1]+"\t"+d3;
					addMsg2(p);			
					broadcastMsg(username,msgTalkToAll);}		
				if(parts[0].equals("TALKTO")){
						ClientHandler clientHandler=clientHandlerMap.get(parts[1]);
						if(null!=clientHandler){
							String msgTalkTo="TALKTO#"+username+"#"+parts[2];	
							Date date3=new Date();	
				SimpleDateFormat f2=new   SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
							String d3=f2.format(date3);
							String p2="用戶 "+clientHandler+" 對"+username+"說: "+parts[2]+"\t"+d3;
							addMsg2(p2);
							
							clientHandler.dos.writeUTF(msgTalkTo);
							clientHandler.dos.flush();
						}		
				}			
			}catch(IOException e){
				isConnected=false;
				e.printStackTrace();
			}
	}	
}

/**
 * 將某個用戶發來的資訊廣播個其他用戶
 */
private void broadcastMsg(String fromUsername,String msg) throws IOException{
	 for(String toUserName : clientHandlerMap.keySet()){
		if(fromUsername.equals(toUserName)==false){
			DataOutputStream dos=clientHandlerMap.get(toUserName).dos;
			dos.writeUTF(msg);
			dos.flush();
		}
	}
}
}
/**
 * 添加訊息到文本框
   */
private void addMsg(String msg){
	//在文本區添加一條訊息,并加上換行
	textAreaRecord.append(msg+"\n");
	textAreaRecord.setCaretPosition(textAreaRecord.getText().length());

轉載請註明出處,本文鏈接:https://www.uj5u.com/ruanti/280243.html

標籤:其他

上一篇:第七章 青蛙跳臺階和漢諾塔問題

下一篇:寫給Java 工程師們一封信

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 面試突擊第一季,第二季,第三季

    第一季必考 https://www.bilibili.com/video/BV1FE411y79Y?from=search&seid=15921726601957489746 第二季分布式 https://www.bilibili.com/video/BV13f4y127ee/?spm_id_fro ......

    uj5u.com 2020-09-10 05:35:24 more
  • 第三單元作業總結

    1.前言 這應該是本學期最后一次寫作業總結了吧。總體來說,對作業的節奏也差不多掌握了,作業做起來的效率也更高了。雖然和之前的作業一樣,作業中都要用到新的知識,但是相比之前,更加懂得了如何利用工具以及資料。雖然之間卡過殼,但總體而言,這幾次作業還算完成的比較好。 2.作業程序總結 相比前兩個單元,此單 ......

    uj5u.com 2020-09-10 05:35:41 more
  • 北航OO(2020)第四單元博客作業暨課程總結博客

    北航OO(2020)第四單元博客作業暨課程總結博客 本單元作業的架構設計 在本單元中,由于UML圖具有比較清晰的樹形結構,因此我對其中需要進行查詢操作的元素進行了包裝,在樹的父節點中存盤所有孩子的參考。考慮到性能問題,我采用了快取機制,一次查詢后盡可能快取已經遍歷過的資訊,以減少遍歷次數。 本單元我 ......

    uj5u.com 2020-09-10 05:35:48 more
  • BUAA_OO_第四單元

    一、UML決議器設計 ? 先看下題目:第四單元實作一個基于JDK 8帶有效性檢查的UML(Unified Modeling Language)類圖,順序圖,狀態圖分析器 MyUmlInteraction,實際上我們要建立一個有向圖模型,UML中的物件(元素)可能與同級元素連接,也可與低級元素相連形成 ......

    uj5u.com 2020-09-10 05:35:54 more
  • 6.1邏輯運算子

    邏輯運算子 1. && 短路與 運算式1 && 運算式2 01.運算式1為true并且運算式2也為true 整體回傳為true 02.運算式1為false,將不會執行運算式2 整體回傳為false 03.只要有一個運算式為false 整體回傳為false 2. || 短路或 運算式1 || 運算式2 ......

    uj5u.com 2020-09-10 05:35:56 more
  • BUAAOO 第四單元 & 課程總結

    1. 第四單元:StarUml檔案決議 本單元采用了圖模型決議UML。 UML檔案可以抽象為圖、子圖、邊的邏輯結構。 在實作中,圖的節點包括類、介面、屬性,子圖包括狀態圖、順序圖等。 采用了三次遍歷UML元素的方法建圖,第一遍遍歷建點,第二、三次遍歷設定屬性、連邊,實作圖物件的初始化。這里借鑒了一些 ......

    uj5u.com 2020-09-10 05:36:06 more
  • 談談我對C# 多型的理解

    面向物件三要素:封裝、繼承、多型。 封裝和繼承,這兩個比較好理解,但要理解多型的話,可就稍微有點難度了。今天,我們就來講講多型的理解。 我們應該經常會看到面試題目:請談談對多型的理解。 其實呢,多型非常簡單,就一句話:呼叫同一種方法產生了不同的結果。 具體實作方式有三種。 一、多載 多載很簡單。 p ......

    uj5u.com 2020-09-10 05:36:09 more
  • Python 資料驅動工具:DDT

    背景 python 的unittest 沒有自帶資料驅動功能。 所以如果使用unittest,同時又想使用資料驅動,那么就可以使用DDT來完成。 DDT是 “Data-Driven Tests”的縮寫。 資料:http://ddt.readthedocs.io/en/latest/ 使用方法 dd. ......

    uj5u.com 2020-09-10 05:36:13 more
  • Python里面的xlrd模塊詳解

    那我就一下面積個問題對xlrd模塊進行學習一下: 1.什么是xlrd模塊? 2.為什么使用xlrd模塊? 3.怎樣使用xlrd模塊? 1.什么是xlrd模塊? ?python操作excel主要用到xlrd和xlwt這兩個庫,即xlrd是讀excel,xlwt是寫excel的庫。 今天就先來說一下xl ......

    uj5u.com 2020-09-10 05:36:28 more
  • 當我們創建HashMap時,底層到底做了什么?

    jdk1.7中的底層實作程序(底層基于陣列+鏈表) 在我們new HashMap()時,底層創建了默認長度為16的一維陣列Entry[ ] table。當我們呼叫map.put(key1,value1)方法向HashMap里添加資料的時候: 首先,呼叫key1所在類的hashCode()計算key1 ......

    uj5u.com 2020-09-10 05:36:38 more
最新发布
  • 【中介者設計模式詳解】C/Java/JS/Go/Python/TS不同語言實作

    * 中介者模式是一種行為型設計模式,它可以用來減少類之間的直接依賴關系,
    * 將物件之間的通信封裝到一個中介者物件中,從而使得各個物件之間的關系更加松散。
    * 在中介者模式中,物件之間不再直接相互互動,而是通過中介者來中轉訊息。 ......

    uj5u.com 2023-04-20 08:20:47 more
  • 露天煤礦現場調研和交流案例分享

    他們集團的資訊化公司及研究院在一個礦區正在做智能礦山的統一平臺的 試點,專案投資大概1億,包括了礦山的各方面的內容,顯示得我們這次交流有點多余。他們2年前開始做智能礦山的規劃,有很多煤礦行業專家的加持,他們的描述是非常完美,但是去年底應該上線的平臺,現在還沒有看到影子。他們確實有很多場景需求,但是被... ......

    uj5u.com 2023-04-20 08:20:25 more
  • 《社區人員管理》實戰案例設計&個人案例分享

    設計是一個讓人夢想成真程序,開始編碼、測驗、除錯之前進行需求分析和架構設計,才能保證關鍵方面都做正確 ......

    uj5u.com 2023-04-20 08:20:17 more
  • 軟體架構生態化-多角色交付的探索實踐

    作為一個技術架構師,不僅僅要緊跟行業技術趨勢,還要結合研發團隊現狀及痛點,探索新的交付方案。在日常中,你是否遇到如下問題 “ 業務需求排期長研發是瓶頸;非研發角色感受不到研發技改提效的變化;引入ISV 團隊又擔心質量和安全,培訓周期長“等等,基于此我們探索了一種新的技術體系及交付方案來解決如上問題。 ......

    uj5u.com 2023-04-20 08:20:10 more
  • 【中介者設計模式詳解】C/Java/JS/Go/Python/TS不同語言實作

    * 中介者模式是一種行為型設計模式,它可以用來減少類之間的直接依賴關系,
    * 將物件之間的通信封裝到一個中介者物件中,從而使得各個物件之間的關系更加松散。
    * 在中介者模式中,物件之間不再直接相互互動,而是通過中介者來中轉訊息。 ......

    uj5u.com 2023-04-20 08:19:44 more
  • 露天煤礦現場調研和交流案例分享

    他們集團的資訊化公司及研究院在一個礦區正在做智能礦山的統一平臺的 試點,專案投資大概1億,包括了礦山的各方面的內容,顯示得我們這次交流有點多余。他們2年前開始做智能礦山的規劃,有很多煤礦行業專家的加持,他們的描述是非常完美,但是去年底應該上線的平臺,現在還沒有看到影子。他們確實有很多場景需求,但是被... ......

    uj5u.com 2023-04-20 08:19:07 more
  • 《社區人員管理》實戰案例設計&個人案例分享

    設計是一個讓人夢想成真程序,開始編碼、測驗、除錯之前進行需求分析和架構設計,才能保證關鍵方面都做正確 ......

    uj5u.com 2023-04-20 08:18:57 more
  • 軟體架構生態化-多角色交付的探索實踐

    作為一個技術架構師,不僅僅要緊跟行業技術趨勢,還要結合研發團隊現狀及痛點,探索新的交付方案。在日常中,你是否遇到如下問題 “ 業務需求排期長研發是瓶頸;非研發角色感受不到研發技改提效的變化;引入ISV 團隊又擔心質量和安全,培訓周期長“等等,基于此我們探索了一種新的技術體系及交付方案來解決如上問題。 ......

    uj5u.com 2023-04-20 08:18:49 more
  • 05單件模式

    #經典的單件模式 public class Singleton { private static Singleton uniqueInstance; //一個靜態變數持有Singleton類的唯一實體。 // 其他有用的實體變數寫在這里 //構造器宣告為私有,只有Singleton可以實體化這個類! ......

    uj5u.com 2023-04-19 08:42:51 more
  • 【架構與設計】常見微服務分層架構的區別和落地實踐

    軟體工程的方方面面都遵循一個最基本的道理:沒有銀彈,架構分層模型更是如此,每一種都有各自優缺點,所以請根據不同的業務場景,并遵循簡單、可演進這兩個重要的架構原則選擇合適的架構分層模型即可。 ......

    uj5u.com 2023-04-19 08:42:41 more