一口气看完一个微博源码(一)
一口气看完一个项目源码(一)之用户注册
今天咋们看的这个项目源码是一个微博客户端,和服务端通讯用socket写的,项目名称:口袋微博,和前面那个项目不同,这个项目略难一点,不过没关系,让我们一起来学习学习吧。
按照使用流程,首先是注册页面,因此我们来写注册页面,先把注册页面的布局写了:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"
android:gravity="center_horizontal"
android:background="@drawable/back"
android:paddingTop="25px"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
> <!-- 声明一个线性布局 -->
<LinearLayout
android:orientation="horizontal"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
> <!-- 声明一个显示昵称的线性布局 -->
<TextView
android:text="@string/tvName"
android:layout_width="100px"
style="@style/text"
android:layout_height="wrap_content"
android:layout_gravity="center_vertical"
/>
<EditText
android:id="@+id/etName"
android:singleLine="true"
android:layout_width="160px"
android:layout_height="wrap_content"
/>
</LinearLayout>
<LinearLayout
android:orientation="horizontal"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
> <!-- 声明显示密码的线性布局 -->
<TextView
android:text="@string/tvPwd"
style="@style/text"
android:layout_width="100px"
android:layout_height="wrap_content"
android:layout_gravity="center_vertical"
/>
<EditText
android:id="@+id/etPwd1"
android:singleLine="true"
android:password="true"
android:layout_width="160px"
android:layout_height="wrap_content"
/>
</LinearLayout>
<LinearLayout
android:orientation="horizontal"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
> <!-- 声明显示确认密码的线性布局 -->
<TextView
android:text="@string/tvPwd2"
style="@style/text"
android:layout_width="100px"
android:layout_height="wrap_content"
android:layout_gravity="center_vertical"
/> <!-- 声明TextView控件 -->
<EditText
android:id="@+id/etPwd2"
android:singleLine="true"
android:password="true"
android:layout_width="160px"
android:layout_height="wrap_content"
/> <!-- 声明输入确认密码EditText控件 -->
</LinearLayout>
<LinearLayout
android:orientation="horizontal"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
> <!-- 声明包含Email输入的线性布局 -->
<TextView
android:text="@string/tvEmail"
style="@style/text"
android:layout_width="100px"
android:layout_height="wrap_content"
android:layout_gravity="center_vertical"
/>
<EditText
android:id="@+id/etEmail"
android:singleLine="true"
android:layout_width="160px"
android:layout_height="wrap_content"
/>
</LinearLayout>
<LinearLayout
android:orientation="horizontal"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
> <!-- 声明包含心情输入的线性布局 -->
<TextView
android:text="@string/tvStatus"
style="@style/text"
android:layout_width="100px"
android:layout_height="wrap_content"
android:layout_gravity="center_vertical"
/>
<EditText
android:id="@+id/etStatus"
android:singleLine="true"
android:text="@string/etStatus"
android:layout_width="160px"
android:layout_height="wrap_content"
/>
</LinearLayout>
<LinearLayout
android:orientation="horizontal"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
>
<Button
android:id="@+id/btnReg"
style="@style/button"
android:layout_width="120px"
android:layout_height="wrap_content"
android:text="@string/btnReg"
/>
<Button
android:id="@+id/btnBack"
style="@style/button"
android:layout_width="120px"
android:layout_height="wrap_content"
android:text="@string/btnBack"
/>
</LinearLayout>
<LinearLayout
android:orientation="vertical"
android:visibility="visible"
android:id="@+id/regResult"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
>
<LinearLayout
android:orientation="horizontal"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
>
<TextView
android:text="@string/regSuccess"
style="@style/text"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
/>
<EditText
android:id="@+id/etUno"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:cursorVisible="false"
/>
</LinearLayout>
<Button
android:id="@+id/btnEnter"
style="@style/text"
android:layout_gravity="right"
android:text="@string/btnEnter"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
/>
</LinearLayout>
</LinearLayout>
布局写好以后,我们需要给这些布局添加点击事件(主要是注册),通过自定义一个MyConnection指定地址和端口号连接socket得到DataOutputStream和一个DataInputStream,把要发送的注册信息封装好(这里封装用的是消息头+分隔符的形式)把消息发给服务端,然后通过DataInputStream读取服务端返回的结果,如果服务端返回结果的请求头为<#REG_SUCCESS#>代表注册成功,同时用一个变量记住返回的结果,否则就是注册失败。
注册代码如下:
/**
* 方法:连接服务器,进行注册
*/
public void register(){
new Thread(){
public void run(){
Looper.prepare();
//1.获得用户输入的数据并进行验证
EditText etName = (EditText)findViewById(R.id.etName); //获得昵称EditText对象
EditText etPwd1 = (EditText)findViewById(R.id.etPwd1); //获得密码EditText对象
EditText etPwd2 = (EditText)findViewById(R.id.etPwd2); //获得确认密码EditText对象
EditText etEmail = (EditText)findViewById(R.id.etEmail); //获得邮箱EditText对象
EditText etStatus = (EditText)findViewById(R.id.etStatus); //获得心情EditText对象
String name = etName.getEditableText().toString().trim(); //获得昵称
String pwd1 = etPwd1.getEditableText().toString().trim(); //获得密码
String pwd2 = etPwd2.getEditableText().toString().trim(); //获得确认密码
String email = etEmail.getEditableText().toString().trim(); //获得邮箱
String status = etStatus.getEditableText().toString().trim(); //获得状态
if(name.equals("") || pwd1.equals("") || pwd2.equals("") || email.equals("") || status.equals("")){
Toast.makeText(RegActivity.this, "请将注册信息填写完整", Toast.LENGTH_LONG).show();
return;
}
if(!pwd1.equals(pwd2)){ //判断两次输入的密码是否一致
Toast.makeText(RegActivity.this, "两次输入的密码不一致!", Toast.LENGTH_LONG).show();
return;
}
//2.连接服务器开始传数据
try{
mc = new MyConnector(SERVER_ADDRESS, SERVER_PORT);
String regInfo = "<#REGISTER#>"+name+"|"+pwd1+"|"+email+"|"+status;
mc.dout.writeUTF(regInfo);
String result = mc.din.readUTF();
pd.dismiss();
if(result.startsWith("<#REG_SUCCESS#>")){ //返回信息为注册成功
result= result.substring(15); //去掉信息头
uno = result; //记录用户的ID
myHandler.sendEmptyMessage(0); //发出Handler消息
Toast.makeText(RegActivity.this, "注册成功!", Toast.LENGTH_LONG).show();
}
else{ //注册失败
Toast.makeText(RegActivity.this, "注册失败!请重试!", Toast.LENGTH_LONG).show();
}
}
catch(Exception e){
e.printStackTrace();
}
}
}.start();
}接下来我们来写服务端的代码,因为服务端用的socket,因此需要一个入口函数,这个入口在Server.java文件中,通过它开启一个服务线程:
public class Server{
public static void main(String args[]){
try{
ServerSocket ss = new ServerSocket(8888);
ServerThread st = new ServerThread(ss);
st.start();
System.out.println("Listening...");
}catch(Exception e){
e.printStackTrace();
}
}
}我们转到这个线程类里面,发现实现是通过调用serverSocket的accept()方法,监听客户端请求。对于请求过来的数据,专门写一个代理类ServerAgent.java来处理。
package wpf;
import java.net.ServerSocket;
import java.net.Socket;
import java.net.SocketException;
public class ServerThread extends Thread{
public ServerSocket ss; //声明ServerSocket对象
public boolean flag = false;
public ServerThread(ServerSocket ss){ //构造器
this.ss = ss;
flag = true;
}
public void run(){
while(flag){
try{
Socket socket = ss.accept();
ServerAgent sa = new ServerAgent(socket);
sa.start();
}
catch(SocketException se){
try{
ss.close();
ss = null;
System.out.println("ServerSocket closed");
}catch(Exception ee){
ee.printStackTrace();
}
}
catch(Exception e){
e.printStackTrace();
}
}
}
}我们来看ServerAgent是怎么实现的。
代理线程通过构造函数拿到数据输入和输出流,然后在run方法里面处理用户请求。接下来就是对请求头的判断,我们这里先只讲注册请求,收到的是注册请求,解析用户发过来的数据,然后通过写一个数据库工具类DBUtil类处理用户发送过来的数据。并将处理结果返回给客户端,然后客户端通过判断从服务器获取的请求结果更新ui界面。两个主要的方法:
public ServerAgent(Socket socket){
this.socket = socket;
try {
this.din = new DataInputStream(socket.getInputStream());
this.dout = new DataOutputStream(socket.getOutputStream());
flag =true;
} catch (IOException e) {
e.printStackTrace();
}
}
public void run(){
while(flag){
try {
String msg = din.readUTF(); //接收客户端发来的消息
// System.out.println("收到的消息是:"+msg);
if(msg.startsWith("<#REGISTER#>")){ //消息为用户注册
msg = msg.substring(12); //获得字符串值
String [] sa = msg.split("\\|"); //切割字符串
String regResult = DBUtil.registerUser(sa[0], sa[1], sa[2], sa[3], "1");
if(regResult.equals(REGISTER_FAIL)){ //注册失败
dout.writeUTF("<#REG_FAIL#>");
}
else{
dout.writeUTF("<#REG_SUCCESS#>"+regResult);
}
}
}
于是,我们转到DbUtil类里面,看书怎么保存到数据库中的。我们发现存储使用一般存储方式,只是里面有设置存储编码方式,还用到了prepareStatement。处理结束返回处理结果,如果连接为null返回连接失败,如果保存到数据库成功就返回注册完的用户id,否则返回注册失败。完了之后不要忘了释放资源。
//方法:注册用户
public static String registerUser(String u_name,String u_pwd,String u_email,String u_state,String h_id){
String result=null;
Connection con = null; //声明数据库连接对象
PreparedStatement ps = null; //声明语句对象
try{
con = getConnection();
if(con == null){ //判断是否成功获取连接
result = CONNECTION_OUT;
return result; //返回方法的执行
}
ps = con.prepareStatement("insert into user(u_no,u_name,u_pwd,u_email,u_state,h_id)" +
"values(?,?,?,?,?,?)"); //构建SQL语句
String u_no = String.valueOf(getMax(USER)); //获得分配给用户的帐号
u_name = new String(u_name.getBytes(),"ISO-8859-1"); //转成ISO-8859-1以插入数据库
u_state = new String(u_state.getBytes(),"ISO-8859-1"); //转成ISO-8859-1以插入数据库
int no = Integer.valueOf(u_no);
int hid = Integer.valueOf(h_id);
ps.setInt(1, no); //设置PreparedStatement的参数
ps.setString(2, u_name);
ps.setString(3, u_pwd);
ps.setString(4, u_email);
ps.setString(5, u_state);
ps.setInt(6,hid);
int count = ps.executeUpdate(); //执行插入
if(count == 1){ //如果插入成功
result = u_no; //获得玩家的帐号
}
else{ //如果没有插入数据
result = REGISTER_FAIL; //获得出错信息
}
}catch(Exception e){
e.printStackTrace();
}
finally{
try{
if(ps != null){
ps.close();
ps = null;
}
}catch(Exception e){
e.printStackTrace();
}
try{
if(con != null){
con.close();
con = null;
}
}catch(Exception e){
e.printStackTrace();
}
}
return result;
}接下来便是客户端处理服务端返回数据:
客户端处理完服务端发过来的数据,就通过handler更新界面,让进入个人中心这个按钮可见,同时拿到用户id转到个人中心页面。这样,一个用户注册的模块写好了。
Handler myHandler = new Handler(){
@Override
public void handleMessage(Message msg) {
switch(msg.what){
case 0:
View linearLayout = (LinearLayout)findViewById(R.id.regResult); //获得线性布局
linearLayout.setVisibility(View.VISIBLE); //设置可见性
EditText etUno = (EditText)findViewById(R.id.etUno);
etUno.setText(uno);
break;
}
super.handleMessage(msg);
}
};下一篇介绍用户登录时如何实现的。