1.提交1html
結對實現中綴表達式轉後綴表達式的功能 MyBC.java 結對實現從上面功能中獲取的表達式中實現後綴表達式求值的功能,調用MyDC.java 上傳測試代碼運行結果截圖和碼雲連接
實驗原理java
實驗代碼git
中綴轉後綴MyBC.java算法
一、設立空棧用於存放操做符 二、從左到右掃描中綴式,若遇到操做數,直接輸出,並以空格做爲分隔符; 三、若遇到運算符,則與棧頂元素比較,比棧頂級別高則進棧,不然退出棧頂元素並輸出,並以空格做爲分隔符,優先級從高到低爲')' '/' (÷*) (+-) ; 四、若遇到左括號,進棧;若遇到右括號,則一直退棧輸出,直到退到左括號止; 五、當棧變成空時,輸出的結果即爲後綴表達式。
import java.util.Stack; public class MyBC { String C = new String(); String End = ""; //存儲數字的字符串 public void ChangeString(String str) { C = str; } public String ChangeOrder() { Stack store = new Stack(); //建立一個存儲字符的棧 for (int i = 0; i < C.length(); i++) { char op = C.charAt(i); //將索引值爲i處的字符的值返回 if (op >= '0' && op <= '9') { End = End + op; } else if (op == '(') { store.push(op); } else if (op == '+' || op == '-' || op == '*' || op == '÷'|| op == '/') { End = End + " "; if (store.empty()) { store.push(op); } else if (compareValue(op) > compareValue((char) store.peek())) { //比較運算符優先級 store.push(op); } else { End = End + String.valueOf(store.pop()); i--; } } else if (op == ')') { while ((char) store.peek() != '(') { End = End + " " + String.valueOf(store.pop()); } store.pop(); } } while (!store.empty()) { End = End + " " + String.valueOf(store.pop()); } return End; } public int compareValue(char chi) { int number = 0; switch (chi) { case '(': number = 1; break; case '+': case '-': number = 2; break; case '*': case '÷': number = 3; break; case '/': number = 4; break; case ')': number = 5; break; default: number = 0; break; } return number; } }
後綴表達式計算MyDC.java編程
一、設置一個空棧用於存儲數字元素; 二、從左到右掃描後綴表達式,遇操做數,進棧; 三、若遇運算符,則從棧頂彈出兩個操做數,執行對應的運算,並將運算後的結果壓棧棧; 四、重複執行直至後綴表達式掃描完畢,彈出棧中的最後一個元素即爲表達式的值。
import java.util.StringTokenizer; import java.util.Stack; public class MyDC { String q; Stack stack; public MyDC() { stack = new Stack(); } void set(String question) { //輸入後續排列的字符串 q = question; } public Rational get() { Rational op1 = new Rational(); Rational op2 = new Rational(); Rational result = new Rational(); result.setNumerator(0); StringTokenizer token = new StringTokenizer(q, " "); String temp; while (token.hasMoreTokens()) { temp = token.nextToken(); if (Isop(temp) == 1) { //遇到操做符,彈出棧頂的兩個數進行運算 op2 = (Rational) stack.pop(); op1 = (Rational) stack.pop();//彈出最上面兩個操做數 result = cal(temp.charAt(0), op1, op2);//根據運算符進行運算 stack.push(result);//將計算結果壓棧 } else { Rational num = new Rational(); num.setNumerator(Integer.parseInt(temp)); stack.push(num);//操做數入棧 } } return result; } Rational cal(char op, Rational a, Rational b) { //對棧頂彈出的兩個數進行運算 Rational c = new Rational(); switch (op) { case '+': c = a.add(b); break; case '-': c = a.sub(b); break; case '*': c = a.muti(b); break; case '÷': case '/': if(b.getNumerator()==0) { System.out.println("生成的算式計算時出現了分母爲0的狀況!"); System.exit(0); } else { c = a.div(b); break; } default: System.out.println("Wrong!"); } return c; } int Isop(String op) { //判斷是否是運算符 if (op.equals("+") || op.equals("-") || op.equals("*") || op.equals("÷") || op.equals("/")) { return 1; } else { return 0; } } }
public class Rational { int numerator = 1 ; //分子 int denominator = 1; //分母 void setNumerator(int a) { //設置分子 int c=f(Math.abs(a),denominator); //計算最大公約數 numerator = a/c; denominator = denominator/c; if(numerator<0&&denominator<0) { numerator = -numerator; denominator = -denominator; } } void setDenominator(int b) { //設置分母 int c=f(numerator,Math.abs(b)); //計算最大公約數 numerator = numerator/c; denominator = b/c; if(numerator<0&&denominator<0) { numerator = -numerator; denominator = -denominator; } } int getNumerator() { return numerator; } int getDenominator() { return denominator; } int f(int a,int b) { //求a和b的最大公約數 if(a==0) { return 1; } if(a<b) { int c=a; a=b; b=c; } int r=a%b; while(r!=0) { a=b; b=r; r=a%b; } return b; } Rational add(Rational r) { //加法運算 int a=r.getNumerator(); int b=r.getDenominator(); int newNumerator=numerator*b+denominator*a; //計算出新分子 int newDenominator=denominator*b; //計算出新分母 Rational result=new Rational(); result.setNumerator(newNumerator); result.setDenominator(newDenominator); return result; } Rational sub(Rational r) { //減法運算 int a=r.getNumerator(); int b=r.getDenominator(); int newNumerator=numerator*b-denominator*a; int newDenominator=denominator*b; Rational result=new Rational(); result.setNumerator(newNumerator); result.setDenominator(newDenominator); return result; } Rational muti(Rational r) { //乘法運算 int a=r.getNumerator(); int b=r.getDenominator(); int newNumerator=numerator*a; int newDenominator=denominator*b; Rational result=new Rational(); result.setNumerator(newNumerator); result.setDenominator(newDenominator); return result; } Rational div(Rational r) { //除法運算 int a=r.getNumerator(); int b=r.getDenominator(); int newNumerator=numerator*b; int newDenominator=denominator*a; Rational result=new Rational(); if(a==0) { System.out.println("該算式無解"); result.setNumerator(0); } else { result.setNumerator(newNumerator); result.setDenominator(newDenominator); } return result; } }
測試代碼Test數組
import java.util.*; public class Test { public static void main(String[] args) { String question = ""; String question1 = ""; Scanner scanner = new Scanner(System.in); System.out.println("請輸入題目:"); question = scanner.nextLine(); MyBC change = new MyBC(); change.ChangeString(question); question1 = change.ChangeOrder(); System.out.println(question1); MyDC getanswer = new MyDC(); getanswer.set(question1); Rational answer = getanswer.get(); int a = answer.getNumerator(); int b = answer.getDenominator(); float result = (float)a/b; System.out.println("結果爲(保留兩位小數):"); System.out.println(String.format("%.2f",result)); } }
測試結果截圖安全
2.提交2服務器
一、基於Java Socket實現客戶端/服務器功能,傳輸方式用TCP 二、客戶端讓用戶輸入中綴表達式,而後把中綴表達式調用MyBC.java的功能轉化爲後綴表達式,把後綴表達式經過網絡發送給服務器 三、服務器接收到後綴表達式,調用MyDC.java的功能計算後綴表達式的值,把結果發送給客戶端 四、客戶端顯示服務器發送過來的結果 五、上傳測試結果截圖和碼雲連接
實驗原理網絡
Client
端的IP地址
ipconfig
查看;InetAddress
類中的geetByName
方法獲取輸入的IP地址;Socket
的構造方法;
public Socket(java.lang.String host, int port)
其中,host
是IP地址,port
是端口號。getInputStream()
方法得到一個輸入流getOutputStream()
方法得到一個輸出流ServerSocke
t實現套接字accept()
方法鏈接客戶端和服務器的套接字getInputStream()
方法得到一個輸入流getOutputStream()
方法得到一個輸出流實驗代碼socket
import java.io.*; import java.net.*; import java.util.Scanner; public class Client { public static void main(String args[]) { Socket mysocket; DataInputStream in=null; DataOutputStream out=null; Scanner scanner = new Scanner(System.in); try{ System.out.print("輸入服務器的IP:"); String IP = scanner.nextLine(); InetAddress address=InetAddress.getByName(IP); mysocket = new Socket(address, 2010); in=new DataInputStream(mysocket.getInputStream()); out=new DataOutputStream(mysocket.getOutputStream()); System.out.println("請輸入中綴表達式:"); while (scanner.hasNext()) { String question = scanner.next(); MyBC change = new MyBC(); change.ChangeString(question); String question1 = change.ChangeOrder(); System.out.println("後綴表達式爲:"+question1); out.writeUTF(question1); String s=in.readUTF(); //in讀取信息,堵塞狀態 System.out.println("客戶收到服務器的回答:"+s); Thread.sleep(500); System.out.println("請輸入中綴表達式:"); } } catch(Exception e) { System.out.println("服務器已斷開"+e); } } }
import java.io.*; import java.net.*; public class Server { public static void main(String[] args) { ServerSocket serverForClient = null; Socket socketOnServer = null; DataOutputStream out = null; DataInputStream in = null; String question = ""; try { serverForClient = new ServerSocket(2010); } catch (IOException e) { System.out.println(e); } System.out.println("等待客戶呼叫"); try { socketOnServer = serverForClient.accept(); out = new DataOutputStream(socketOnServer.getOutputStream()); in = new DataInputStream(socketOnServer.getInputStream()); while(true) { question = in.readUTF(); // in讀取信息,堵塞狀態 System.out.println("服務器收到客戶傳遞的後綴表達式爲:" + question); MyDC getanswer = new MyDC(); getanswer.set(question); Rational answer = getanswer.get(); int a = answer.getNumerator(); int b = answer.getDenominator(); float result = (float) a / b; System.out.println("計算出的結果爲"+String.format("%.2f",result)); out.writeUTF(String.format("%.2f",result)); Thread.sleep(500); } } catch (Exception e) { System.out.println("客戶已斷開" + e); } } }
測試結果截圖
客戶端
服務器
3.提交3
一、基於Java Socket實現客戶端/服務器功能,傳輸方式用TCP 二、客戶端讓用戶輸入中綴表達式,而後把中綴表達式調用MyBC.java的功能轉化爲後綴表達式,把後綴表達式用3DES或AES算法加密後經過網絡把密文發送給服務器 三、服務器接收到後綴表達式表達式後,進行解密(和客戶端協商密鑰,能夠用數組保存),而後調用MyDC.java的功能計算後綴表達式的值,把結果發送給客戶端 四、客戶端顯示服務器發送過來的結果 五、上傳測試結果截圖和碼雲連接
實驗原理
一、獲取密鑰生成器KeyGenerator kg=KeyGenerator.getInstance("DESede") 二、初始化密鑰生成器kg.init(168); 三、生成密鑰SecretKey k=kg.generateKey( ); 四、獲取主要編碼格式byte[ ] kb=k.getEncoded( ); 五、傳送密鑰長度及密鑰內容 六、建立密碼器Cipher cp=Cipher.getInstance("DESede"); 七、初始化密碼器cp.init(Cipher.ENCRYPT_MODE, k); 八、獲取等待加密的明文byte ptext[]=s.getBytes("UTF8"); 九、執行加密byte []ptext=cp.doFinal(ctext)
一、接收密鑰長度 二、接收密鑰內容 三、建立密碼器Cipher cp=Cipher.getInstance("DESede"); 四、初始化密碼器cp.init(Cipher.DECRYPT_MODE, k); 五、執行解密byte []ptext=cp.doFinal(ctext)
實驗代碼
import java.io.*; import java.net.*; import javax.crypto.*; import java.util.Scanner; public class Client { public static void main(String args[]) throws Exception{ Socket mysocket; DataInputStream in=null; DataOutputStream out=null; Scanner scanner = new Scanner(System.in); try { System.out.print("輸入服務器的IP:"); String IP = scanner.nextLine(); InetAddress address=InetAddress.getByName(IP); mysocket = new Socket(address, 2010); in = new DataInputStream(mysocket.getInputStream()); out = new DataOutputStream(mysocket.getOutputStream()); KeyGenerator kg = KeyGenerator.getInstance("DESede"); kg.init(168); SecretKey k = kg.generateKey(); byte []kb = k.getEncoded(); out.writeUTF(kb.length+ ""); System.out.println("產生的密鑰爲"); for(int i=0;i<kb.length;i++) { System.out.print(kb[i]+ " "); out.writeUTF(kb[i] +""); } System.out.println("\n請輸入中綴表達式:"); while(scanner.hasNext()) { String question = scanner.next(); MyBC change = new MyBC(); change.ChangeString(question); String question1 = change.ChangeOrder(); System.out.println("後綴表達式爲:"+question1); Cipher cp = Cipher.getInstance("DESede"); cp.init(Cipher.ENCRYPT_MODE,k); byte ptext[] = question1.getBytes("UTF8"); byte ctext[] = cp.doFinal(ptext); out.writeUTF(ctext.length + ""); for(int i=0;i<ctext.length;i++) { out.writeUTF(ctext[i] +""); } String s=in.readUTF(); //in讀取信息,堵塞狀態 System.out.println("客戶收到服務器的回答:"+s); Thread.sleep(500); System.out.println("請輸入中綴表達式:"); } } catch (IOException e) { System.out.println("服務器已斷開"+e); } } }
import java.io.*; import java.net.*; import javax.crypto.*; import javax.crypto.spec.*; public class Server { public static void main(String[] args) throws Exception{ ServerSocket serverForClient = null; Socket socketOnServer = null; DataOutputStream out = null; DataInputStream in = null; try { serverForClient = new ServerSocket(2010); } catch (IOException e) { System.out.println(e); } System.out.println("等待客戶呼叫"); try { socketOnServer = serverForClient.accept(); out = new DataOutputStream(socketOnServer.getOutputStream()); in = new DataInputStream(socketOnServer.getInputStream()); String keylength = in.readUTF(); byte []kb = new byte[Integer.parseInt(keylength)]; System.out.println("收到的密鑰爲:"); for(int i = 0;i<Integer.parseInt(keylength);i++) { String str = in.readUTF(); kb[i] = Byte.parseByte(str); System.out.print(kb[i] + " "); } while(true) { SecretKeySpec k = new SecretKeySpec(kb, "DESede"); Cipher cp = Cipher.getInstance("DESede"); cp.init(Cipher.DECRYPT_MODE, k); String clength = in.readUTF(); byte ctext[] = new byte[Integer.parseInt(clength)]; for (int i = 0;i<Integer.parseInt(clength);i++) { String temp = in.readUTF(); ctext[i] = Byte.parseByte(temp); } byte[] ptext = cp.doFinal(ctext); String question = new String(ptext,"UTF8"); System.out.print("\n後綴表達式爲:"+ question); MyDC getanswer = new MyDC(); getanswer.set(question); Rational answer = getanswer.get(); int a = answer.getNumerator(); int b = answer.getDenominator(); float result = (float) a / b; System.out.println("\n計算出的結果爲"+String.format("%.2f",result)); out.writeUTF(String.format("%.2f",result)); Thread.sleep(500); } } catch (IOException e) { System.out.println("客戶已斷開" + e); } } }
測試結果截圖
客戶端
服務器
4.提交4
一、基於Java Socket實現客戶端/服務器功能,傳輸方式用TCP 二、客戶端讓用戶輸入中綴表達式,而後把中綴表達式調用MyBC.java的功能轉化爲後綴表達式,把後綴表達式用3DES或AES算法加密經過網絡把密文發送給服務器 三、客戶端和服務器用DH算法進行3DES或AES算法的密鑰交換 四、服務器接收到後綴表達式表達式後,進行解密,而後調用MyDC.java的功能計算後綴表達式的值,把結果發送給客戶端 五、客戶端顯示服務器發送過來的結果 六、上傳測試結果截圖和碼雲連接
實驗原理
Key_DH
和KeyAgree
主類的參數改成由客戶端和服務器傳遞,並修改方法名
public static void DH(String str1,String str2) throws Exception
public static void Agree(String str1,String str2)
KeyAgreement ka=KeyAgreement.getInstance("DH")
;ka.init(prk)
;ka.doPhase(pbk,true)
;byte[ ] sb=ka.generateSecret()
;實驗代碼
import javax.crypto.Cipher; import javax.crypto.spec.SecretKeySpec; import java.io.*; import java.security.*; import java.util.Scanner; import java.net.*; public class Client { public static void main(String[] args) { Socket mysocket; DataInputStream in=null; DataOutputStream out=null; Scanner scanner = new Scanner(System.in); try { System.out.print("輸入服務器的IP:"); String IP = scanner.nextLine(); InetAddress address=InetAddress.getByName(IP); mysocket = new Socket(address, 2010); in = new DataInputStream(mysocket.getInputStream()); out = new DataOutputStream(mysocket.getOutputStream()); Key_DH.DH("Lpub.dat","Lpri.dat"); FileInputStream my = new FileInputStream("Lpub.dat"); ObjectInputStream mypub = new ObjectInputStream(my); Key kp = (Key) mypub.readObject(); ByteArrayOutputStream DH = new ByteArrayOutputStream(); ObjectOutputStream myDH = new ObjectOutputStream(DH); myDH.writeObject(kp); byte []pub = DH.toByteArray(); out.writeUTF(pub.length+""); for(int i=0;i<pub.length;i++) { out.writeUTF(pub[i]+ ""); } Thread.sleep(1000); int length = Integer.parseInt(in.readUTF()); byte cpub[] = new byte[length]; for(int i=0;i<length;i++) { String temp = in.readUTF(); cpub[i] = Byte.parseByte(temp); } ByteArrayInputStream ckey1 = new ByteArrayInputStream(cpub); ObjectInputStream ckey = new ObjectInputStream(ckey1); Key k = (Key) ckey.readObject(); FileOutputStream f2 = new FileOutputStream("W1pub.dat"); ObjectOutputStream b2 = new ObjectOutputStream(f2); b2.writeObject(k); KeyAgree.Agree("W1pub.dat","Lpri.dat"); FileInputStream f = new FileInputStream("sb.dat"); byte[] keysb = new byte[24]; f.read(keysb); System.out.println("公共密鑰爲:"); for (int i = 0;i<24;i++) { System.out.print(keysb[i]+" "); } System.out.println("\n請輸入中綴表達式:"); while(scanner.hasNext()) { String question = scanner.next(); MyBC change = new MyBC(); change.ChangeString(question); String question1 = change.ChangeOrder(); System.out.println("後綴表達式爲:" + question1); SecretKeySpec k1 = new SecretKeySpec(keysb, "DESede"); Cipher cp = Cipher.getInstance("DESede"); cp.init(Cipher.ENCRYPT_MODE, k1); byte ptext[] = question1.getBytes("UTF8"); byte ctext[] = cp.doFinal(ptext); System.out.println("加密後的後綴表達式爲:"); for (int i = 0; i < ctext.length; i++) { System.out.print(ctext[i] + " "); } out.writeUTF(ctext.length + ""); for (int i = 0; i < ctext.length; i++) { out.writeUTF(ctext[i] + ""); } String s=in.readUTF(); //in讀取信息,堵塞狀態 System.out.println("\n客戶收到服務器的回答:"+s); Thread.sleep(500); System.out.println("請輸入中綴表達式:"); } } catch (Exception e) { System.out.println(e); } } }
import javax.crypto.Cipher; import javax.crypto.spec.SecretKeySpec; import java.io.*; import java.net.ServerSocket; import java.net.Socket; import java.security.*; public class Server { public static void main(String[] args) throws Exception { ServerSocket serverForClient = null; Socket socketOnServer = null; DataOutputStream out = null; DataInputStream in = null; try { serverForClient = new ServerSocket(2010); } catch (IOException e) { System.out.println(e); } System.out.println("等待客戶呼叫"); try { socketOnServer = serverForClient.accept(); out = new DataOutputStream(socketOnServer.getOutputStream()); in = new DataInputStream(socketOnServer.getInputStream()); Key_DH.DH("Wpub.dat","Wpri.dat"); int length = Integer.parseInt(in.readUTF()); byte cpub[] = new byte[length]; for(int i=0;i<length;i++) { String temp = in.readUTF(); cpub[i] = Byte.parseByte(temp); } ByteArrayInputStream ckey1 = new ByteArrayInputStream(cpub); ObjectInputStream ckey = new ObjectInputStream(ckey1); Key k1 = (Key) ckey.readObject(); FileOutputStream f2 = new FileOutputStream("Lpub.dat"); ObjectOutputStream b2 = new ObjectOutputStream(f2); b2.writeObject(k1); FileInputStream my = new FileInputStream("Wpub.dat"); ObjectInputStream mypub = new ObjectInputStream(my); Key kp = (Key) mypub.readObject(); ByteArrayOutputStream DH = new ByteArrayOutputStream(); ObjectOutputStream myDH = new ObjectOutputStream(DH); myDH.writeObject(kp); byte []pub = DH.toByteArray(); out.writeUTF(pub.length+""); for(int i=0;i<pub.length;i++) { out.writeUTF(pub[i]+ ""); } KeyAgree.Agree("Lpub.dat","Wpri.dat"); FileInputStream f = new FileInputStream("sb.dat"); byte[] keysb = new byte[24]; f.read(keysb); System.out.println("公共密鑰爲:"); for (int i = 0;i<24;i++) { System.out.print(keysb[i]+" "); } while(true) { SecretKeySpec k = new SecretKeySpec(keysb, "DESede"); Cipher cp = Cipher.getInstance("DESede"); cp.init(Cipher.DECRYPT_MODE, k); String clength = in.readUTF(); byte ctext[] = new byte[Integer.parseInt(clength)]; for (int i = 0; i < Integer.parseInt(clength); i++) { String temp = in.readUTF(); ctext[i] = Byte.parseByte(temp); } byte[] ptext = cp.doFinal(ctext); String question = new String(ptext, "UTF8"); System.out.print("\n解密後的後綴表達式爲:" + question); MyDC getanswer = new MyDC(); getanswer.set(question); Rational answer = getanswer.get(); int a = answer.getNumerator(); int b = answer.getDenominator(); float result = (float) a / b; System.out.println("\n計算出的結果爲"+String.format("%.2f",result)); out.writeUTF(String.format("%.2f",result)); Thread.sleep(500); } } catch(Exception e) { System.out.println("客戶已斷開" + e); } } }
測試結果截圖
客戶端
服務器
5.任務五
一、基於Java Socket實現客戶端/服務器功能,傳輸方式用TCP 二、客戶端讓用戶輸入中綴表達式,而後把中綴表達式調用MyBC.java的功能轉化爲後綴表達式,把後綴表達式用3DES或AES算法加密經過網絡把密文和明文的MD5値發送給服務器 三、客戶端和服務器用DH算法進行3DES或AES算法的密鑰交換 四、服務器接收到後綴表達式表達式後,進行解密,解密後計算明文的MD5值,和客戶端傳來的MD5進行比較,一致則調用MyDC.java的功能計算後綴表達式的值,把結果發送給客戶端 五、客戶端顯示服務器發送過來的結果 六、上傳測試結果截圖和碼雲連接
實驗原理
實驗代碼
import javax.crypto.Cipher; import javax.crypto.spec.SecretKeySpec; import java.io.*; import java.net.Socket; import java.security.*; import java.util.Scanner; import java.net.*; public class Client { public static void main(String[] args) { Socket mysocket; DataInputStream in=null; DataOutputStream out=null; Scanner scanner = new Scanner(System.in); try { System.out.print("輸入服務器的IP:"); String IP = scanner.nextLine(); InetAddress address=InetAddress.getByName(IP); mysocket = new Socket(address, 2010); in = new DataInputStream(mysocket.getInputStream()); out = new DataOutputStream(mysocket.getOutputStream()); Key_DH.DH("Lpub.dat","Lpri.dat"); FileInputStream my = new FileInputStream("Lpub.dat"); ObjectInputStream mypub = new ObjectInputStream(my); Key kp = (Key) mypub.readObject(); ByteArrayOutputStream DH = new ByteArrayOutputStream(); ObjectOutputStream myDH = new ObjectOutputStream(DH); myDH.writeObject(kp); byte []pub = DH.toByteArray(); out.writeUTF(pub.length+""); for(int i=0;i<pub.length;i++) { out.writeUTF(pub[i]+ ""); } Thread.sleep(1000); int length = Integer.parseInt(in.readUTF()); byte cpub[] = new byte[length]; for(int i=0;i<length;i++) { String temp = in.readUTF(); cpub[i] = Byte.parseByte(temp); } ByteArrayInputStream ckey1 = new ByteArrayInputStream(cpub); ObjectInputStream ckey = new ObjectInputStream(ckey1); Key k = (Key) ckey.readObject(); FileOutputStream f2 = new FileOutputStream("W1pub.dat"); ObjectOutputStream b2 = new ObjectOutputStream(f2); b2.writeObject(k); KeyAgree.Agree("W1pub.dat","Lpri.dat"); FileInputStream f = new FileInputStream("sb.dat"); byte[] keysb = new byte[24]; f.read(keysb); System.out.println("公共密鑰爲:"); for (int i = 0;i<24;i++) { System.out.print(keysb[i]+" "); } System.out.println("\n請輸入中綴表達式:"); while(scanner.hasNext()) { String question = scanner.next(); MyBC change = new MyBC(); change.ChangeString(question); String question1 = change.ChangeOrder(); System.out.println("後綴表達式爲:" + question1); String mtoMD5 = DigestPass.MD5(question1); System.out.println("明文的MD5值爲:"+mtoMD5); out.writeUTF(mtoMD5); SecretKeySpec k1 = new SecretKeySpec(keysb, "DESede"); Cipher cp = Cipher.getInstance("DESede"); cp.init(Cipher.ENCRYPT_MODE, k1); byte ptext[] = question1.getBytes("UTF8"); byte ctext[] = cp.doFinal(ptext); System.out.println("加密後的後綴表達式爲:"); for (int i = 0; i < ctext.length; i++) { System.out.print(ctext[i] + " "); } out.writeUTF(ctext.length + ""); for (int i = 0; i < ctext.length; i++) { out.writeUTF(ctext[i] + ""); } String s=in.readUTF(); //in讀取信息,堵塞狀態 System.out.println("\n客戶收到服務器的回答:"+s); Thread.sleep(500); System.out.println("請輸入中綴表達式:"); } } catch (Exception e) { System.out.println(e); } } }
import javax.crypto.Cipher; import javax.crypto.spec.SecretKeySpec; import java.io.*; import java.net.ServerSocket; import java.net.Socket; import java.security.*; public class Server { public static void main(String[] args) throws Exception { ServerSocket serverForClient = null; Socket socketOnServer = null; DataOutputStream out = null; DataInputStream in = null; try { serverForClient = new ServerSocket(2010); } catch (IOException e) { System.out.println(e); } System.out.println("等待客戶呼叫"); try { socketOnServer = serverForClient.accept(); out = new DataOutputStream(socketOnServer.getOutputStream()); in = new DataInputStream(socketOnServer.getInputStream()); Key_DH.DH("Wpub.dat","Wpri.dat"); int length = Integer.parseInt(in.readUTF()); byte cpub[] = new byte[length]; for(int i=0;i<length;i++) { String temp = in.readUTF(); cpub[i] = Byte.parseByte(temp); } ByteArrayInputStream ckey1 = new ByteArrayInputStream(cpub); ObjectInputStream ckey = new ObjectInputStream(ckey1); Key k1 = (Key) ckey.readObject(); FileOutputStream f2 = new FileOutputStream("L1pub.dat"); ObjectOutputStream b2 = new ObjectOutputStream(f2); b2.writeObject(k1); FileInputStream my = new FileInputStream("Wpub.dat"); ObjectInputStream mypub = new ObjectInputStream(my); Key kp = (Key) mypub.readObject(); ByteArrayOutputStream DH = new ByteArrayOutputStream(); ObjectOutputStream myDH = new ObjectOutputStream(DH); myDH.writeObject(kp); byte []pub = DH.toByteArray(); out.writeUTF(pub.length+""); for(int i=0;i<pub.length;i++) { out.writeUTF(pub[i]+ ""); } KeyAgree.Agree("L1pub.dat","Wpri.dat"); FileInputStream f = new FileInputStream("sb.dat"); byte[] keysb = new byte[24]; f.read(keysb); System.out.println("公共密鑰爲:"); for (int i = 0;i<24;i++) { System.out.print(keysb[i]+" "); } while(true) { String c = in.readUTF(); SecretKeySpec k = new SecretKeySpec(keysb, "DESede"); Cipher cp = Cipher.getInstance("DESede"); cp.init(Cipher.DECRYPT_MODE, k); String clength = in.readUTF(); byte ctext[] = new byte[Integer.parseInt(clength)]; for (int i = 0; i < Integer.parseInt(clength); i++) { String temp = in.readUTF(); ctext[i] = Byte.parseByte(temp); } byte[] ptext = cp.doFinal(ctext); String question = new String(ptext, "UTF8"); System.out.println("\n解密後的後綴表達式爲:" + question); String mtoMD5 = DigestPass.MD5(question); System.out.println("MD5的值爲"+ mtoMD5); if(mtoMD5.equals(c)) { System.out.println("傳遞的MD5值和解密的後綴表達式的MD5值相同,能夠解密!"); MyDC getanswer = new MyDC(); getanswer.set(question); Rational answer = getanswer.get(); int a = answer.getNumerator(); int b = answer.getDenominator(); float result = (float) a / b; System.out.println("計算出的結果爲"+String.format("%.2f",result)); out.writeUTF(String.format("%.2f",result)); } else { System.out.println("密文有誤,不能解密!"); } Thread.sleep(500); } } catch(Exception e) { System.out.println("客戶已斷開" + e); } } }
測試結果截圖
客戶端
服務器
本次實驗綜合了教材第十三章和Java密碼學的相關內容,在和夥伴的討論中,對於java的網絡和安全方面的編程有了更好地掌握。