文档章节

Selenium2 Python 自动化测试实战学习笔记

henni_719
 henni_719
发布于 2017/04/22 17:15
字数 4902
阅读 11
收藏 0

第一章  自动化测试基础

1.1 软件测试分类

软件测试 V 模型:

需求分析---设计---编码

验收测试--------系统测试---集成测试—单元测试

 

单元测试:是对程序中的单个子程序或具有独立功能的代码段进行测试的过程

集成测试:检查模块之间的接口是否正确

系统测试:针对整个产品进行系统测试,验证系统是否满足了需求规格的定义,以及软件系统的正确性和性能等是否满足其规约所指定的需求

验收测试:确保软件准备就绪,向软件购买者展示该软件系统需求满足其用户的需求

黑盒测试:把软件看做一个黑盒子,我们不去关心盒子里面的结构,只关心软件的输入数据和输出结果。主要针对软件界面和软件功能进行测试

白盒测试:研究源代码和程序的执行结果。它是按照程序内部的结构测试程序,通过测试来检验产品内部动作是否按照设计规格说明书的规定正常进行,检验程序中的每条通路是否都能按预定要求正确工作

灰盒测试:既关注输出对于输入的正确性,同时也关注不像白盒那样详细、完整,只是通过一些表征性的现象、事件、标志来判断内部运行的状态。

冒烟测试:是指在对一个新版本进行系统大规模的测试之前,先验证一下软件的基本功能是否实现,是否具备可测性

回归测试:修改了旧代码后,重新进行测试以确认修改后没有引入新的错误或导致其他代码产生错误

随机测试:是指测试中的所有输入数据都是随机生成的,其目的是模拟用户的真实操作,并发现一些边缘性的错误。

1.2 分层的自动化测试

                             UI测试

                                            集成/接口测试

单元测试

单元测试:单元测试框架,java的Junint、testNG,C#的Nuit,Python的unittest、pytest等

在自动化测试中最怕的是变化,因为变化的直接结果就是导致测试用例的运行失败,那么就需要对自动化脚本进行维护;如何控制失败,降低维护成本对自化的成败至关重要。

UI 层的自动化测试工具非常多,比较主流的是QTP,Robot Framework、watir、Selenium 等。

 

1.3 什么样的项目适合自动化测试

 

1.4 自动化测试及工具简述

自动化测试的概念有广义与狭义之分;广义上来讲所有借助工具来进行软件测试都可以称之为自动化测试;从狭义来讲,主要指基于UI层的自动化测试;除此之外还有基代码编写阶段的单元自动化测试,基本集成测试阶段的接口自动化测试。

1.5 Selenium 工具介绍

     Selenium IDE是嵌入到Firefox浏览器中的一个插件,实现简单的浏览器操作的录制和回放功能。主要应用是快速创建bug重现脚本,在测试人员的测试过程中,发现了Bug之后可以通过IDE将重现的步骤录制下来,以帮助开发人员更容易的重现BUG.

Selenium Grid:是一种自动化的测试辅助工具,Grid通过利用现有的计算机基础设施,能加快web-app的功能测试。利用Grid,可以很方便地同时在多台机器上和异构环境中并行运行多个测试实例。其特点是:

1.      并行执行

2.      通过一个主机统一控制用例在不同环境、不同浏览器下运行

3.      灵活添加变动测试机

  Selenium RC:支持多种不同的语言编写自动化测试脚本,通过Selenium RC的服务器作为代理服务器去访问应用从而达到测试的目的。Selenium RC 使用分 Client Libraries 和 Selenium Server,Client Libraries 库主要用于编写测试脚本,用来控制Selenium Server的库。Selenium Server 负责控制浏览器行为,总的来说,Selenium Server 主要包括3 个部分:Launcher、Http Proxy、Core。其中Selenium Core 是被Selenium Server 嵌入到浏览器页面中的。其实Selenium Core

就是一堆JS 函数的集合,就是通过这些JS 函数,我们才可以实现用程序对浏览器进行操作。Launcher 用于启动浏览器,把selnium Core 加载到浏览器页面当中,并把浏览器的代理设置为SeleniumServer 的Http Proxy。

1.6 前端工具

   FireBug:集HTML查看和编辑、Javascript 控制台、网络状况监视器、cookie 查看于一体,是开发JavaScript、CSS、HTML和Ajax 的得力助手。

  

 

 

 

 

 

 

 

 

 

第二章          测试环境搭建

2.1 window 下环境搭建

2.1.1 安装 Python

      访问Python 官方网站:https://www.Python.org/

2.1.2 安装steuptools 与pip

setuptools 与pip 下载地址:

https://pypi.Python.org/pypi/setuptools

https://pypi.Python.org/pypi/pip

  通过解压工具进行解压将得到两个文件夹,在Windows 命令提示符进入到文件解压目录,通过Python执行安装文件setup.py 进行安装。安装setuptools:

C:\package\setuptools-7.0>Pythonsetup.py install

安装pip 的方法与setuptools 相同,切换到pip 解压目录,运行setup.py 文件:

C:\package\pip-1.5.6>Pythonsetup.py install

2.1.4ActivePython

   访问ActivePython 下载地址:

http://www.activestate.com/activePython/downloads

2.2 编写第一个脚本

baidu.py

#codiong=uft-8
from selenium import webdriver
 
driver=webdriver.Chrom()
driver.get(“http://www.baidu.com”)
 
driver.find_element_by_id(“kw”).send_keys(“Selenium2”)
driver.find_element_by_id(“su”).Click()
 
driver.quit()

 

 

 

 

 

第三章           Python基础

3.1  输出和输入

Python 提供print方法来打印信息:print “hello”

调用print方法,用户双引号(“ ”)把需要打印的信息引起来就被输出了。

>>>name=”zhangsan”

>>>print “hello %s,Nice tomeet you!” %name

%s(string)只能打印字符串,如果想打印数字,那么就要使用%d(data)

>>>age=27

>>>print “You are %d !” %age

如果不知道自己要打印的是什么类型的数据,那么可以用%r来表示

>>> n=100

>>>print “Youprint is %r .” %n

>>> n=”abc”

>>>print “Youprint is %r .” %n

 

Python提供input输入:

</pre><pre name="code" class="python">#coding=utf-8
n=input(“Enter any content:”)
print “Your input is %r ” %n

但是input()方法比较矫情,要求用户输入的数据类型一定正确,比如字符串一定要加引号,如果不加会报错。

我们可以使用raw_input()方法,用户输入什么就是什么。

 

在Python当中,不区分单引号(‘ ’)与双引号(“ ”),单双引号都可以表示一个字符串

>>>print “hello”

>>>print ‘world’

可以嵌套使用,但不能交叉使用。

 

Python的单注释用井号(#)表示。

多行注释用三对引号表示,不分单、双引号。

 

 

 

 

 

3.2  分支与循环

   分支一般就用if..else..语句,循环一般就用for语句。Xx.py

Results=72
If  Results>=90:
       Print u’优秀’
Elif  Results>=70:
                Print u’良好’
Elif  Results>=60:
       Print u’及格’
Else:
       Print u’不及格’

For 语句:循环一个字符串的每一个字符。

Strings=“hello world”
For I in strings:
     Print I

循环数字,循环数字要借助range()函数。

For I in range(1,10):
       Print I

range(start,end,scan):start表示开始的位置,end表示结束的位置,scan表示每一次循环的步长。

3.3  数组与字典

   数组用中括号[]表示,里面的每一个元素用逗号(,)隔开。数组下标从0开始的。

>>>shuzu=[1,2,3,’a’,5]

>>>print shuzu

>>>print shuzu[0]

>>>printshuzu[4]

 

   字典以花括号{}表示,里面的元素是成对出现的,一个key对应一个value;一对元素用冒号(:)分割;不同元素用逗号(,)分开。

>>>zidian={“username”:”password”,’man’:’woman’,1:2}

>>>zidian.keys()

>>>zidian.values()

>>>zidian.items()

   字典里的每一对元素准确的来说是键值对,一个键(key)对应一个值(value)。Keys()函数可以输出所有键的值;values()函数可以输出所有值的值;items()函数输出一对键值对的值。

 

3.4  函数与类、方法

3.4.1函数

在python当中通过def关键字来定义函数,下面来定义一个函数:

>>>def add(a,b):

                     printa+b

>>>add(3,5)

创建一个add(),这个函数接收两个参数,a,b,通过print打印a+b的结果。通常情况下,函数不会直接答应结果,而是将处理结果通过return关键字返回。

>>>def add(a,b):

            return  a+b

>>>add(3,5)

有时候调用add()函数的时候不想传参,可以为函数设置默认参数。

>>>def add(a=6,b=7):

                               returna+b

>>>add()

>>>add(5,8)

3.4.2 类与方法

在Python中用class关键字来创建类。

Demo.py

class A():
            def add(self,a,b):
                      Returnself.a+self.b
count=A()
print count.add(3,5)

创建了一个A()类,在类下面创建了一个add()方法。方法创建同样适用关键字def,唯一不同的是方法必须有一个且必须是第一默认参数self,但是这个参数不用传值。

XX.py

class A():
     def add(self,a,b):
           return a+b
class B(A):
     def sub(self,a,b):
           return a-b
count=B()
print count.add(4,5)

用count变量来等于B类,因为B类继承了A类,所以B类也拥有了add()方法,所以count可以调用add()方法

 

3.5  模组

3.5.1 引用模块

在Python语言中通过from..import..的方式引用模块,下面引用time模块。

XX.PY

import time
print time.ctime()

如果只用到time模块下面的ctime()方法,可以这样使用:

Xx.py

from time import ctime
print ctime()

 

一次性把time模块下面的所有方法都引进来:

Xx.py

#coding=utf-8
from time import *
print ctime()
print u“休息一两秒”
sleep(2)

Python 所安装的模块存放在C:\Python27\Lib\site-packages\目录下面

3.5.2 模块调用

我们可以自己创建一个模块,然后通过另一个程序调用。

创建一个目录(project),在目录下面创建一个文件(pub.py),在文件中创建一个函数。

Project\pub.py

def add(a,b):
       print a+b

在相同的目录下面创建一个文件(count.py)

Project\count.py

import pub
print pub.add(3,5)

这样就实现了跨文件的函数调用。

 

**知识延伸:

在project目录下面多了一个pub.pyc文件,pyc文件是一种二进制文件,是由py文件经过编译后,生成的文件,是一种byte code,py文件变成pyc文件后,加载的速度有所提高,而且pyc是一种跨平台的字节码,是由python的虚拟机来执行。

 

 

3.5.3 跨目录模块调用

调用文件和被调用的文件不在同一目录下面,调用目录如下:

----project/model/pub.py

----project/count.py

Ps:删除刚才调用生成的pub.pyc文件。再来执行count.py,python会给出提示信息,找不到pub模块。那么python是如何找到模块的呢?

首先要知道,python在执行import语句时,到底进行了什么操作,按python的文档,它执行如下操作:

第一步,创建一个新的,空的module对象(它可能包含多个module)

第二步,把这个module对象插入sys.module中

第三步,装载module的代码(如果需要,首先必须编译)

第四步,执行新的module中对应的代码

在执行第三步时,首先要找到,module程序所在的位置,搜索顺序是:

当前路径(以及从当前目录指定的sys.path),然后是pythonPATH,然后是python的安装设置相关的默认路径。正因为这样的顺序,如果当前路径或pythoPATH中存在与标准module同样module,则会覆盖标准module。也就是说,如果当前目录下存在XML.py,那么执行import XML时,导入的是当前目录下的module,而不是系统标准的XML。

我们可以先构建一个package,以普通的方式导入,就可以直接访问此package中各个module了。在python中的package必须包含一个__init__.PY的文件.

 

现在我们就可以将..project/model/目录添加到系统环境变量下的path下面,但是这样不够灵活,我们可以调用python的sys模块来实现。

Project\count.py

import sys
sys.path.append(‘\model’)
 
from model import pub
print pub.add(3,5)

调用sys模块,把model目录通过append()方法追加到系统环境变量path下面。Append()是一个非常有用的方法,它通常用于向一个数据或集合尾部添加新数据。

PS:”\model”是一个相对路径,如果pub.py在别的目录下面,就需要用绝对路径了,如:append(“D:\\project\\model”)

现在运行程序依然会告诉我们找不到model,我们还需要在model目录下面创建一个__init__.PY的文件,内容可以为空。这个文件告诉python model是一个可以被调用的模块。现在就可以正常运行了。

那么我们可以在__init__.PY的文件目录下面放点什么呢?例如在model目录下面有两三个文件(a.py、b.py、c.py),那么打开__init__.PY的文件:

import a,b

 

Project\count.py

import sys
sys.path.append(‘model’)
from model import *
print pub.add(2,3)

现在我们用星号(*)表示导入model 下面所有文件,这个所有具体包含什么由__init__.py 决定,因为只import 了a 和b 了.

3.6  异常

Python用异常对象(exception object来表示异常情况。遇到错误后,会引发异常。如果异常对象未被处理或捕捉,程序会用所谓的回溯(Traceback,一种错误信息)终止执行。

Python提供try..except..语句来接收这个异常。

在python中所有的异常都继承Exception,所以我们可以使用它来接收所有的异常。

Abnormal.py                           

try:
     open(“abc.txt”,”r”)
     print aa
except Exception:
     print u“异常了”

从python2.5版本之后,所有异常类有了新的基类BaseException,Exception同样也继承BaseException,所以我们也可以使用BaseException来接收所有的异常。

Python直接输出异常信息:

Abnormal.py

           

try:
     open(“abc.txt”,”r”)
      print aa
except BaseException,msg:
       print msg

在BaseException后面定义msg变量用于接收异常信息,通过print将其打印出来:

Python中常见的异常:

BaseException 新的所有异常类的基类

Exception 所有异常类的基类,但继承BaseException 类

AssertionError assert 语句失败

AttributeError 试图访问一个对象没有属性

IOError 输入输出异常,试图打一个不存的文件(包括其它情况)时引起

NameError 使用一个还未赋值对象的变量

IndexError 在使用序列中不存在的所引进引发

IndentationError 语法错误,代码没有正确的对齐

KeyboardInterrupt Ctrl+C 被按下,程序被强行终止

TypeError 传入的对象类型与要求不符

SyntaxError Python 代码逻辑语法出错,不能执行

3.6.2更多异常用法

   try...except 与else 配合使用:

Abnormal.py

                           

try:
     open(“abc.txt”,”r”)
      prin taa
except BaseException,msg:
      print msg
else:
       print u‘没有异常!’

Else语句只有在没有异常的情况下才会被执行,但是有些情况下不管是否出现异常这些操作都能被执行,比如文件的关闭,所得释放,把数据库连接返还给连接池等操作。我们可以使用Try…finally..语句来完成。

首先创建一个poem.txt文件:

Pome.txt

ABC
EFG
HIJK
LMN
OPQ

通过一个小程序来读取文件中的内容。

ReadFile.py

#coding=utf-8
import time
 
files=file(“poem.txt”,”r”)
strs=files.readlines()
 
try:
  for I in strs:
       print I
       time.sleep(1)
finally:
       files.close()
       print “Clean up…closed the file”

3.6.3抛出异常

Python中提供raise方法来抛出一个异常:Abnormal.py

filename=raw_input(‘please input filename:’)
if filename==’hello’:
   raise NameError(‘input file name error!’)

第四章          WebDrive API

4.1 从定位元素开始

         在Python语言中对应的定位方法如下:

 find_element_by_id()

find_element_by_name()

find_element_by_class_name()

find_element_by_tag_name()

find_element_by_link_text()

find_element_by_partial_link_text()

find_element_by_xpath()

find_element_by_css_selector()

 

用by定位元素,find_element()方法只用于定位元素。它需要两个参数,第一个参数是定位方式,这个由By提供;第二个参数是定位的值。在使用By时需要将By类导入。

from selenium.webdriver.common.by import By

4.2 控制浏览器

 WebDriver提供了set_window_size()方法来设置浏览器的大小。

 Test.py

#coding=utf-8
from selenium import webdriver
driver=webdriver.Chorme()
driver.get(“http://m.mail.10086.cn”)
 
print u“设置浏览器的宽480,高800显示”
driver.set_window_size(480,800)
driver.quit()

Maximize_window()是浏览器最大化

 

WebDriver提供back()和forward()方法来模拟后退和前进。

Test.py

</pre><pre name="code" class="python">#coding=utf-8
from selenium import webdriver

driver=webdriver.Chrome()

firt_url=http://www.baidu.com
print “now access %r” %first_url
driver.get(first_url)

second_url=http://news.baidu.com
print “now access %s” %second_url
driver.get(second_url)


 
Print “back to %s” %(first_url)
driver.back()
 
print “forward to %s” % second_url
driver.forward()
 
driver.quit()

4.3 简单元素操作

 在WebDriver中,大多简单有趣的页面交互的方法都将通过WebElement接口提供,最常用的操作页面元素的方法如下:

Clear()  清除文本,如果是一个文本输入框
Send_keys(*value)在元素上模拟按键输入

Click()      单击元素     

Login163.py

#coding=utf-8
from seleniumimport webdriver
 
driver=webdriver.Chrome()
driver.get(“http://www.126.com”)
 
driver.find_element_by_id(“idInput”).clear()
driver.find_element_by_id(“idInput”).send_keys(“username”)
driver.find_element_by_id(“pwdInput”).clear()
driver.find_element_by_id(“pwdInput”).send_keys(“password”)
driver.find_element_by_id(“loginBtn”).click()
 
driver.close()

Click()方法和send_keys()方法是web页面操作中最常用到的两个方法。

Click()方法不仅仅用于点击按钮,还可以单击任何可以点击文字/图片链接、复选框、单选框、甚至是下拉框等。

Submit()方法用于提交表单,这里特别用于没提交按钮的情况,例如输入关键字之后的“回车”操作,那么就可以通过submit()来提交搜索框的内容。

Youdao.py

#coding=utf-8
from seleniumimport webdriver
 
driver=webdriver.Chrome()
dirver.get(“http://www.youdao.com”)
driver.find_element_by_id(“query”).send_keys(“hello”)
driver.find_element_by_id(“query”).submit()
 
driver.quit()

有时候submit()可以和click()方法互换使用,submit()同样可以提交一个按钮。

         Size           返回元素的尺寸

         Text                      获取元素的文本

         Get_attribute(name)     获取属性值

         Is_displayed()                       设置该元素是否用户可见。

Baidu.py

#coding=utf-8
from seleniumimport webdriver
 
driver=webdriver.Chrome()
driver.get(“http://www.baidu.com”)
 
size=driver.find_element_by_id(“kw”).size
print “kwelement size is %d” %size
 
text=driver.find_element_by_id(“cp”).text
print “cpelement text is %r” %text
 
attribute=driver.find_element_by_id(“kw”).get_attribute(“type”)
print “The typeattribute of kw element is %r” %attribut
 
result=driver.find_element_by_id(“kw”).is_displayed()
driver.close()

4.4 鼠标事件

在webdriver中这些关于鼠标操作的方法由ActionChains类提供。

ActionChains类提供的鼠标操作的常用方法:

l  Perform()                             执行所有ActionChains()中存储的行为

l  Context_click()                 右击

l  Double_click()                   双击

l  Drag_and_drop()              拖动

l  Move_to_element()        鼠标悬停

 

鼠标右击操作:xx.py

#conding=utf-8
from seleniumimport webdriver
fromselenium.webdriver.common.action_chains import ActionChains
driver=webdrive.Chrome()
driver.get(“http://yun.baidu.com/”)
 
driver.find_element_by_id(“TANGRAM__PSP_4__userName”).clear()
driver.find_element_by_id(“TANGRAM__PSP_4__userName”).send_keys(“username”)
driver.find_element_by_id(“TANGRAM__PSP_4__password”).clear()
driver.find_element_by_id(“TANGRAM__PSP_4__password”).send_keys(“password”)
driver.find_element_by_id(“TANGRAM__PSP_4__submit”).click()
 
disk_NET=driver.find_element_by_class_name(“pulldown-nav”)
ActionChains(driver).move_to_element(disk_NET)
DC=driver.find_element_by_class_name(“li disk”)
ActionChains(driver).double_click(DC)

Drag_and_drop(source,target):在原元素上按下鼠标左键,然后移动到目标元素上释放

4.5 键盘事件

Keys类提供键盘上几乎所有的按键的方法,send_keys()方法可以模拟键盘输入,除此之外还可以模拟键盘上的一些组合键,例如Ctrl+A,Ctrl+C等

KEYS.py

#coding=utf-8
from selenium import webdriver
from selenium.webdriver.commom.keys import Keys
 
driver=webdriver.Chrome()
driver.get(“http://www.baidu.com”)
 
driver.find_element_by_id(“kw”).send_keys(“seleniumm”)
#删除多输入的一个m
driver.find_element_by_id(“kw”).send_keys(Keys.BACK_SPACE)
 
#Ctrl+a全选输入框内容
driver.find_element_by_id(“kw”).send_keys(Keys.CONTROL,’a’)
#Ctrl+x全选输入框内容
driver.find_element_by_id(“kw”).send_keys(Keys.CONTROL,’x’)
#Ctrl+v全选输入框内容
driver.find_element_by_id(“kw”).send_keys(Keys.CONTROL,’v’)
driver.find_element_by_id(“su”).send_keys(Keys.ENTER)
 
driver.close()

导入键盘类包:from selenium.webdriver.commom.keys import Keys

 

经常用到键盘操作:

send_keys(Keys.BACK_SPACE)删除键(BackSpace)

send_keys(Keys.SPACE)空格键(Space)

send_keys(Keys.TAB)制表键(Tab)

send_keys(Keys.ESCAPE)回退键(Esc)

send_keys(Keys.ENTER)回车键(Enter)

send_keys(Keys.CONTROL,'a')全选(Ctrl+A)

send_keys(Keys.CONTROL,'c')复制(Ctrl+C)

send_keys(Keys.CONTROL,'x')剪切(Ctrl+X)

send_keys(Keys.CONTROL,'v')粘贴(Ctrl+V)

send_keys(Keys.F1)键盘F1

 

 

 

 

 

 

 

 

 

 

© 著作权归作者所有

henni_719
粉丝 2
博文 466
码字总数 343938
作品 0
信阳
QA/测试工程师
私信 提问
python资料全集

python: 微信公众号开发小记——2.80端口上的服务 python: 微信公众号开发小记——3.接入三方登录 使用python编写一个壁纸网站的简单爬虫 python: python List 用法 Python 中各个时间复杂度...

d_watson
2016/04/15
175
0
八月暑期福利,10本Python热门书籍免费送!

八月第一周,网易云社区联合博文视点为大家带来Python专场送书福利,10本关于Python的书籍内容涉及Python入门、绝技、开发、数据分析、深度学习、量化投资等。以下为书籍简介,送书福利请见文...

网易云
2018/08/02
0
0
筒子们,免费学习《Python自动化测试开发实践》课啦,请接住~

经常会有人问: 蛋哥,最近有没有什么免费的技术课啊? 蛋哥,最近有什么关于Python的课程么? 蛋哥,最近有没有测试的直播课额? 蛋哥,…… 应广大群众的强烈呼唤,为了让大家尽可能多的学...

蜗牛学院
2018/04/13
0
0
selenium webdriver (13) -- 结合pyunit生成测试报告

测试包含测试用例设计,测试执行,测试报告输出 测试用例设计一般是静态的,以文档的方式进行存储 测试执行,可以是手动的,也可以自动化用例执行 测试报告,可以是手动的,按照一定格式的测...

terry_hding
2016/08/04
189
0
Python在DevOps中的应用

互联网时代,只有能够快速试验新想法,并在第一时间,安全、可靠的交付业务价值,才能保持竞争力。DevOps推崇的自动化构建/测试/部署,以及系统度量等技术实践,是互联网时代必不可少的。 大...

抚琴煮酒
2018/08/10
0
0

没有更多内容

加载失败,请刷新页面

加载更多

docker百万连接设置相关资料

借助Docker单机秒开数十万TCP连接

youngjdong
17分钟前
0
0
这可能是史上最全 Redis 高可用解决方案总结

本文主要针对 Redis 常见的几种使用方式及其优缺点展开分析。 一、常见使用方式 Redis 的几种常见使用方式包括: 1.Redis 单副本; 2.Redis 多副本(主从); 3.Redis Sentinel(哨兵); 4....

别打我会飞
19分钟前
0
0
Qt编写数据可视化大屏界面电子看板11-自定义控件

一、前言 说到自定义控件,我是感觉特别熟悉的几个字,本人亲自原创的自定义控件超过110个,都是来自各个行业的具体应用真实需求,而不是凭空捏造的,当然有几个小控件也有点凑数的嫌疑,在编...

飞扬青云
22分钟前
0
0
第二讲:编写UDP Socket小程序

1、编写服务器端代码 文件-->新建 新建一个控制台程序: 下面用的是一张旧图,只要点确认就可以了。 切换到FileView视图 编译、链接 StdAfx.cpp( 里面的代码不用修改) 编辑UDPServer.cpp文...

一匹狼工作室
22分钟前
2
0
android ------ 实现高德定位并获取相应信息 ( 最新版高德SDK 和 Android SDK版本)

Android开发项目时常常会遇到定位这个功能, 很久以前写过一篇了,官方也更新了一些东西,我也更新下 以前使用的是jar包 导入来实现高德定位 老版本 链接:https://blog.csdn.net/DickyQie/...

切切歆语
25分钟前
1
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部