一般网络数据通过http来get,post,那么其中的数据不可能杂乱无章,比如我要post一段数据,肯定是要有一定的格式,协议的。常用的就是xml和json了。在此先要搭建个简单的服务器吧,首先呢下载xampp,然后安装之类的就不再多讲了,参考http://cnbin.github.io/blog/2015/06/05/mac-an-zhuang-he-shi-yong-xampp/。安装好后,启动xampp,之后在浏览器输入localhost或者127.0.0.1就可以看到如下所示了:

这个就表示服务器已经运行了,具体的代码都是放在这个/Applications/XAMPP/htdocs目录下的。然后待会儿编写个xml文件也放在这里。
编写个简单的xml文件吧。

<Person>

  2  <Teacher>

  3     <name>xiao hong</name>

  4    <age>25</age>

  5    <sex>woman</sex>

  6    <class>english</class>

  7  </Teacher>

  8  <Student>

  9     <name>xiao ming</name>

 10    <age>15</age>

 11    <sex>man</sex>

 12  </Student>

 13 </Person>

 14 



这里为了方便,我在htdocs下面新建了一个test文件夹,然后再新建了一个person.xml文件,习惯了用vim,这里就用vim来实现了一把,保存退出后,我们去看下效果,打开chrome浏览器,输入http://localhost/test/person.xml。发现浏览器如下图所示:

显示的内容就是我们文件的内容,接下去通过app去获取这个信息。

xml解析主要有三种方式,SAX,Pull,Dom。下面就用这几种方法来实现下。

首先是SAX方式,SAX方式主要是两部分组成,一部分是解析器,也就是XMLReader接口,负责读取XML文档,另一部分是事件处理器ContentHandler,负责对发送事件响应和进行XML文档处理。

继承DefaultHandler,并重写5个父类的方法。

1、startDocument方法:开始XML解析的时候调用。

2、startElement方法:开始解析某个节点的时候调用。

3、characters方法:获取节点内容的时候调用。

4、endElement方法:解析完某个节点的时候调用。

5、endDocument方法:完成XML解析时候调用。

先新建类SAXContentHandler类,继承DefaultHandler类,编写代码如下:

package com.jared.emxmlstudy;

import android.util.Log;

import org.xml.sax.Attributes;
import org.xml.sax.SAXException;
import org.xml.sax.helpers.DefaultHandler; /**
* Created by jared on 16/2/19.
*/
public class SAXContentHandler extends DefaultHandler { private static final String TAB = "SAXContentHandler"; private String nodeName;
private StringBuffer mName;
private StringBuffer mAge;
private StringBuffer mSex;
private StringBuffer mClass; @Override
public void startDocument() throws SAXException {
mName = new StringBuffer();
mAge = new StringBuffer();
mSex = new StringBuffer();
mClass = new StringBuffer();
} @Override
public void startElement(String uri, String localName, String qName, Attributes attributes) throws SAXException {
nodeName = localName;
} @Override
public void characters(char[] ch, int start, int length) throws SAXException {
if("name".equals(nodeName)) {
mName.append(ch, start, length);
}else if("age".equals(nodeName)) {
mAge.append(ch, start, length);
} else if("sex".equals(nodeName)) {
mSex.append(ch, start, length);
} else if("class".equals(nodeName)) {
mClass.append(ch, start, length);
}
} @Override
public void endElement(String uri, String localName, String qName) throws SAXException {
if("Teacher".equals(localName)) {
Log.d(TAB, "This is Teacher");
Log.d(TAB, "name is:" + mName.toString().trim());
Log.d(TAB, "age is:" + mAge.toString().trim());
Log.d(TAB, "sex is:" + mSex.toString().trim());
Log.d(TAB, "class is:" + mClass.toString().trim());
mName.setLength(0);
mAge.setLength(0);
mSex.setLength(0);
mClass.setLength(0);
} else if("Student".equals(localName)) {
Log.d(TAB, "This is Student");
Log.d(TAB, "name is:" + mName.toString().trim());
Log.d(TAB, "age is:" + mAge.toString().trim());
Log.d(TAB, "sex is:" + mSex.toString().trim());
mName.setLength(0);
mAge.setLength(0);
mSex.setLength(0);
}
} @Override
public void endDocument() throws SAXException { }
}

这里要使用三种方法,所以修改布局如下:

<?xml version="1.0" encoding="utf-8"?>
<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"
android:orientation="vertical"
android:layout_margin="10dp"
tools:context="com.jared.emxmlstudy.MainActivity"> <Button
android:id="@+id/getXmlsax"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="SAX方式获取"/> <Button
android:id="@+id/getXmlpull"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="Pull方式获取"/> <Button
android:id="@+id/getXmldom"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="Dom方式获取"/> </LinearLayout>

然后MainActivity中添加代码如下;

package com.jared.emxmlstudy;

import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.view.View;
import android.widget.Button; import com.loopj.android.http.AsyncHttpClient;
import com.loopj.android.http.AsyncHttpResponseHandler; import org.xml.sax.InputSource;
import org.xml.sax.XMLReader; import java.io.StringReader; import javax.xml.parsers.SAXParserFactory; import cz.msebera.android.httpclient.Header; public class MainActivity extends AppCompatActivity { private static final String xmlUrl = "http://192.168.1.102/test/person.xml"; private Button mGetXmlSax; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main); mGetXmlSax = (Button)findViewById(R.id.getXmlsax);
mGetXmlSax.setOnClickListener(new myOnClickListener()); } private class myOnClickListener implements View.OnClickListener {
@Override
public void onClick(View view) {
switch (view.getId()) {
case R.id.getXmlsax:
sendRequestWithAsyncHttpClient(xmlUrl);
break;
default:
break;
}
}
} private void sendRequestWithAsyncHttpClient(String url) { AsyncHttpClient client = new AsyncHttpClient();
client.get(url, new AsyncHttpResponseHandler() {
@Override
public void onSuccess(int i, Header[] headers, byte[] bytes) {
try {
String response = new String(bytes, 0, bytes.length, "utf-8");
parseXMLWithSax(response);
} catch (Exception e) {
e.printStackTrace();
}
} @Override
public void onFailure(int i, Header[] headers, byte[] bytes, Throwable throwable) { }
});
} private void parseXMLWithSax(String xmlData) {
try {
SAXParserFactory factory = SAXParserFactory.newInstance();
XMLReader xmlReader = factory.newSAXParser().getXMLReader();
SAXContentHandler mHandler = new SAXContentHandler();
xmlReader.setContentHandler(mHandler);
xmlReader.parse(new InputSource(new StringReader(xmlData)));
} catch (Exception e) {
e.printStackTrace();
}
}
}

这里用到了AsyncHttpClient,具体库的添加参考上一篇文章Android开发学习之路--网络编程之初体验。好了,这里在发送请求的时候,成功再调用parseXMLWithSax进行解析。这里的地址是192.168.1.102,因为是真机调试,连接到了同一个网段,然后手机就可以访问我们的服务器了。

实例化一个factory,通过XMLReader来读取解析。运行点击按钮如下显示:

02-19 21:01:58.661 17390-17390/? D/SAXContentHandler: This is Teacher
02-19 21:01:58.661 17390-17390/? D/SAXContentHandler: name is:xiao hong
02-19 21:01:58.661 17390-17390/? D/SAXContentHandler: age is:25
02-19 21:01:58.661 17390-17390/? D/SAXContentHandler: sex is:woman
02-19 21:01:58.661 17390-17390/? D/SAXContentHandler: class is:english
02-19 21:01:58.661 17390-17390/? D/SAXContentHandler: This is Student
02-19 21:01:58.661 17390-17390/? D/SAXContentHandler: name is:xiao ming
02-19 21:01:58.661 17390-17390/? D/SAXContentHandler: age is:15
02-19 21:01:58.661 17390-17390/? D/SAXContentHandler: sex is:man

可以发现已经得到我们想要的信息了。

接着使用Pull方式,开始解析可以通过调用它的next方法,获取下一个事件,可以通过getAttribute方法获取属性,通过nextText方法来获取节点的值。编写代码如下:

package com.jared.emxmlstudy;

import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.util.Log;
import android.view.View;
import android.widget.Button; import com.loopj.android.http.AsyncHttpClient;
import com.loopj.android.http.AsyncHttpResponseHandler; import org.xml.sax.InputSource;
import org.xml.sax.XMLReader;
import org.xmlpull.v1.XmlPullParser;
import org.xmlpull.v1.XmlPullParserFactory; import java.io.StringReader; import javax.xml.parsers.SAXParserFactory; import cz.msebera.android.httpclient.Header; public class MainActivity extends AppCompatActivity { private static final String TAB = "XMLParse";
private static final String xmlUrl = "http://192.168.1.102/test/person.xml"; private Button mGetXmlSax;
private Button mGetXmlPull; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main); mGetXmlSax = (Button)findViewById(R.id.getXmlsax);
mGetXmlPull = (Button)findViewById(R.id.getXmlpull); mGetXmlSax.setOnClickListener(new myOnClickListener());
mGetXmlPull.setOnClickListener(new myOnClickListener()); } private class myOnClickListener implements View.OnClickListener {
@Override
public void onClick(View view) {
switch (view.getId()) {
case R.id.getXmlsax:
sendRequestWithSax(xmlUrl);
break;
case R.id.getXmlpull:
sendRequestWithPull(xmlUrl);
break;
default:
break;
}
}
} private void sendRequestWithPull(String url) {
AsyncHttpClient client = new AsyncHttpClient();
client.get(url, new AsyncHttpResponseHandler() {
@Override
public void onSuccess(int i, Header[] headers, byte[] bytes) {
try {
String response = new String(bytes, 0, bytes.length, "utf-8");
parseXMLWithPull(response);
} catch (Exception e) {
e.printStackTrace();
}
} @Override
public void onFailure(int i, Header[] headers, byte[] bytes, Throwable throwable) { }
});
} private void parseXMLWithPull(String xmlData) {
try {
XmlPullParserFactory factory = XmlPullParserFactory.newInstance();
XmlPullParser xmlPullParser = factory.newPullParser();
xmlPullParser.setInput(new StringReader(xmlData));
int eventType = xmlPullParser.getEventType();
String mName = "";
String mAge = "";
String mSex = "";
String mClass = ""; while (eventType != XmlPullParser.END_DOCUMENT) {
String nodeName = xmlPullParser.getName();
switch (eventType) {
case XmlPullParser.START_TAG: {
if("name".equals(nodeName)) {
mName = xmlPullParser.nextText();
} else if("age".equals(nodeName)) {
mAge = xmlPullParser.nextText();
} else if("sex".equals(nodeName)) {
mSex = xmlPullParser.nextText();
} else if("class".equals(nodeName)) {
mClass = xmlPullParser.nextText();
}
break;
}
case XmlPullParser.END_TAG: {
if("Teacher".equals(nodeName)) {
Log.d(TAB, "This is Teacher");
Log.d(TAB, "name is:" + mName.trim());
Log.d(TAB, "age is:" + mAge.trim());
Log.d(TAB, "sex is:" + mSex.trim());
Log.d(TAB, "class is:" + mClass.trim());
} else if("Student".equals(nodeName)) {
Log.d(TAB, "This is Student");
Log.d(TAB, "name is:" + mName.trim());
Log.d(TAB, "age is:" + mAge.trim());
Log.d(TAB, "sex is:" + mSex.trim());
}
}
default:
break;
}
eventType = xmlPullParser.next();
}
} catch (Exception e) {
e.printStackTrace();
}
} private void sendRequestWithSax(String url) { AsyncHttpClient client = new AsyncHttpClient();
client.get(url, new AsyncHttpResponseHandler() {
@Override
public void onSuccess(int i, Header[] headers, byte[] bytes) {
try {
String response = new String(bytes, 0, bytes.length, "utf-8");
parseXMLWithSax(response);
} catch (Exception e) {
e.printStackTrace();
}
} @Override
public void onFailure(int i, Header[] headers, byte[] bytes, Throwable throwable) { }
});
} private void parseXMLWithSax(String xmlData) {
try {
SAXParserFactory factory = SAXParserFactory.newInstance();
XMLReader xmlReader = factory.newSAXParser().getXMLReader();
SAXContentHandler mHandler = new SAXContentHandler();
xmlReader.setContentHandler(mHandler);
xmlReader.parse(new InputSource(new StringReader(xmlData)));
} catch (Exception e) {
e.printStackTrace();
}
}
}

运行结果如下:

02-19 21:52:01.391 19388-19388/? D/XMLParse: This is Teacher
02-19 21:52:01.391 19388-19388/? D/XMLParse: name is:xiao hong
02-19 21:52:01.391 19388-19388/? D/XMLParse: age is:25
02-19 21:52:01.391 19388-19388/? D/XMLParse: sex is:woman
02-19 21:52:01.391 19388-19388/? D/XMLParse: class is:english
02-19 21:52:01.391 19388-19388/? D/XMLParse: This is Student
02-19 21:52:01.391 19388-19388/? D/XMLParse: name is:xiao ming
02-19 21:52:01.391 19388-19388/? D/XMLParse: age is:15
02-19 21:52:01.391 19388-19388/? D/XMLParse: sex is:man

最后一种是Dom方式,Dom方式主要比较耗费内存,需要遍历所有,一般手机上的app开发不太适用。那就简单实现下吧,还是利用Async-HttpClient,接着编写代码如下:

    private void parseXMLWithDom(String xmlData) {
DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
String mName = "";
String mAge = "";
String mSex = "";
String mClass = "";
try {
DocumentBuilder builder = factory.newDocumentBuilder();
//Document document = builder.parse(xmlData);
Document document = builder.parse(new InputSource(new StringReader(xmlData)));
Element root = document.getDocumentElement();
Log.d(TAB, "根节点名称:" + root.getTagName());
NodeList items = root.getElementsByTagName("Teacher"); Element personElement = (Element)items.item(0);
Log.d(TAB, "根节点名称:" + personElement.getTagName());
NodeList childNodes = personElement.getChildNodes();
Log.d(TAB, "This is Teacher");
for(int i = 0; i < childNodes.getLength(); i++) {
Node grandElement = childNodes.item(i);
if(grandElement.getNodeType() == Node.ELEMENT_NODE) {
if("name".equals(grandElement.getNodeName())) {
mName = grandElement.getFirstChild().getNodeValue();
Log.d(TAB, "name is:" + mName.trim());
} else if("age".equals(grandElement.getNodeName())) {
mAge = grandElement.getFirstChild().getNodeValue();
Log.d(TAB, "age is:" + mAge.trim());
} else if("sex".equals(grandElement.getNodeName())) {
mSex = grandElement.getFirstChild().getNodeValue();
Log.d(TAB, "sex is:" + mSex.trim());
} else if("class".equals(grandElement.getNodeName())) {
mClass = grandElement.getFirstChild().getNodeValue();
Log.d(TAB, "class is:" + mClass.trim());
}
}
}
} catch (Exception e){
e.printStackTrace();
}
}

这里简单的就实现了Teacher的,student的就没有添加,相信也是很容易的了,其余的代码和上述类似,运行效果如下:

02-20 09:16:49.601 1353-1353/? D/XMLParse: 根节点名称:Person
02-20 09:16:49.601 1353-1353/? D/XMLParse: 根节点名称:Teacher
02-20 09:16:49.601 1353-1353/? D/XMLParse: This is Teacher:9
02-20 09:16:49.601 1353-1353/? D/XMLParse: name is:xiao hong
02-20 09:16:49.601 1353-1353/? D/XMLParse: age is:25
02-20 09:16:49.601 1353-1353/? D/XMLParse: sex is:woman
02-20 09:16:49.601 1353-1353/? D/XMLParse: class is:english

关于xml基本上先学习这些知识了。接着学习json的知识。

首先和xml一样,新建一个person.json文件,如下:

  1 [{"name":"xiao hong", "age":"25", "sex":"wonan"},

  2  {"name":"xiao ming", "age":"15", "sex":"man"},

  3  {"name":"xiao qiang", "age": 30, "sex":"man"}]

保存到和xml同一级目录下,运行浏览器如下图所示:

如图可知配置已经ok了,那么接下来就开始完成代码了,这里要使用JSONObject和GSON来实现,布局如下:

<?xml version="1.0" encoding="utf-8"?>
<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"
android:orientation="vertical"
android:layout_margin="10dp"
tools:context="com.jared.emjsonstudy.MainActivity"> <Button
android:id="@+id/getJSONObject"
android:text="Get Json With JSONObject"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:textAllCaps="false"/> <Button
android:id="@+id/getGSON"
android:text="Get Json With GSON"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:textAllCaps="false"/> </LinearLayout>

接着实现代码,还是用了Async-HttpClient来实现,代码如下:

package com.jared.emjsonstudy;

import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.Button; import com.loopj.android.http.AsyncHttpClient;
import com.loopj.android.http.AsyncHttpResponseHandler; import org.json.JSONArray;
import org.json.JSONObject; import cz.msebera.android.httpclient.Header; public class MainActivity extends AppCompatActivity { private static final String TAB = "JSONStudy";
private static final String JSON_URL = "http://192.168.1.102/test/person.json";
private Button mGetJSONObjectBtn;
private Button mGetGSONBtn; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main); mGetJSONObjectBtn = (Button)findViewById(R.id.getJSONObject);
mGetGSONBtn = (Button)findViewById(R.id.getGSON); mGetJSONObjectBtn.setOnClickListener(new myOnClickListener());
mGetGSONBtn.setOnClickListener(new myOnClickListener());
} private class myOnClickListener implements View.OnClickListener {
@Override
public void onClick(View view) {
switch (view.getId()) {
case R.id.getJSONObject:
getJSONWithJSONObject(JSON_URL);
break;
case R.id.getGSON:
break;
default:
break;
}
}
} private void parseJSONWithJSONObject(String jsonData) {
try {
JSONArray jsonArray = new JSONArray(jsonData);
for (int i = 0; i < jsonArray.length(); i++) {
JSONObject jsonObject = jsonArray.getJSONObject(i);
String mName = jsonObject.getString("name");
String mAge = jsonObject.getString("age");
String mSex = jsonObject.getString("sex"); Log.d(TAB, "name is: " + mName);
Log.d(TAB, "age is: " + mAge);
Log.d(TAB, "sex is:" + mSex);
}
} catch (Exception e) {
e.printStackTrace();
}
} private void getJSONWithJSONObject(String url) {
AsyncHttpClient client = new AsyncHttpClient();
client.get(url, new AsyncHttpResponseHandler() {
@Override
public void onSuccess(int i, Header[] headers, byte[] bytes) {
try {
String response = new String(bytes, 0, bytes.length, "utf-8");
parseJSONWithJSONObject(response);
} catch (Exception e) {
e.printStackTrace();
}
} @Override
public void onFailure(int i, Header[] headers, byte[] bytes, Throwable throwable) { }
});
}
}

如上代码明显比xml的简单多了,JSONArray获取到Json数据,然后通过JSONObject来获取对应键值的内容。因为我的电脑ip地址是192.168.1.102,手机和电脑在同一个网段,所以直接利用真机来测试,效果如下:

02-20 10:13:20.521 4947-4947/? D/JSONStudy: name is: xiao hong
02-20 10:13:20.521 4947-4947/? D/JSONStudy: age is: 25
02-20 10:13:20.521 4947-4947/? D/JSONStudy: sex is:wonan
02-20 10:13:20.521 4947-4947/? D/JSONStudy: name is: xiao ming
02-20 10:13:20.521 4947-4947/? D/JSONStudy: age is: 15
02-20 10:13:20.521 4947-4947/? D/JSONStudy: sex is:man
02-20 10:13:20.521 4947-4947/? D/JSONStudy: name is: xiao qiang
02-20 10:13:20.521 4947-4947/? D/JSONStudy: age is: 30
02-20 10:13:20.521 4947-4947/? D/JSONStudy: sex is:man

接着使用google的开源库GSON来实现,用到开源库,那就先下载了。gson下载地址:GSON下载地址点击这里。gson gitbub地址:https://github.com/google/gson

    GSON库主要是将一段JSON格式的字符串自动映射成一个对象,从而不需要编写代码去解析。这里新建一个Person类来获取数据,代码如下:

package com.jared.emjsonstudy;

/**
* Created by jared on 16/2/20.
*/
public class Person {
private String name;
private String age;
private String sex; public String getName() {
return name;
} public String getAge() {
return age;
} public String getSex() {
return sex;
} public void setName(String name) {
this.name = name;
} public void setAge(String age) {
this.age = age;
} public void setSex(String sex) {
this.sex = sex;
}
}

MainActivity中添加代码:

  void parseJSONWithGSON(String jsonData) {
Gson gson = new Gson();
List<Person> personList = gson.fromJson(jsonData,
new TypeToken<List<Person>>(){}.getType());
for (Person person :personList) {
Log.d(TAB, "Gson: name is: " + person.getName());
Log.d(TAB, "Gson: age is: " + person.getAge());
Log.d(TAB, "Gson: sex is:" + person.getSex());
}
} void getJSONWithGSON(String url) {
AsyncHttpClient client = new AsyncHttpClient();
client.get(url, new AsyncHttpResponseHandler() {
@Override
public void onSuccess(int i, Header[] headers, byte[] bytes) {
try {
String response = new String(bytes, 0, bytes.length, "utf-8");
parseJSONWithGSON(response);
} catch (Exception e) {
e.printStackTrace();
}
} @Override
public void onFailure(int i, Header[] headers, byte[] bytes, Throwable throwable) { }
});
}

这里new了一个Gson,然后通过fromJson的方法,通过TypeToken获取数据并保存到Person列表中。运行看下效果:

02-20 10:42:56.381 6434-6434/? D/JSONStudy: Gson: name is: xiao hong
02-20 10:42:56.381 6434-6434/? D/JSONStudy: Gson: age is: 25
02-20 10:42:56.381 6434-6434/? D/JSONStudy: Gson: sex is:wonan
02-20 10:42:56.381 6434-6434/? D/JSONStudy: Gson: name is: xiao ming
02-20 10:42:56.381 6434-6434/? D/JSONStudy: Gson: age is: 15
02-20 10:42:56.381 6434-6434/? D/JSONStudy: Gson: sex is:man
02-20 10:42:56.381 6434-6434/? D/JSONStudy: Gson: name is: xiao qiang
02-20 10:42:56.381 6434-6434/? D/JSONStudy: Gson: age is: 30
02-20 10:42:56.381 6434-6434/? D/JSONStudy: Gson: sex is:man

当然如果想要生存json数据,也是可以用gson的tojson方法的。

Android开发学习之路--网络编程之xml、json的更多相关文章

  1. Android开发学习之路--网络编程之初体验

    一般手机都是需要上网的,一般我们的浏览器就是个webview.这里简单实现下下功能,先编写Android的layout布局: <?xml version="1.0" enco ...

  2. Android开发学习之路--Android系统架构初探

    环境搭建好了,最简单的app也运行过了,那么app到底是怎么运行在手机上的,手机又到底怎么能运行这些应用,一堆的电子元器件最后可以运行这么美妙的界面,在此还是需要好好研究研究.这里从芯片及硬件模块-& ...

  3. GO语言的进阶之路-网络编程之socket

    GO语言的进阶之路-网络编程之socket 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.什么是socket; 在说socket之前,我们要对两个概念要有所了解,就是IP和端口 ...

  4. Android开发学习之路-RecyclerView滑动删除和拖动排序

    Android开发学习之路-RecyclerView使用初探 Android开发学习之路-RecyclerView的Item自定义动画及DefaultItemAnimator源码分析 Android开 ...

  5. Android开发学习之路--基于vitamio的视频播放器(二)

      终于把该忙的事情都忙得差不多了,接下来又可以开始good good study,day day up了.在Android开发学习之路–基于vitamio的视频播放器(一)中,主要讲了播放器的界面的 ...

  6. Android开发学习之路--Android Studio cmake编译ffmpeg

      最新的android studio2.2引入了cmake可以很好地实现ndk的编写.这里使用最新的方式,对于以前的android下的ndk编译什么的可以参考之前的文章:Android开发学习之路– ...

  7. Android开发学习之路--Activity之初体验

    环境也搭建好了,android系统也基本了解了,那么接下来就可以开始学习android开发了,相信这么学下去肯定可以把android开发学习好的,再加上时而再温故下linux下的知识,看看androi ...

  8. Android开发学习之路--MAC下Android Studio开发环境搭建

    自从毕业开始到现在还没有系统地学习android应用的开发,之前一直都是做些底层的驱动,以及linux上的c开发.虽然写过几个简单的app,也对android4.0.3的源代码做过部分的分析,也算入门 ...

  9. 我的大学Android开发学习之路——从开始到微信/支付宝/抖音Offer

    前言 笔者2016年高考考入华中科技大学计算机科学与技术专业. 2017年底(大二寒假)拿到今日头条(字节跳动)深圳研发中心Android开发实习生Offer,在深圳研发中心实习至2018年3月. 2 ...

随机推荐

  1. Python SMTP邮件发送

    SMTP是发送邮件的协议,Python内置对SMTP的支持,可以发送纯文本邮件.HTML邮件以及带附件的邮件. Python对SMTP支持有smtplib和email两个模块: email负责构造邮件 ...

  2. Java多线程volatile和synchronized总结

    volatile是轻量级的synchronized,在多处理器(多线程)开发中保证了共享变量的"可见性".可见性表示当一个线程修改了一个共享变量时,另外一个线程能读到这个修改的值. ...

  3. URLDecoder: Illegal hex characters in escape (%) pattern - For input string

    原因:后台发布文章的时候,内容里面有%,导致后台URLDecoder.decode()转码的时候报错. 看了java.net.URLDecoder的decode()的源码,原来是转码错误. 贴出部分代 ...

  4. 女儿开始bababababa的发声了

    女儿八个半月,开始bababababa的发声了,而不是像以前总啊啊啊的.

  5. 初始化mysql数据库——Activiti BPM

    package com.initialize; import org.activiti.engine.ProcessEngine; import org.activiti.engine.Process ...

  6. ctf writeup之程序员密码

    起因 在v2ex上看到有人发了一篇帖子,说做了一个程序员小游戏,遂试玩了一下. 游戏的地址在这里: http://www.bettertomissthantomeet.com/pages/level. ...

  7. [转载]致创业者:APP已死 服务永生

    前几日,有位创业者和我讲他在带领团队做一个将爱踢球的人集中在一起的App,我告诉他你的创业方向错了.原因在于你的目的是要为爱踢球的人提供服务,而你现在却在竭尽全力的做App,你应该做的是设计你为爱踢球 ...

  8. Go 语言结构体

    Go 语言中数组可以存储同一类型的数据,但在结构体中我们可以为不同项定义不同的数据类型. 结构体是由一系列具有相同类型或不同类型的数据构成的数据集合. 结构体表示一项记录,比如保存图书馆的书籍记录,每 ...

  9. C# 定积分求周长&面积原理 代码实现

    前言: 前些日子,因为工作原因,接触到了求解曲线周长,真的是搞了很久,学生时代真的很简单,但是如今的我来说,忘记了....很多人跟我应该一样. 所以来巩固加强一下记忆.一开始的时候,求周长嘛,找公式呗 ...

  10. Spark Streaming应用启动过程分析

    本文为SparkStreaming源码剖析的第三篇,主要分析SparkStreaming启动过程. 在调用StreamingContext.start方法后,进入JobScheduler.start方 ...