博客
关于我
【转载】android与PC,C#与Java 利用protobuf 进行无障碍通讯【Socket】
阅读量:252 次
发布时间:2019-03-01

本文共 14569 字,大约阅读时间需要 48 分钟。



【转载自:】                                                                                                              (还有Git命令等)

 protobuf 是什么?

Protocol buffers是一种编码方法构造的一种有效而可扩展的格式的数据。
 
谷歌使用其内部几乎RPC协议和文件格式的所有协议缓冲区。

 

参考文档

 

 API的  

 

protobuf 适用的语言

正宗(Google 自己内部用的)的protobuf支持三种语言:Java 、c++和Pyton,很遗憾的是并不支持.Net 或者 Lua 等语言,但社区的力量是不容忽视的,由于protobuf确实比Json、XML有速度上的优势和使用的方便,并且可以做到向前兼容、向后兼容等众多特点,所以protobuf社区又弄了个protobuf.net的组件并且还支持众多语言,详细可以看这个链接:,具体某种语言的使用请各自对号入座,本篇只是讲使用android 与c++服务器通讯(测试过)或者与PC 通讯,使用java与C#之间互相通讯方面的DEMO,方面读者做参考。

 

使用protobuf协议

定义protobuf协议 

定义protobuf协议必须创建一个以.proto为后缀的文件,以本篇为例,本篇创建了一个叫msg.proto的消息文件,内容如下:

package msginfo;
message CMsg
{
    required 
string
 msghead 
=
 
1
;
    required 
string
 msgbody 
=
 
2
;
}
message CMsgHead
{
    required int32 msglen 
=
 
1
;
    required int32 msgtype 
=
 
2
;
    required int32 msgseq 
=
 
3
;
    required int32 termversion 
=
 
4
;
    required int32 msgres 
=
 
5
;
    required 
string
 termid 
=
 
6
;
}
message CMsgReg
{
    optional int32 area 
=
 
1
;
    optional int32 region 
=
 
2
;
    optional int32 shop 
=
 
3
;
    optional int32 ret 
=
 
4
;
    optional 
string
 termid 
=
 
5
[defalut
=
"
12345
"
];
}
message CMsgLogin
{
    optional int32 ret 
=
 
1
;
}
message CMsgLogout
{
    optional int32 ret 
=
 
1
;

}

 

package在Java里面代表这个文件所在的包名,在c#里面代表该文件的命名空间,message代表一个类,

required
代表该字段必填,
optional
代表该字段可选,并可以为其设置默认值,默认值格式 :[
defalut
=字符串就是"123" ,整型就是 123]。

 

 

  如何编译该proto文件

java或android 使用的编译方法 

 正宗的proto可以在Linux下编译也有提供win版编译,由于Linux下编译要配置什么g++呀,之类的有点麻烦,之前做的步骤都忘得差不多,那还是回到win版编译吧,而net 版则是需要在win版下编译。

  正宗google 的protobuf 下载列表请参照:  ,选择其中的win版本下载。解压后会得到一个protoc.exe 文件,此时就可以开始编译了,先以java 为例,编译的步骤如下:

 

  • cmd 打开命令工具
  • 以我电脑为例,该exe 文件我放在F:\protoc 目录下,先cd 到该目录 cd F:\protoc
  •  
  • 再次进入目录后会发现该目录多了一个文件夹,即以该proto的package命名的的目录,会产生一个Msg.java的文件,这时这个文件就可以使用到我们的java或者 android 工程了。
  • 最后一步下载一个protobuf-java-2.3.0.jar的jar 包引用到你的java和android工程 里面,OK。可以使用你的protobuf了。如下图:
c#或者以后的Windows Phone 7 使用的编译方法:

.net 版的protobuf来源于proto社区,有两个版本。一个版本叫protobuf-net,官方站点:  写法上比较符合c#一贯的写法。另一个版本叫protobuf-csharp-sport ,

官方站点: 写法上跟java上的使用极其相似,比较遵循Google 的原生态写法,所以做跨平台还是选择第二版本吧。因为你会发现几乎和java的写法没啥两样,本篇也是使用这个版本。

 

进入该站点,下载你要的win版。 编译步骤如下:

  • 将刚才你的proto文件放在你解压出来的目录与protoc.exe 、ProtoGen.exe、ProtoGen.exe.config放于一起。其他文件可以删除或者 备份。
  • 还是打开命令行,定位于对应的目录里面,你放proto文件的目录里面。
  • 输入:protoc --descriptor_set_out=msg.protobin --include_imports msg.proto         
  • msg.protobin是要生成的prtobobin文件,可以使用这个bin文件生成cs文件
  • 再输入protogen msg.protobin  使用该bin文件生成cs文件,这样你就可以得到该 msg.cs 的CSharp版文件了,同时在VS里面使用要引入Google.ProtocolBuffers.dll。为了方便你可以将其做成一个批处理文件代码如下:
  • echo on
    protoc 
    --
    descriptor_set_out
    =
    msg.protobin 
    --
    include_imports msg.proto 
    protogen msg.protobin  
     

     将其另存为.bat文件即可

     

 

  使用protobuf编译后的文件来进行socket连接

android 与PC

android 做为客户端向PC的Java服务端发送数据,服务端得到数据进行解析,并打印出来 。

客户端代码:

package net.testSocket;
import java.io.IOException;
import java.io.InputStream;
import java.net.Socket;
import java.net.UnknownHostException;
import socket.exception.SmsClientException;
import socket.exception.SmsObjException;
import msginfo.Msg.CMsg;
import msginfo.Msg.CMsgHead;
import msginfo.Msg.CMsgReg;
import android.app.Activity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.TextView;
import com.google.protobuf.InvalidProtocolBufferException;
//
客户端的实现
public
 
class
 TestSocket extends Activity {
    
private
 TextView text1;
    
private
 Button but1; 
    Socket socket 
=
 
null
;
    
public
 
void
 onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        
//
 Thread desktopServerThread=new Thread(new AndroidServer());
        
//
 desktopServerThread.start();
        setContentView(R.layout.main);
        text1 
=
 (TextView) findViewById(R.id.text1);
        but1 
=
 (Button) findViewById(R.id.but1); 
        but1.setOnClickListener(
new
 Button.OnClickListener() {
            @Override
            
public
 
void
 onClick(View v) {
                
//
 edit1.setText("");
                
//
 Log.e("dddd", "sent id");
                
//
 new Thread() {
                
//
 public void run() {
                
try
 {
                    
//
 socket=new Socket("192.168.1.102",54321);
                    
//
socket = new Socket("192.168.1.110", 10527);
                     socket 
=
 
new
 Socket(
"
192.168.1.116
"
12345
);
                    
//
得到发送消息的对象 
                    
//
SmsObj smsobj = new SmsObj(socket);
                    
                    
//
设置消息头和消息体并存入消息里面
                    
//
 head
                    CMsgHead head 
=
 CMsgHead.newBuilder().setMsglen(
5
)
                            .setMsgtype(
1
).setMsgseq(
3
).setTermversion(
41
)
                            .setMsgres(
5
).setTermid(
"
11111111
"
).build();
                    
//
 body
                    CMsgReg body 
=
 CMsgReg.newBuilder().setArea(
22
)
                            .setRegion(
33
).setShop(
44
).build();
                    
//
 Msg
                    CMsg msg 
=
 CMsg.newBuilder()
                            .setMsghead(head.toByteString().toStringUtf8())
                            .setMsgbody(body.toByteString().toStringUtf8())
                            .build();
                    
//
 PrintWriter out = new PrintWriter(new BufferedWriter(
                    
//
 new OutputStreamWriter(socket.getOutputStream())),
                    
//
 true);
                    
//
 out.println(m.toString());
                    
//
 out.println(m.toByteString().toStringUtf8());
                    
//
 向服务器发送信息
                    msg.writeTo(socket.getOutputStream());
                    
//
byte[] b = msg.toByteArray();
                    
//
smsobj.sendMsg(b);
                    
//
 System.out.println("====msg==="
                    
//
 + m.toByteString().toStringUtf8());
                    
                    
//
 byte[] backBytes = smsobj.recvMsg();
                    
//
                    
//
 接受服务器的信息
                    InputStream input 
=
 socket.getInputStream();
                    
//
 DataInputStream dataInput=new DataInputStream();
                    
//
byte[] by = smsobj.recvMsg(input);
                    
byte
[] by
=
recvMsg(input);
                    setText(CMsg.parseFrom(by));
                    
//
 BufferedReader br = new BufferedReader(
                    
//
 new InputStreamReader(socket.getInputStream()));
                    
//
 String mstr = br.readLine();
                    
//
 if (!str .equals("")) {
                    
//
 text1.setText(str);
                    
//
 } else {
                    
//
 text1.setText("数据错误");
                    
//
 }
                    
//
 out.close();
                    
//
 br.close();
                    input.close();
                    
//
smsobj.close();
                    socket.close();
                } 
catch
 (UnknownHostException e) {
                    e.printStackTrace();
                } 
catch
 (IOException e) {
                    e.printStackTrace();
                } 
catch
 (Exception e) {
                    System.
out
.println(e.toString());
                }
                
//
 };
                
//
 }.start();
            }
        });
    }
    
    
/*
*
     * 接收server的信息
     * 
     * @return
     * @throws SmsClientException
     * @author fisher
     
*/
    
public
 
byte
[] recvMsg(InputStream inpustream) throws SmsObjException {
        
try
 {
 
            
byte
 len[] 
=
 
new
 
byte
[
1024
];
            
int
 count 
=
 inpustream.read(len);  
        
            
byte
[] temp 
=
 
new
 
byte
[count];
            
for
 (
int
 i 
=
 
0
; i 
<
 count; i
++
) {   
                    temp[i] 
=
 len[i];                              
            } 
            
return
 temp;
        } 
catch
 (Exception localException) {
            
throw
 
new
 SmsObjException(
"
SmapObj.recvMsg() occur exception!
"
                    
+
 localException.toString());
        }
    }
    
/*
*
     * 得到返回值添加到文本里面
     * 
     * @param g
     * @throws InvalidProtocolBufferException
     
*/
    
public
 
void
 setText(CMsg g) throws InvalidProtocolBufferException {
        CMsgHead h 
=
 CMsgHead.parseFrom(g.getMsghead().getBytes());
        StringBuffer sb 
=
 
new
 StringBuffer();
        
if
 (h.hasMsglen())
            sb.append(
"
==len===
"
 
+
 h.getMsglen() 
+
 
"
\n
"
);
        
if
 (h.hasMsgres())
            sb.append(
"
==res===
"
 
+
 h.getMsgres() 
+
 
"
\n
"
);
        
if
 (h.hasMsgseq())
            sb.append(
"
==seq===
"
 
+
 h.getMsgseq() 
+
 
"
\n
"
);
        
if
 (h.hasMsgtype())
            sb.append(
"
==type===
"
 
+
 h.getMsgtype() 
+
 
"
\n
"
);
        
if
 (h.hasTermid())
            sb.append(
"
==Termid===
"
 
+
 h.getTermid() 
+
 
"
\n
"
);
        
if
 (h.hasTermversion())
            sb.append(
"
==Termversion===
"
 
+
 h.getTermversion() 
+
 
"
\n
"
);
        CMsgReg bo 
=
 CMsgReg.parseFrom(g.getMsgbody().getBytes());
        
if
 (bo.hasArea())
            sb.append(
"
==area==
"
 
+
 bo.getArea() 
+
 
"
\n
"
);
        
if
 (bo.hasRegion())
            sb.append(
"
==Region==
"
 
+
 bo.getRegion() 
+
 
"
\n
"
);
        
if
 (bo.hasShop())
            sb.append(
"
==shop==
"
 
+
 bo.getShop() 
+
 
"
\n
"
);
        
if
 (bo.hasRet())
            sb.append(
"
==Ret==
"
 
+
 bo.getRet() 
+
 
"
\n
"
);
        
if
 (bo.hasTermid())
            sb.append(
"
==Termid==
"
 
+
 bo.getTermid() 
+
 
"
\n
"
);
        text1.setText(sb.toString());
    }

}

 

服务端代码:

 package server;

import java.io.DataInputStream;
import java.io.DataOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.ServerSocket;
import java.net.Socket;
import msginfo.Msg.CMsg;
import msginfo.Msg.CMsgHead;
import msginfo.Msg.CMsgReg;
public
 
class
 AndroidServer implements Runnable {
    
public
 
void
 run() {
        
try
 {
            System.
out
.println(
"
beign:
"
);
            ServerSocket serverSocket 
=
 
new
 ServerSocket(
12345
);
            
while
 (
true
) {
                System.
out
.println(
"
等待接收用户连接:
"
);
                
//
 接受客户端请求
                Socket client 
=
 serverSocket.accept();
                DataOutputStream dataOutputStream;
                DataInputStream dataInputStream;
                
try
 {
                    
//
 接受客户端信息
                    
//
 BufferedReader in = new BufferedReader(
                    
//
 new InputStreamReader(client.getInputStream()));
                    
//
 String str = in.readLine();
                    
//
 System.out.println("read length:  " + str.length());
                    
//
 System.out.println("read:  " + str);
                    
//
 InputStream inputstream = client.getInputStream();
                    
//
 byte[] buffer = new byte[1024 * 4];
                    
//
 int temp = 0;
                    
//
 while ((temp = inputstream.read(buffer)) != -1) {
                    
//
 str = new String(buffer, 0, temp);
                    
//
 System.out.println("===str===" + str);
                    
//
 File file = new File("user\\log\\login.log");
                    
//
 appendLog(file, str);
                    InputStream inputstream 
=
 client.getInputStream();
                    dataOutputStream 
=
 
new
 DataOutputStream(
                            client.getOutputStream());
                    
//
dataInputStream = new DataInputStream(inputstream);
                    
//
 byte[] d = new BufferedReader(new InputStreamReader(
                    
//
 dataInputStream)).readLine().getBytes();
                    
//
 byte[] bufHeader = new byte[4];
                    
//
 dataInputStream.readFully(bufHeader);
                    
//
 int len = BytesUtil.Bytes4ToInt(bufHeader);
                    
//
 System.out.println(d.length);
                    
//
 System.out.println(dataInputStream.readLine().toString());
                    
byte
 len[] 
=
 
new
 
byte
[
1024
];
                    
int
 count 
=
 inputstream.read(len);  
                
                    
byte
[] temp 
=
 
new
 
byte
[count];
                    
                    
for
 (
int
 i 
=
 
0
; i 
<
 count; i
++
) {   
                        
                            temp[i] 
=
 len[i];                              
                    } 
                    
//
 协议正文
//
                     byte[] sendByte = new byte[30];
//
                    
//
                     dataInputStream.readFully(sendByte);
//
                     for (byte b : sendByte) {
//
                     System.out.println(""+b);
//
                     }
                    CMsg msg 
=
 CMsg.parseFrom(temp);
                    
//
                    
//
                    CMsgHead head 
=
 CMsgHead.parseFrom(msg.getMsghead()
                            .getBytes());
                    System.
out
.println(
"
==len===
"
 
+
 head.getMsglen());
                    System.
out
.println(
"
==res===
"
 
+
 head.getMsgres());
                    System.
out
.println(
"
==seq===
"
 
+
 head.getMsgseq());
                    System.
out
.println(
"
==type===
"
 
+
 head.getMsgtype());
                    System.
out
.println(
"
==Termid===
"
 
+
 head.getTermid());
                    System.
out
.println(
"
==Termversion===
"
                            
+
 head.getTermversion());
                    CMsgReg body 
=
 CMsgReg.parseFrom(msg.getMsgbody()
                            .getBytes());
                    System.
out
.println(
"
==area==
"
 
+
 body.getArea());
                    System.
out
.println(
"
==Region==
"
 
+
 body.getRegion());
                    System.
out
.println(
"
==shop==
"
 
+
 body.getShop());
                    
//
 PrintWriter out = new PrintWriter(new BufferedWriter(
                    
//
 new OutputStreamWriter(client.getOutputStream())),
                    
//
 true);
                    
//
 out.println("return    " +msg.toString());
                    
//
 in.close();
                    
//
 out.close();
                    sendProtoBufBack(dataOutputStream);
                    inputstream.close();
                    
//
dataInputStream.close();
                } 
catch
 (Exception ex) {
                    System.
out
.println(ex.getMessage());
                    ex.printStackTrace();
                } 
finally
 {
                    client.close();
                    System.
out
.println(
"
close
"
);
                }
            }
        } 
catch
 (IOException e) {
            System.
out
.println(e.getMessage());
        }
    }
    
public
 
static
 
void
 main(String[] args) {
        Thread desktopServerThread 
=
 
new
 Thread(
new
 AndroidServer());
        desktopServerThread.start();
    }
    
private
 
byte
[] getProtoBufBack() {
        
//
 head
        CMsgHead head 
=
 CMsgHead.newBuilder().setMsglen(
5
)
                .setMsgtype(
1
).setMsgseq(
3
).setTermversion(
41
)
                .setMsgres(
5
).setTermid(
"
11111111
"
).build();
        
//
 body
        CMsgReg body 
=
 CMsgReg.newBuilder().setArea(
22
)
                .setRegion(
33
).setShop(
44
).build();
        
//
 Msg
        CMsg msg 
=
 CMsg.newBuilder()
                .setMsghead(head.toByteString().toStringUtf8())
                .setMsgbody(body.toByteString().toStringUtf8())
                .build();
        
return
 msg.toByteArray();
    }
    
private
 
void
 sendProtoBufBack(DataOutputStream dataOutputStream) {
        
byte
[] backBytes 
=
 getProtoBufBack();
        
//
 协议头部
    
//
    Integer len2 = backBytes.length;
        
//
 前四个字节,标示协议正文长度
    
//
    byte[] cmdHead2 = BytesUtil.IntToBytes4(len2);
        
try
 {
            
//
dataOutputStream.write(cmdHead2, 0, cmdHead2.length);
            dataOutputStream.write(backBytes, 
0
, backBytes.length);
            dataOutputStream.flush();
        } 
catch
 (IOException e) {
            e.printStackTrace();
        }
    }
}

 最后得到的效果:

客户端:

 

 服务端:

 

 

protobuf .net版的实现代码如下:

using
 System;
using
 System.IO;
using
 System.Net;
using
 System.Net.Sockets;
using
 System.Threading;
using
 Google.ProtocolBuffers;
using
 msginfo;
using
 System.Text;
using
 System.Collections;
using
 System.Collections.Generic;
namespace
 protobuf_csharp_sport
{
    
class
 Program
    {
        
private
 
static
 ManualResetEvent allDone 
=
 
new
 ManualResetEvent(
false
);
        
static
 
void
 Main(
string
[] args)
        {
            beginProtocbuf();
        }
        
private
 
static
 
void
 beginProtocbuf()
        {
            
//
启动服务端
            TcpListener server 
=
 
new
 TcpListener(IPAddress.Parse(
"
127.0.0.1
"
), 
12345
);
            server.Start();
            server.BeginAcceptTcpClient(clientConnected, server); 
            Console.WriteLine(
"
SERVER : 等待数据 ---
"
);
            
//
启动客户端
            ThreadPool.QueueUserWorkItem(runClient);
            allDone.WaitOne();
            Console.WriteLine(
"
SERVER : 退出 ---
"
);
            
//
 server.Stop();
        }
        
//
服务端处理
        
private
 
static
 
void
 clientConnected(IAsyncResult result)
        {
            
try
            {
                TcpListener server 
=
 (TcpListener)result.AsyncState;
                
using
 (TcpClient client 
=
 server.EndAcceptTcpClient(result))
                {
                    
using
 (NetworkStream stream 
=
 client.GetStream())
                    {
                        
//
获取
                        Console.WriteLine(
"
SERVER : 客户端已连接,数据读取中 --- 
"
);
                        
byte
[] myRequestBuffer 
=
 
new
 
byte
[
1024
];
                        
int
 myRequestLength 
=
 
0
;
                        
do
                        {
                            myRequestLength 
=
 stream.Read(myRequestBuffer, 
0
, myRequestBuffer.Length);
                        }
                        
while
 (stream.DataAvailable);
                         
                        CMsg msg 
=
 CMsg.ParseFrom(myRequestBuffer.RemoveEmptyByte(myRequestLength));
                        CMsgHead head 
=
 CMsgHead.ParseFrom(Encoding.ASCII.GetBytes(msg.Msghead));
                        CMsgReg body 
=
 CMsgReg.ParseFrom(Encoding.ASCII.GetBytes(msg.Msgbody));
                        IDictionary
<
Google.ProtocolBuffers.Descriptors.FieldDescriptor, 
object
>
 d 
=
 head.AllFields;
                        
foreach
 (var item 
in
 d)
                        {
                            Console.WriteLine(item.Value.ToString());
                        }
                        d 
=
 body.AllFields;
                        Console.WriteLine(
"
===========================================
"
);
                        
foreach
 (var item 
in
 d)
                        {
                            Console.WriteLine(item.Value.ToString());
                        }
                      
                        Console.WriteLine(
"
SERVER : 响应成功 ---
"
);
                        Console.WriteLine(
"
SERVER: 关闭连接 ---
"
);
                        stream.Close();
                    }
                    client.Close();
                }
            }
            
finally
            {
                allDone.Set();
            }
        }
        
//
客户端请求
        
private
 
static
 
void
 runClient(
object
 state)
        {
            
try
            {
                CMsgHead head 
=
 CMsgHead.CreateBuilder()
                    .SetMsglen(
5
)
                    .SetMsgtype(
1
)
                    .SetMsgseq(
3
)
                    .SetTermversion(
4
)
                    .SetMsgres(
5
)
                    .SetTermid(
"
11111111
"
)
                    .Build();
                CMsgReg body 
=
 CMsgReg.CreateBuilder().
                    SetArea(
22
)
                   .SetRegion(
33
)
                   .SetShop(
44
)
                   .Build();
                CMsg msg 
=
 CMsg.CreateBuilder()
                    .SetMsghead(head.ToByteString().ToStringUtf8())
                    .SetMsgbody(body.ToByteString().ToStringUtf8())
                    .Build();
                Console.WriteLine(
"
CLIENT : 对象构造完毕 ...
"
);
                
using
 (TcpClient client 
=
 
new
 TcpClient())
                {
                    
//
 client.Connect(new IPEndPoint(IPAddress.Parse("192.168.1.116"), 12345));
                    client.Connect(
new
 IPEndPoint(IPAddress.Parse(
"
127.0.0.1
"
), 
12345
));
                    Console.WriteLine(
"
CLIENT : socket 连接成功 ...
"
);
                    
using
 (NetworkStream stream 
=
 client.GetStream())
                    {
                        
//
发送
                        Console.WriteLine(
"
CLIENT : 发送数据 ...
"
);
                      
                        msg.WriteTo(stream);
                        
//
关闭
                        stream.Close();
                    }
                    client.Close();
                    Console.WriteLine(
"
CLIENT : 关闭 ...
"
);
                }
            }
            
catch
 (Exception error)
            {
                Console.WriteLine(
"
CLIENT ERROR : {0}
"
, error.ToString());
            }
        }
    }
//
end class
    
public
 
static
 
class
 ExtensionClass {
        
public
 
static
 
byte
[] RemoveEmptyByte(
this
 
byte
[] by,
int
 length) 
        {
            
byte
[] returnByte 
=
 
new
 
byte
[length];
            
for
 (
int
 i 
=
 
0
; i 
<
 length; i
++
)
            {
                returnByte[i] 
=
 by[i];
            }
            
return
 returnByte;
        }
    }

} 

 

 运行的效果:

 

 这样就OK了,之后就可以把java 服务端的IP或端口改成C# IP和服务端的商品一样,或者反过来也是可以的。c++版本经过测试也是可以的。简直是一个爽字。

你可能感兴趣的文章
天载免息股票钢铁股掀涨停潮
查看>>
树莓派SSH 连接不上:socket error Event:32 Error:10053
查看>>
wxPython的使用
查看>>
红黑树(1):B-树
查看>>
站长常用Shell脚本整理分享(全)
查看>>
linux 内核提权总结(demo+exp分析) -- 任意读写(二)
查看>>
2020年电工(初级)考试题及电工(初级)考试题库
查看>>
2020年电气试验考试技巧及电气试验模拟试题
查看>>
2021年电工(中级)考试报名及电工(中级)模拟试题
查看>>
直接插入排序
查看>>
Java发送邮件必带超时时间配置
查看>>
redis安裝并与SpringBoot整合
查看>>
一文搞懂Python中的所有数组数据类型
查看>>
温故而知新,重温 Java 7 的那些“新”特性
查看>>
drawRoundRect 边线跟角线粗细不一样
查看>>
DOM Insertion, Inside 追加元素内容
查看>>
轻量级阻塞和重量级阻塞
查看>>
访问http://ip:port/nexus,出现404
查看>>
springboot配置freemarker模板
查看>>
Canvas绘制动画
查看>>