java socket通讯乱码问题的解决

java socket通讯乱码问题的解决使用socket通讯经常会遇到客户端、服务器端字符编码不一致的情况,如果传输的信息包含中文,这时我们可能就需要对传输的信息的按照指定的字符集进行解码或者编码首先我们了解jdkapi中的几个基本的概念String类StringpublicString(byte[] bytes,String charsetName)th

大家好,又见面了,我是你们的朋友全栈君。

使用socket通讯经常会遇到客户端、服务器端字符编码不一致的情况,如果传输的信息包含中文,这时我们可能就需要对传输的信息的按照指定的字符集进行解码

关于乱码的问题,最关键的要明白接受到的信息是什么编码的,就需要用什么字符集进行解码。

首先我们了解jdk api中的几个基本的概念

String类

String

public String(byte[] bytes,
              String charsetName)
       throws UnsupportedEncodingException

构造一个新的
String,方法是使用指定的字符集解码指定的字节数组。新的
String 的长度是一个字符集函数,因此不能等于字节数组的长度。

getBytes

public byte[] getBytes()

使用平台默认的字符集将此
String 解码为字节序列,并将结果存储到一个新的字节数组中。

getBytes

public byte[] getBytes(String charsetName)
                throws UnsupportedEncodingException

使用指定的字符集将此
String 解码为字节序列,并将结果存储到一个新的字节数组中。

在项目开发中遇到这种情况对方系统的编码为gb18030,而我们系统的编码为utf-8,两个系统直接使用socket进行通讯

在通讯过程中我们系统作为客户端需要按照gb18030进行报文发送,而当接受到对方系统的报文时我们需要将报文按照gb18030进行解码

具体测试代码如下:

socket服务端:

import java.io.BufferedInputStream;
import java.io.BufferedReader;
import java.io.ByteArrayInputStream;
import java.io.InputStreamReader;
import java.io.OutputStreamWriter;
import java.io.PrintWriter;
import java.net.ServerSocket;
import java.net.Socket;

public class SocketServer {
    
    /**
     * @throws Exception
     */
    public void startServer(int port) throws Exception {
        ServerSocket serverSocket = new ServerSocket(port);
        Socket server = null;
        try {
            while (true) {
                server = serverSocket.accept();
                System.out.println("socket服务端启动……");
                try {
                    BufferedReader input = new BufferedReader(new InputStreamReader(new ByteArrayInputStream("服务端发给客户端的信息".getBytes())));
                    BufferedInputStream in = new BufferedInputStream(server.getInputStream());
                    PrintWriter out = new PrintWriter(new OutputStreamWriter(server.getOutputStream()));
                    
                    String serverstring = null;
                    String clientstring = null;
                    out.println("欢迎客户端连入");
                    out.flush();
                    while (true) {
                        byte[] buf = new byte[10000];
                        int size=0;
                        if ((size=in.read(buf))!=-1) {
                            clientstring = new String(buf,0,size,"GB18030");
                            if (clientstring != null) {
                                System.out.println("client:" + clientstring);
                            }
                        }    
                        serverstring = input.readLine();
                        out.println(serverstring);
                        out.flush();
                        break;
                    }
                } catch (Exception e) {
                    e.printStackTrace();
                } finally {
                    server.close();
                }
            }
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            serverSocket.close();
        }
    }
    
    public static void main(String[] args) {
        SocketServer ss = new SocketServer();
        int port = 8888;
        try {
            ss.startServer(port);
        } catch (Exception e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
    }
}

socket客户端

 

import java.io.BufferedInputStream;
import java.io.OutputStreamWriter;
import java.io.PrintWriter;
import java.net.Socket;


public class SocketClient {
    
    /**
     * @throws Exception
     */
    public void makeConn(String ip, int port) throws Exception {
        Socket client = new Socket(ip, port);
        try {
//            BufferedReader input = new BufferedReader(new InputStreamReader(System.in));
//            BufferedReader input = new BufferedReader(new InputStreamReader(new ByteArrayInputStream("客户端发给服务器端的信息".getBytes())));
            BufferedInputStream in = new BufferedInputStream(client.getInputStream());
            PrintWriter out = new PrintWriter(new OutputStreamWriter(client.getOutputStream(),"GB18030"));
            
            String clientString = null;
            String serverString = null;
            while (true) {
                byte[] buf = new byte[10000];
                int size=0;
                if ((size=in.read(buf))!=-1) {
                    serverString = new String(buf,0,size,"GB18030");
                    if (serverString != null) {
                        System.out.println("server:" + serverString);
                    }
                }
//                clientString = input.readLine();
                clientString = "客户端发给服务器端的信息";
                if (clientString != null) {
                    out.println(clientString);
                    out.flush();
                    break;
                }
            }
            in.close();
            out.close();
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            client.close();
        }
    }
    
    public static void main(String[] args) {
        SocketClient sc = new SocketClient();
        int port = 8888;
        try {
            sc.makeConn("localhost", port);
        } catch (Exception e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
    }
}

上述代码的关键在于以下几部分:

由于socket服务端的编码是gb18030的,因此在接受到socket服务端的报文后我们需要按照gb18030进行解码

clientstring = new String(buf,0,size,”GB18030″);

而socket客户端因为编码是utf-8的,而socket服务端只能识别gb18030的编码,所以socket客户端在发送报文时需要做转换

 PrintWriter out = new PrintWriter(new OutputStreamWriter(client.getOutputStream(),”GB18030″));

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/156985.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • 移位运算(无符号移位运算,有符号移位运算)

    移位运算(无符号移位运算,有符号移位运算)移位运算,所有移位以5和-5为例移位运算左移(<<)正数负数带符号右移(>>)(右移向前面补对应的符号位所对应的值(正数补0,负数补1))正数负数不带符号右移(>>>)(>>>为java独有语法)正数负数移位运算可以移位运算的类型有:iuint,int,lang等类型.我们本次使用int类型一个int类型占4个字节,共32位,带符号…

    2022年7月13日
    18
  • 国外常用的免费DNS域名解析服务器「建议收藏」

    国外常用的免费DNS域名解析服务器「建议收藏」在国内注册的域名默认使用的是国内域名注册商提供的DNS服务器,国内的DNS服务器可能受政策的影响停止解析域名,网络上传说以后没有BeiAn的域名国内将不给解析。为了避免国内的这些政策,建议使用国外的域名服务:如果您还没有注册域名,请不要在国内注册域名。如果您已经在国内注册了域名,但由于国内注册商赖皮,不给您转移密码,您还可以使用国外的免费DNS服务器。国外免费DNS

    2022年6月22日
    188
  • 2011年1月31日nod32id,nod32免费激活码_NOD32用户名密码

    2011年1月31日nod32id,nod32免费激活码_NOD32用户名密码
    用户名:EAV-40746393密码:f77rk82e7n(适用类型ESS/EAV)

    用户名:EAV-40746397密码:uj7javkt8c(适用类型ESS/EAV)

    用户名:EAV-40899369密码:bsfrudfene(适用类型ESS/EAV)

    用户名:EAV-40898530密码:mpju75r5u7(适用类型ESS/EAV)

    用户名:EAV-40746391密码:aj7urdnr6

    2022年6月16日
    57
  • 微信浏览器

    微信浏览器

    2021年10月23日
    48
  • matlab在axis,matlab中axis的用法

    matlab在axis,matlab中axis的用法>>axis([02*pi-0.90.9])图5.1.3使用了图形修饰的plot函数绘制的正弦曲线5.1.3图形的比较显示在一般默认的情况下,MATLAB每次使用plot……>>axis([02*pi-0.90.9])图5.1.3使用了图形修饰的plot函数绘制的正弦曲线5.1.3图形的比较显示在一般默认的情况下…

    2022年6月3日
    41
  • windows下安装 redis并开机自启动

    windows下安装 redis并开机自启动

    2021年11月23日
    40

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号