概述:android修改以太网的网络参数配置和wifi是不同的,这里讲述以太网的修改方法,如果想修改wifi的网络参数配置,网上文章很多。这里抛砖引玉,希望大神提供更好的解决方案。
分析:1.如果你是系统应用开发,应该不存在什么难点,系统应用可以获取应用最高权限。2.如果是纯第三方开发那就要慢慢踩一些坑,6.0版本对于权限做了很大的改动,所以你不熟悉情况下只有摸着石头过河。EthernetManager且被谷歌官方给隐藏了,第一想到用反射去调接口,思路没问题,但是每个版本不一定能通吃啊。shell命令可以设置,但是6.0的系统不好拿到root权限,即使拿到了root权限,修改的网络参数重启设备后又恢复DHCP,也就是android默认的设备启动自动分配权限。
思路:
1.源码修改,把隐藏的api注释给去掉重新打成jar包,这个过程需要底层哥们弄一下,或者自己花时间去修改编译。
2.IDE引用这个jar包,覆盖系统的sdk。这里一定会报错,看你怎么处理,如果你是eclpise这个就简单,引用jar包的时候放在最上面,第一个加载,错误就能解决。如果是AS,这个步骤略显复杂如下:
a1:添加依赖的时候放在第一位置。
a2:编译时,始终加载class.jar文件
gradle.projectsEvaluated {
tasks.withType(JavaCompile) {
options.compilerArgs.add('-Xbootclasspath/p:app\libs\classes.jar')
}
}
b : 在module的build.gradle中添加一下代码,这里为什么加这个,后面结合代码说明!!!
preBuild {
doLast { def imlFile = file(project.name + ".iml")
println 'Change ' + project.name + '.iml order'
try {
def parsedXml = (new XmlParser()).parse(imlFile)
def jdkNode = parsedXml.component[1].orderEntry.find { it.'@type' == 'jdk' }
parsedXml.component[1].remove(jdkNode)
def sdkString = "Android API " + android.compileSdkVersion.substring("android-".length()) + " Platform"
new Node(parsedXml.component[1], 'orderEntry', ['type': 'jdk', 'jdkName': sdkString, 'jdkType': 'Android SDK'])
groovy.xml.XmlUtil.serialize(parsedXml, new FileOutputStream(imlFile))
} catch (FileNotFoundException e) { // nop, iml not found
}
}
}
c : 代码示例:
//DHCP切换为静态,并配置网络信息
private void setStaticIp(){
try {
//每个IpConfiguration对象内部都包含了一个StaticIpConfiguration对象,对于DHCP方式来说这个对象赋为null
StaticIpConfiguration staticIpConfiguration = new StaticIpConfiguration();//用于保存静态IP、dns、gateway、netMask相关参数配置
InetAddress mIpAddr = NetworkUtils.numericToInetAddress(et_ip.getText().toString());//把192.168.1.1这种格式字符串转化为IP地址对象
String[] strs = et_mask.getText().toString().split("\\.");
int count = 0;
for(String str : strs){
if(str.equals("255")){
count++;
}
}
int prefixLength = count*8;
LinkAddress mIpAddress = new LinkAddress(mIpAddr,prefixLength);//prefixLength就是表示子网掩码字符有几个255,比如255.255.255.0的prefixLength为3
InetAddress mGateway = NetworkUtils.numericToInetAddress(et_gw.getText().toString());//默认网关
ArrayList<InetAddress> mDnsServers = new ArrayList<InetAddress>();//DNS
mDnsServers.add(NetworkUtils.numericToInetAddress(et_dns1.getText().toString()));
mDnsServers.add(NetworkUtils.numericToInetAddress(et_dns2.getText().toString()));
staticIpConfiguration.ipAddress = mIpAddress;
staticIpConfiguration.gateway = mGateway;
staticIpConfiguration.dnsServers.addAll(mDnsServers);
//ProxySettings为代理服务配置,主要有STATIC(手动代理)、PAC(自动代理)两种,NONE为不设置代理,UNASSIGNED为未配置代理(framework会使用NONE替代它)
//ProxyInfo包含代理配置信息
IpConfiguration config = new IpConfiguration(IpConfiguration.IpAssignment.STATIC, IpConfiguration.ProxySettings.NONE, staticIpConfiguration, ProxyInfo.buildDirectProxy(null,0));
mEthManager.setConfiguration(config);//执行该方法后,系统会先通过EthernetConfigStore保存IP配置到data/misc/ethernet/ipconfig.txt,再更新以太网配置、通过EthernetNetworkFactory重启eth设备(最终通过NetworkManagementService来操作开启关闭设备、更新状态)
//NetworkManagementService服务中提供了各种直接操作eth设备的API,如开关、列举、读写配置eth设备,都是通过发送指令实现与netd通信
//Netd 就是Network Daemon 的缩写,表示Network守护进程,Netd负责跟一些涉及网络的配置,操作,管理,查询等相关的功能实现
}catch (Exception e) {
e.printStackTrace();
}
}
//静态切换为DHCP
public void setMoveIp(){
try {
//对比STATIC,只需要把StaticIpConfiguration赋值为null
IpConfiguration config = new IpConfiguration(IpConfiguration.IpAssignment.DHCP, IpConfiguration.ProxySettings.NONE, null, ProxyInfo.buildDirectProxy(null,0));
mEthManager.setConfiguration(config);
}catch (Exception e) {
e.printStackTrace();
}
}
d : 如果没有执行b步骤,你会发现很对对象加载不到,因为默认SDK加载顺序在jar包前面,而官方的SDK就是阉割版,找不到对象,b过程就是把jar的顺序放在官方的SDK前面。如图: