一.客户端通信工具类:

import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.net.SocketException; import org.apache.commons.net.ftp.FTP;
import org.apache.commons.net.ftp.FTPClient;
import org.apache.commons.net.ftp.FTPFile;
import org.apache.commons.net.ftp.FTPReply; /**
* 用于Android和FTP服务器进行交互的工具类
*/
public class FTPUtils {
private FTPClient ftpClient = null;
private static FTPUtils ftpUtilsInstance = null;
private String ftpUrl;
private int ftpPort;
private String userName;
private String userPassword;
private final int connectTimeout = 5000; private FTPUtils() {
ftpClient = new FTPClient();
} /*
* 得到类对象实例(因为只能有一个这样的类对象,所以用单例模式)
*/
public static FTPUtils getInstance() {
if (ftpUtilsInstance == null) {
ftpUtilsInstance = new FTPUtils();
}
return ftpUtilsInstance;
} /**
* 设置FTP服务器
*
* @param FTPUrl
* FTP服务器ip地址
* @param FTPPort
* FTP服务器端口号
* @param UserName
* 登陆FTP服务器的账号
* @param UserPassword
* 登陆FTP服务器的密码
* @return
*/
public boolean initFTPSetting(String FTPUrl, int FTPPort, String UserName,
String UserPassword) {
ftpUrl = FTPUrl;
ftpPort = FTPPort;
userName = UserName;
userPassword = UserPassword; int reply; try {
//设定连接超时时间
ftpClient.setConnectTimeout(connectTimeout); // 要连接的FTP服务器Url,Port
ftpClient.connect(FTPUrl, FTPPort); // 登陆FTP服务器
ftpClient.login(UserName, UserPassword); // 看返回的值是不是230,如果是,表示登陆成功
reply = ftpClient.getReplyCode(); if (!FTPReply.isPositiveCompletion(reply)) {
// 断开
ftpClient.disconnect();
return false;
} return true; } catch (SocketException e) {
e.printStackTrace();
return false;
} catch (IOException e) {
e.printStackTrace();
return false;
}
} /**
* 断开FTP服务器
*/
public boolean disconnectFTP(){
try {
ftpClient.disconnect();
return true;
} catch (IOException e) {
e.printStackTrace();
return false;
}
} /**
* 上传文件
*
* @param FilePath
* 要上传文件所在SDCard的路径
* @param FileName
* 要上传的文件的文件名(如:Sim唯一标识码)
* @return true为成功,false为失败
*/
public boolean uploadFile(String FilePath, String FileName) { if (!ftpClient.isConnected()) {
if (!initFTPSetting(ftpUrl, ftpPort, userName, userPassword)) {
return false;
}
} try { // 设置存储路径
// ftpClient.makeDirectory("/FtpFileTest");
// ftpClient.changeWorkingDirectory("/FtpFileTest"); // 设置上传文件需要的一些基本信息
ftpClient.setBufferSize(1024);
ftpClient.setControlEncoding("UTF-8");
ftpClient.enterLocalPassiveMode();
ftpClient.setFileType(FTP.BINARY_FILE_TYPE); // 文件上传吧~
FileInputStream fileInputStream = new FileInputStream(FilePath);
ftpClient.storeFile(FileName, fileInputStream); // 关闭文件流
fileInputStream.close(); // 退出登陆FTP,关闭ftpCLient的连接
ftpClient.logout();
ftpClient.disconnect(); } catch (IOException e) {
e.printStackTrace();
return false;
}
return true;
} /**
* 下载文件
*
* @param FilePath
* 要存放的文件的路径
* @param FileName
* 远程FTP服务器上的那个文件的名字
* @return true为成功,false为失败
*/
public boolean downLoadFile(String FilePath, String FileName) { if (!ftpClient.isConnected()) {
if (!initFTPSetting(ftpUrl, ftpPort, userName, userPassword)) {
return false;
}
} try {
// 转到指定下载目录
ftpClient.changeWorkingDirectory("/data"); // 列出该目录下所有文件
FTPFile[] files = ftpClient.listFiles(); // 遍历所有文件,找到指定的文件
for (FTPFile file : files) {
if (file.getName().equals(FileName)) {
// 根据绝对路径初始化文件
File localFile = new File(FilePath); // 输出流
OutputStream outputStream = new FileOutputStream(localFile); // 下载文件
ftpClient.retrieveFile(file.getName(), outputStream); // 清除缓存
outputStream.flush(); // 关闭流
outputStream.close();
}
} // 退出登陆FTP,关闭ftpCLient的连接
ftpClient.logout();
ftpClient.disconnect(); } catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} return true;
}
}

二.利用Android设备作为FTP服务器,以下是FTPServer类

import java.io.File;
import java.io.IOException;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map; import org.apache.ftpserver.FtpServer;
import org.apache.ftpserver.FtpServerFactory;
import org.apache.ftpserver.ftplet.Authority;
import org.apache.ftpserver.ftplet.DefaultFtplet;
import org.apache.ftpserver.ftplet.FtpException;
import org.apache.ftpserver.ftplet.FtpRequest;
import org.apache.ftpserver.ftplet.FtpSession;
import org.apache.ftpserver.ftplet.Ftplet;
import org.apache.ftpserver.ftplet.FtpletResult;
import org.apache.ftpserver.ftplet.UserManager;
import org.apache.ftpserver.listener.ListenerFactory;
import org.apache.ftpserver.usermanager.PropertiesUserManagerFactory;
import org.apache.ftpserver.usermanager.SaltedPasswordEncryptor;
import org.apache.ftpserver.usermanager.impl.BaseUser;
import org.apache.ftpserver.usermanager.impl.WritePermission; import android.os.Environment; public class FtpServerlet extends DefaultFtplet{ private FtpServer mFtpServer; private final int mPort = 2121; private final String mDirectory = Environment.getExternalStorageDirectory().getPath() + "/FtpFileTest"; private final String mUser = "way"; private final String mPassword = "way"; private static FtpServerlet mInstance; public static FtpServerlet getInstance(){
if(mInstance == null){
mInstance = new FtpServerlet();
}
return mInstance;
} /**
* FTP启动
* @throws FtpException
*/
public void start() throws FtpException { if (null != mFtpServer && false == mFtpServer.isStopped()) {
return;
} File file = new File(mDirectory);
if (!file.exists()) {
file.mkdirs();
} FtpServerFactory serverFactory = new FtpServerFactory();
ListenerFactory listenerFactory = new ListenerFactory(); // 设定端末番号
listenerFactory.setPort(mPort); // 通过PropertiesUserManagerFactory创建UserManager然后向配置文件添加用户
PropertiesUserManagerFactory userManagerFactory = new PropertiesUserManagerFactory();
userManagerFactory.setPasswordEncryptor(new SaltedPasswordEncryptor());
UserManager userManager = userManagerFactory.createUserManager(); List<Authority> auths = new ArrayList<Authority>();
Authority auth = new WritePermission();
auths.add(auth); //添加用户
BaseUser user = new BaseUser();
user.setName(mUser);
user.setPassword(mPassword);
user.setHomeDirectory(mDirectory);
user.setAuthorities(auths);
userManager.save(user); // 设定Ftplet
Map<String, Ftplet> ftpletMap = new HashMap<String, Ftplet>();
ftpletMap.put("Ftplet", this); serverFactory.setUserManager(userManager);
serverFactory.addListener("default", listenerFactory.createListener());
serverFactory.setFtplets(ftpletMap); // 创建并启动FTPServer
mFtpServer = serverFactory.createServer();
mFtpServer.start();
} /**
* FTP停止
*/
public void stop() { // FtpServer不存在和FtpServer正在运行中
if (null != mFtpServer && false == mFtpServer.isStopped()) {
mFtpServer.stop();
}
} @Override
public FtpletResult onAppendStart(FtpSession session, FtpRequest request)
throws FtpException, IOException {
System.out.println("onAppendStart");
return super.onAppendStart(session, request);
} @Override
public FtpletResult onAppendEnd(FtpSession session, FtpRequest request)
throws FtpException, IOException {
System.out.println("onAppendEnd");
return super.onAppendEnd(session, request);
} @Override
public FtpletResult onLogin(FtpSession session, FtpRequest request)
throws FtpException, IOException {
System.out.println("onLogin");
return super.onLogin(session, request);
} @Override
public FtpletResult onConnect(FtpSession session) throws FtpException,
IOException {
System.out.println("onConnect");
return super.onConnect(session);
} @Override
public FtpletResult onDisconnect(FtpSession session) throws FtpException,
IOException {
System.out.println("onDisconnect");
return super.onDisconnect(session);
} @Override
public FtpletResult onUploadStart(FtpSession session, FtpRequest request)
throws FtpException, IOException {
System.out.println("onUploadStart");
return super.onUploadStart(session, request);
} @Override
public FtpletResult onUploadEnd(FtpSession session, FtpRequest request)
throws FtpException, IOException {
String FtpUploadPath = mDirectory + "/" + request.getArgument();
//接收到文件后立即删除
File uploadFile = new File(FtpUploadPath);
uploadFile.delete();
return super.onUploadEnd(session, request);
}
}

三.客户端和服务器的AndroidManifest中添加相关权限
<!--允许应用打开网络套接口 -->
<uses-permission android:name="android.permission.INTERNET"/>

<!--允许应用访问关于Wi-Fi网络的信息 -->
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE"/>

<!-- 允许应用写用户的外部存储器 -->
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>

<!-- 允许应用进入Wi-Fi的组播方式 -->
<uses-permission android:name="android.permission.CHANGE_WIFI_MULTICAST_STATE" />

<!-- 允许应用改变Wi-Fi的连接状态 -->
<uses-permission android:name="android.permission.CHANGE_WIFI_STATE"/>

<!-- 允许应用改变网络的连接状态 -->
<uses-permission android:name="android.permission.CHANGE_NETWORK_STATE"/>

<!-- 允许应用访问网络上的信息 -->
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"/>

四.所需jar包

commons-net-3.0.1.jar

ftpserver-core-1.0.6.jar

找了半天不知道怎么上传附件,只能麻烦大家自己去下载了。

Android端与Android端利用WIFI进行FTP通信的更多相关文章

  1. 【转】Android端与Android端利用WIFI进行FTP通信

    原文网址:http://www.cnblogs.com/zhangkai5157/p/4303188.html 一.客户端通信工具类: import java.io.File; import java ...

  2. Android直播实现 Android端推流、播放

    最近想实现一个Android直播,但是对于这方面的资料都比较零碎,一开始是打算用ffmpeg来实现编码推流,在搜集资料期间,找到了几个强大的开源库,直接避免了jni的代码,集成后只用少量的java代码 ...

  3. 我的Android进阶之旅------>Android实现用Android手机控制PC端的关机和重启的功能(三)Android客户端功能实现

    我的Android进阶之旅------>Android实现用Android手机控制PC端的关机和重启的功能(一)PC服务器端(地址:http://blog.csdn.net/ouyang_pen ...

  4. pc端和android端应用程序测试有什么区别?(ps面试题)

    pc端和android端应用程序测试有什么区别?(ps面试题) [VIP7]大连-凭海临风(215687736) 2014/4/10 8:56:171.测试环境不同PC平台一般都是windows an ...

  5. Android客户端与服务端交互之登陆示例

    Android客户端与服务端交互之登陆示例 今天了解了一下android客户端与服务端是怎样交互的,发现其实跟web有点类似吧,然后网上找了大神的登陆示例,是基于IntentService的 1.后台 ...

  6. Android 小例子服务端

    这是之前发布的Android项目的服务端源码,只是简单的根据请求返回了一些测试数据,没有实现对数据库的操作,可以根据需求自己实现. 这是mvc4 WebAPI项目,需要用vs2012打开. 如果是用的 ...

  7. 使用Struts2服务端与android交互

    转自:http://www.cnblogs.com/android-html5/archive/2011/09/25/2534107.html android--使用Struts2服务端与androi ...

  8. Android app与PC端交互

    app提交信息到PC端mysql数据库 新建名为SignActivity package com.example.administrator.success; import android.app.A ...

  9. JS判断android ios系统 PC端和移动端

    最近公司上线移动端,需要根据不同的系统跳转到不同的产品页面,百度后发现这一段代码很好用,不但可以判断当前是什么系统,还能知道当前浏览器是什么内核,移动端PC端都已测试无问题! var browser ...

随机推荐

  1. 【刷题】BZOJ 3998 [TJOI2015]弦论

    Description 对于一个给定长度为N的字符串,求它的第K小子串是什么. Input 第一行是一个仅由小写英文字母构成的字符串S 第二行为两个整数T和K,T为0则表示不同位置的相同子串算作一个. ...

  2. BZOJ4557:[JLOI2016/SHOI2016]侦察守卫——题解

    https://www.lydsy.com/JudgeOnline/problem.php?id=4557 小R和B神正在玩一款游戏.这款游戏的地图由N个点和N-1条无向边组成,每条无向边连接两个点, ...

  3. BZOJ2820:YY的GCD——题解

    http://www.lydsy.com/JudgeOnline/problem.php?id=2820 Description 神犇YY虐完数论后给傻×kAc出了一题给定N, M,求1<=x& ...

  4. UVA.699 The Falling Leaves (二叉树 思维题)

    UVA.699 The Falling Leaves (二叉树 思维题) 题意分析 理解题意花了好半天,其实就是求建完树后再一条竖线上的所有节点的权值之和,如果按照普通的建树然后在计算的方法,是不方便 ...

  5. UVA.11464 Even Parity (思维题 开关问题)

    UVA.11464 Even Parity (思维题 开关问题) 题目大意 给出一个n*n的01方格,现在要求将其中的一些0转换为1,使得每个方格的上下左右格子的数字和为偶数(如果存在的话),求使得最 ...

  6. HDOJ.2037 今年暑假不AC (贪心)

    今年暑假不AC 点我挑战此题 题意分析 给出来n组节目的起止时间,让求出所最多能观看的完整节目个数. 贪心策略:按照节目的结束时间升序排序,比较下一项的开始时间是否比上一项的结束时间大,是的话计数器+ ...

  7. hdu5652:India and China Origins(并查集)

    倒序操作用并查集判断是否连通,新技能get√(其实以前就会了 这题细节很多...搞得整个程序都是调试输出,几度看不下去想要重写 并查集到现在大概掌握了两个基本用途:判断是否连通 / 路径压缩(上一篇b ...

  8. getElementsByClassName的原生实现

    DOM 提供了一个名为 getElementById() 的方法,这个方法将返回一个对象,这个对象就是参数 id 所对应的元素节点.另外,getElementByTagName() 方法会返回一个对象 ...

  9. #define与typedef

    #define(宏定义)只是简单的字符串代换(原地扩展),它本身并不在编译过程中进行,而是在这之前(预处理过程)就已经完成了. typedef是为了增加可读性而为标识符另起的新名称(仅仅只是个别名), ...

  10. weakself的另一种写法

    在不久前看AFNetworking的源码时候发现了这么一句: 1 2 3 4 5 6 7 8 9 10 // 不知道这行代码的使用场景的同学你该去自习看看ARC的注意事项和Block的使用了 // A ...