Android与服务器的通信方式主要有两种,一是Http通信,一是Socket通信。两者的最大差异在于,http连接使用的是“请求—响应方式”,即在请求时建立连接通道,当客户端向服务器发送请求后,服务器端才能向客户端返回数据。而Socket通信则是在双方建立起连接后就可以直接进行数据的传输,在连接时可实现信息的主动推送,而不需要每次由客户端想服务器发送请求。 那么,什么是socket?Socket又称套接字,在程序内部提供了与外界通信的端口,即端口通信。通过建立socket连接,可为通信双方的数据传输传提供通道。socket的主要特点有数据丢失率低,使用简单且易于移植。

1.1什么是Socket
  是一种抽象层,应用程序通过它来发送和接收数据,使用Socket可以将应用程序添加到网络中,与处于同一网络中的其他应用程序进行通信。简单来说,Socket提供了程序内部与外界通信的端口并为通信双方的提供了数据传输通道。

1.2Socket的分类
  根据不同的的底层协议,Socket的实现是多样化的。本指南中只介绍TCP/IP协议族的内容,在这个协议族当中主要的Socket类型为流套接字(streamsocket)和数据报套接字(datagramsocket)。流套接字将TCP作为其端对端协议,提供了一个可信赖的字节流服务。数据报套接字使用UDP协议,提供数据打包发送服务。 下面,我们来认识一下这两种Socket类型的基本实现模型。

1.3Socket简单例子:

  服务器端Socket服务代码:

public class MyServer {public static void main(String[] args) throws Exception{ServerSocket ss = new ServerSocket(555);Socket s = ss.accept();DataInputStream dis = new DataInputStream(s.getInputStream());DataOutputStream dos = new DataOutputStream(s.getOutputStream());String str = dis.readUTF();System.out.print("客户端已连接");dos.writeUTF("Server:"+str);dos.flush();dis.close();s.close();ss.close();}}

  Android客户端的代码:

public class MainActivity extends Activity {public static Button mybutton = null;//发送Socket请求public static TextView mytext = null;//显示服务器返回的值
    @Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);mybutton = (Button)findViewById(R.id.button);mytext = (TextView)findViewById(R.id.text);mybutton.setOnClickListener(new mybuttonlistener());}//按钮的点击事件class mybuttonlistener implements OnClickListener{PrintStream out = null;BufferedReader buf = null;Socket s = null;public void onClick(View v) {try {s = new Socket("10.20.90.3", 555);//创建一个IP地址为:10.20.90.3,端口号为:555的Socket对象
                DataOutputStream dos = new DataOutputStream(s.getOutputStream());//获得一个输出流DataInputStream dis = new DataInputStream(s.getInputStream());//获得一个输入流dos.writeUTF("河南理工大学ACM协会");//发送到服务器的请求值String str = dis.readUTF();//获取服务器返回的参数
                mytext.setText(str);dos.flush();dos.close();dis.close();s.close();} catch (UnknownHostException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();}}}}

  最简单的一个服务器<--->客户端Socket事例就实现了,客户端程序执行前,需要首先启动服务器端程序,当服务器端启动完成后,我们就可以通过Android客户端发送Socket请求了。这个事例中,我们向服务器端发送:“河南理工大学ACM协会”字段,服务器为我们返回:“Server:河南理工大学ACM协会”。代码很简单,当然关于Socket的知识绝对不仅仅包含这些,剩下的就看小伙伴们是不是感兴趣了。


  对于List的上滑加载更多,网络上有很多开源控件,本篇接下来我将带领大家一起学习一下,如何实现List的上滑加载更多。

  首先是我们的主Activity的布局文件,布局文件没有其它内容只有一个ListView控件:

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"android:layout_width="match_parent"android:layout_height="match_parent"tools:context="${relativePackage}.${activityClass}" ><ListView android:id="@+id/listview"android:layout_width="fill_parent"android:layout_height="wrap_content"android:visibility="gone"></ListView></RelativeLayout>

  我们的主Activity代码:

public class MainActivity extends Activity {private ListView listview ;//list容器private List<String> data = new ArrayList<String>();//暂存数据的容器private ArrayAdapter<String> adapter;private final int number = 30;//每页的数据量private final int maxpage = 5;//数据的页数private int ItemCount;//显示的记录数private int nextpage;private boolean flag = true; //表示加载数据是否完成private View tipView;//页脚view
@SuppressLint("InflateParams")@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.listview_main);listview = (ListView)findViewById(R.id.listview);listview.setOnScrollListener(new scrollListener());//滑动事件监听listview.setOnItemClickListener(new onItemClickListener());//点击事件监听tipView = getLayoutInflater().inflate(R.layout.listview_tip, null);//获得加载更多的页脚/** 第一次加载*/data.addAll(new DataScroll().getDate(0, 30));//获得第一页需要显示的数据adapter = new ArrayAdapter<String>(getApplicationContext(), R.layout.listview_item, R.id.textview, data);//将数据与对应的数据显示页面配对/** 添加页脚必须依据下面的格式*/listview.addFooterView(tipView);//添加页脚(在加载数据之前)listview.setAdapter(adapter);//必须在此句之前加载页脚listview.removeFooterView(tipView);//删除页脚显示
    }public final class scrollListener implements OnScrollListener{@Overridepublic void onScrollStateChanged(AbsListView view, int scrollState) {}@Overridepublic void onScroll(AbsListView view, int firstVisibleItem,int visibleItemCount, int totalItemCount) {int lastItemId = listview.getLastVisiblePosition();//获得屏幕Item的最后一条记录的IDif((lastItemId+1)==totalItemCount){//判断是否已经拖动到页未if(totalItemCount > 0){//判断是否加载的数据是最后一页数据int currentpage = totalItemCount%number==0 ? totalItemCount/number : totalItemCount/number+1;nextpage = currentpage + 1;if(nextpage<=maxpage && flag){flag = false;ItemCount = totalItemCount;listview.addFooterView(tipView);//显示加载更多页脚new Thread(new Runnable() {@Overridepublic void run() {//模拟网络加载数据try {Thread.sleep(3000);//模拟网络加载} catch (InterruptedException e) {e.printStackTrace();}DataScroll datascroll = new DataScroll();List<String> result = datascroll.getDate(nextpage, number);Hand.sendMessage(Hand.obtainMessage(100, result));}}).start();}}}if(0==firstVisibleItem){if(firstVisibleItem < 0){Toast.makeText(MainActivity.this, "刷新", Toast.LENGTH_SHORT).show();}}}    }//返回到主线程执行Handler Hand = new Handler(){@SuppressWarnings("unchecked")@SuppressLint("HandlerLeak")public void handleMessage(Message msg) {data.addAll((List<String>) msg.obj);adapter.notifyDataSetChanged();//告诉ListView数据已经发生改变,要求更新ListView界面if(listview.getFooterViewsCount()>0){listview.removeFooterView(tipView);}flag = true;}};//List中各Item的点击事件class onItemClickListener implements OnItemClickListener{@Overridepublic void onItemClick(AdapterView<?> parent, View view, int position,long id) {Toast.makeText(getApplicationContext(), position+"          "+id, Toast.LENGTH_SHORT).show();}}}

  接下来我们需要补充一下我们的Item的布局(listview_item.xml):

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"android:layout_width="match_parent"android:layout_height="wrap_content"android:background="#aa00ff"tools:context="${relativePackage}.${activityClass}" ><TextViewandroid:id="@+id/textview"android:layout_width="fill_parent"android:layout_height="60dp"android:textSize="18sp"android:textColor="#aa0000"android:singleLine="true"/></RelativeLayout>

  当然既然是加载更多,我们还需要一个提示用户正在加载更多的一个页脚(listview_tip.xml):

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"android:layout_width="match_parent"android:layout_height="match_parent"tools:context="${relativePackage}.${activityClass}" ><ProgressBarandroid:id="@+id/progressBar1"android:layout_width="wrap_content"android:layout_height="wrap_content"android:layout_alignParentLeft="true"android:layout_below="@+id/textview" /><TextViewandroid:id="@+id/textview"android:layout_width="wrap_content"android:layout_height="wrap_content"android:layout_gravity="center_vertical"android:textSize="18sp"android:textColor="#000000"android:singleLine="true"android:text="数据加载中..." /></LinearLayout>

  当然最重要的加载数据,这里我们因为为了简单起见,就没有通过网络加载更多数据,而是通过一个模拟加载函数类实现的一个模拟加载(DataScroll.java):

/** 模拟获得分页数据*/
public class DataScroll {public List<String> getDate(int nextpage, int maxResult){int offset = 0;if(nextpage!=0){offset = (nextpage-2)*maxResult;}List<String > list = new ArrayList<String>();for(int i=0; i<maxResult; i++){list.add("List数据分批加载"+ ++offset);}return list;}
}

  好了,我们的上滑加载更多的实现就为大家分享完毕,关于下滑刷新,小伙伴们可以通过监听用户滑动事件,来自行研究,很简单相信大家都能实现,有疑问欢迎留言讨论。


  Spinner:Android提供的下拉列表控件,接下来我将通过5个小例子,为大家介绍一下Spinner的系统自带样式与自定义样式,当然系统自带样式相对简单,我们就从简单开始入手,为大家一一介绍Spinner的使用。

  首先是布局文件:

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"android:layout_width="match_parent"android:layout_height="match_parent"tools:context="${relativePackage}.${activityClass}" ><LinearLayout android:id="@+id/linearone"android:layout_width="fill_parent"android:layout_height="fill_parent"android:orientation="vertical"><TextView android:id="@+id/textview"android:layout_width="wrap_content"android:layout_height="wrap_content"android:text="Spinner下拉列表"android:textSize="25dip"android:layout_gravity="center_horizontal"/><Spinnerandroid:id="@+id/spinner1"android:layout_width="wrap_content"android:layout_height="wrap_content" android:entries="@array/arr"android:spinnerMode="dropdown"/><Spinnerandroid:id="@+id/spinner2"android:layout_width="wrap_content"android:layout_height="wrap_content" android:entries="@array/brr"android:spinnerMode="dialog"/><Spinnerandroid:id="@+id/spinner3"android:layout_width="wrap_content"android:layout_height="wrap_content" /><Spinnerandroid:id="@+id/spinner4"android:layout_width="wrap_content"android:layout_height="wrap_content" android:spinnerMode="dialog"/><Spinnerandroid:id="@+id/spinner5"android:layout_width="wrap_content"android:layout_height="wrap_content" android:spinnerMode="dialog"/><Button android:id="@+id/buttonone"android:layout_width="fill_parent"android:layout_height="wrap_content"android:text="下一页"/></LinearLayout>
</RelativeLayout>

  前两个控件使用的是Android自带Spinner下拉列表样式,红色部分是我在res->values->strings文件中设置的值:

<?xml version="1.0" encoding="utf-8"?>
<resources><string name="app_name">下拉列表</string><string name="hello_world">Hello world!</string><string-array name="arr"><item>默认效果</item><item>河南</item><item>北京</item><item>上海</item></string-array><string-array name="brr"><item>弹出效果</item><item>河南</item><item>北京</item><item>上海</item></string-array></resources>

  主Activity代码:

public class Activityone extends Activity {private Spinner spin1;//默认Spinner,在按钮下方显示private Spinner spin2;//默认Spinner,通过Dialog的形式为用户展示private Spinner spin3;//自定义Spinner,在按钮下方显示private Spinner spin4;//自定义Spinner,通过Dialog的形式为用户展示private Spinner spin5;//自定义Spinner,通过Dialog的形式为用户展示private Button buttonone;private String [] array = new String [] {"数组引用","代表","组长","小妹"};private String [] arrayadapt = new String [] {"arrayadapt引用","代表","组长","小妹"};private List<String> list = new ArrayList<String>();@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_one);list.add("list引用");list.add("代表");list.add("组长");list.add("小妹");buttonone=(Button)findViewById(R.id.buttonone);spin1=(Spinner)findViewById(R.id.spinner1);spin2=(Spinner)findViewById(R.id.spinner2);spin3=(Spinner)findViewById(R.id.spinner3);spin4=(Spinner)findViewById(R.id.spinner4);spin5=(Spinner)findViewById(R.id.spinner5);ArrayAdapter<String> arrayadapter = new ArrayAdapter<String>(Activityone.this, android.R.layout.simple_spinner_item, arrayadapt);spin5.setAdapter(arrayadapter);BaseAdapter baseadapterone = new baseadapterone();spin3.setAdapter(baseadapterone);BaseAdapter baseadaptertwo = new baseadaptertwo();spin4.setAdapter(baseadaptertwo);//设置点击结果选择提示spin4.setOnItemSelectedListener(new OnItemSelectedListener() {@Overridepublic void onItemSelected(AdapterView<?> parent, View view,int position, long id) {Toast.makeText(Activityone.this, list.get(position), Toast.LENGTH_SHORT).show();}@Overridepublic void onNothingSelected(AdapterView<?> parent) {}});buttonone.setOnClickListener(new mybuttonistener());}private class baseadapterone extends BaseAdapter{@Overridepublic int getCount() {return array.length;}@Overridepublic Object getItem(int position) {return null;}@Overridepublic long getItemId(int position) {return 0;}@Overridepublic View getView(int position, View convertView, ViewGroup parent) {TextView textview = new TextView(Activityone.this);textview.setText(array[position]);return textview;}}private class baseadaptertwo extends BaseAdapter{@Overridepublic int getCount() {return list.size();}@Overridepublic Object getItem(int position) {return null;}@Overridepublic long getItemId(int position) {return 0;}@Overridepublic View getView(int position, View convertView, ViewGroup parent) {TextView textview = new TextView(Activityone.this);textview.setText(list.get(position));return textview;}}class mybuttonistener implements OnClickListener{@Overridepublic void onClick(View v) {Intent intent = new Intent(Activityone.this,Activitytwo.class);Activityone.this.startActivity(intent);}        }}

  ok关于今天的内容介绍到此介绍,内容很简单,大家感兴趣的话可以实现一下。新手学习,高手交流。

Android之Socket通信、List加载更多、Spinner下拉列表相关推荐

  1. android SwipeRefreshLayout 增加上拉加载更多

    2019独角兽企业重金招聘Python工程师标准>>> 大家可能有的没有swipeRefreshLayout这个类,简单说一下,这是v4包里面的,19.1版本的时候就有了,但是当时的 ...

  2. android listview自动加载更多,如何实现 Android ListView『上拉加载更多』?

    ListView上拉加载更多的UI需求 (1)向上滑动 ListView,当最后一个条目滚入屏幕时开始加载更多条目,在列表底部增加一个 footerView:一个 infinite progressB ...

  3. MVC中实现 加载更多...

    需要实现的功能: 数据太多想初次加载部分数据,在底部加上"加载更多"按钮 点击后加载第二页数据(从数据库只取指定页数据)后接在已有数据后面(类似于android中的下拉加载更多) ...

  4. android 刷新某条数据_Android 支持刷新、加载更多、带反弹效果的RecyclerView

    点击上方"Android技术杂货铺",选择"标星" 干货文章,第一时间送达! 开篇 当前市面上很多支持刷新.加载更多RecyclerView开源库,为何我这里还 ...

  5. Android自定义控件(四)仿网易客户端上拉加载更多

    上一篇仿得网页客户端的抽屉模式,这一篇继续,来写一写加载更多这个功能,通过自定义实现加载更多,先上图: 今天实现的就是如图中最下面的20条载入中...这个功能啦! 先来说一下思路: 1.在listvi ...

  6. Android项目:使用pulltorefresh开源项目扩展为下拉刷新上拉加载更多的处理方法,监听listview滚动方向...

    很多android应用的下拉刷新都是使用的pulltorefresh这个开源项目,但是它的扩展性在下拉刷新同时又上拉加载更多时会有一定的局限性.查了很多地方,发现这个开源项目并不能很好的同时支持下拉刷 ...

  7. Android RecyclerView封装下拉刷新与上拉加载更多

    1 scanlistlibrary 基础组件说明(基于 RecyclerView的封装) 基本数据列表(支持下拉刷新与上拉加载更多) 九宫格数据显示封装(支持下拉刷新与上拉加载更多) 瀑布流数据显示封 ...

  8. Android ListView下拉与上拉刷新加载更多数据 二

    效果图:                 java代码中: public class MainActivity extends Activity implements OnRefreshListene ...

  9. android 加载更多动画效果,Android实践之带加载效果的下拉刷新上拉加载更多

    前言 之前写的一个LoadingBar,这次把LoadingBar加到下拉刷新的头部.从头写一个下拉刷新,附赠上拉加载更多.下面话不多说了,来一起看看详细的介绍吧. 效果图: 实现过程 首先是自定义属 ...

最新文章

  1. 用JS验证asp.net服务端控件
  2. ios 调试的相关内容收集
  3. P7920-[Kubic]Permutation
  4. MATLAB无边框输出图像
  5. 小黄鸡 php,PHP调用小黄鸡 api post发送
  6. SVG 入门教程系列列表
  7. 妙笔生花!文字图像图形生成技术研究进展
  8. python中concat的用法_python pandas concat用法及代码示例
  9. [翻译]PHP中define()和const定义常量的区别
  10. Java网络编程入门(软件结构、网络通信协议、网络编程三要素)
  11. 盘点2018年云计算热点:云原生、全栈云,云大脑,谁能独占鳌头?
  12. 2023年出入境政策-喜忧参半
  13. 什么是GP、LP、VC、PE、FOF
  14. 会当凌绝顶-----盘赔之我见(四)
  15. java中判断字符串是否为数字的方法 StringUtil包函数
  16. java多线程编程实战指南_java多线程编程实战指南 怎么样
  17. vue pc商城---最终篇
  18. R语言统计篇:双因素方差分析
  19. IE被改:http://www.686dh.cn/?……”
  20. 服务器被挂马的一般解决方法

热门文章

  1. 通过rsync+inotify实现数据的实时备份
  2. 武汉东方7神话服饰有限公司
  3. NHibernate Step by Step:序篇 (转)
  4. 合并的路径Path.Combine
  5. CISCO路由器的恢复方法
  6. SaaS加速器 I 商业中心:提供商业助力 共享商业成功
  7. C# Note34: 异常机制相关小点
  8. 创建数据库时指定编码方式
  9. Windows10 + VS2015 环境下对gdal2.0.1进行64bit编译小结
  10. The number of Oracle redo threads (2) is not the same as the number of checkpoint threads (1)