Appium 并发测试

启动多个Appium服务

前面课程只是启动了单个appium服务,只能控制单台设备。如果需要针对多台设备测试那么该如何处理?

首先看下面两个启动appium服务案例。

启动appium服务1

C:\Users\Shuqing>appium -p 4723
​
[Appium] Welcome ***\*to\**** Appium v1.7.2
​
[Appium] Appium REST http interface listener started on 0.0.0.0:4723

启动appium 服务2

C:\Users\Shuqing>appium -p 4725
​
[Appium] Welcome ***\*to\**** Appium v1.7.2
​
[Appium] Non-default server args:
​
[Appium]  port: 4725
​
[Appium] Appium REST http interface listener started on 0.0.0.0:4725

上面案例我们启动了2个不同的appium服务器,他们通过不同的端口来区分不同的服务;如同百米赛跑要给不同的运动员安排不同的赛道,每个运动员也只能在自己指定的赛道进行比赛。

Appium常用参数****

参数**** 默认值**** 含义****
-U, --udid null 连接物理设备的唯一设备标识符
-a, --address 0.0.0.0 监听的 ip 地址
-p, --port 4723 监听的端口
-bp, --bootstrap-port 4724 连接Android设备的端口号(Android-only)
-g, --log null 将日志输出到指定文件
--no-reset false session 之间不重置应用状态
--session-override false 允许 session 被覆盖 (冲突的话)
--app-activity null 打开Android应用时,启动的 Activity(Android-only) 的名字
--app null 本地绝对路径远程 http URL 所指向的一个安装包

更多参数请输入命令: appium -h

Appium安卓手机每次运行时都要安装 Unlock、Setting解决方案****

首先通过如下命令找到appium的安装路径

C:\Users\Shuqing>where appium
​
C:\Users\Shuqing\AppData\Roaming\npm\appium
​
C:\Users\Shuqing\AppData\Roaming\npm\appium.cmd

1.打开 C:\Users\Shuqing\AppData\Roaming\npm\node_modules\appium\node_modules\appium-android-driver\lib 中的android-helpers.js

***\*#\*******\*注释\*******\*475\*******\*行如下代码\****
​
**//await helpers.pushSettingsApp(adb);**
​

​
***\*#\*******\*注释\*******\*486\*******\*行如下代码\****
​
**//await helpers.pushUnlock(adb);**

2.打开C:\Users\Shuqing\AppData\Roaming\npm\node_modules\appium\node_modules\appium-android-driver\build\lib中的android-helpers.js

***\*#\*******\*注释\*******\*1128\*******\*行下面这行代码\****
​
//***\*return\**** _regeneratorRuntime.awrap(helpers.pushSettingsApp(adb))**;**
​

​
***\*#\*******\*修改为如下:\****
​
***\*return\**** context$1$0.abrupt('return', defaultIME)**;**
​

​
***\*#\*******\*注释\*******\*1163\*******\*行下面这行代码\****
​
//***\*return\**** _regeneratorRuntime.awrap(helpers.pushUnlock(adb))**;**
​

​
***\*#\*******\*修改如下:\****
​
***\*return\**** context$1$0.abrupt('return', defaultIME)**;**
​

修改完成后重启Appium服务即可,如果新设备没有这个两个守护App可以手动安装这两个App到设备。

Appium Setting路径**** :

{appium安装路径}\node_modules\_io.appium.settings@2.4.0@io.appium.settings\app\build\outputs\apk\settings_apk-debug.apk
​

​
eg:
​
C:\Users\Shuqing\AppData\Roaming\npm\node_modules\appium\node_modules\_io.appium.settings@2.4.0@io.appium.settings\app\build\outputs\apk\settings_apk-debug.apk

Unlock app路径:

appium安装路径\node_modules\_appium-unlock@2.0.0@appium-unlock\bin \unlock_apk-debug.apk
​

​
eg:
​
C:\Users\Shuqing\AppData\Roaming\npm\node_modules\appium\node_modules\_io.appium.settings@2.4.0@io.appium.settings\app\build\outputs\apk\unlock_apk-debug.apk

多设备启动****

前面我们已经启动了多个appium服务,那么接下来我们可以基于这些服务来启动不同的设备。

测试场景****

连接以下2台设备,然后分别启动考研帮App

· 设备1:127.0.0.1:62001

· 设备2:127.0.0.1:62025

代码实现****
multi_device.py
​
from appium import webdriver
​
import yaml
​
from time import ctime
​

​
with open('desired_caps.yaml','r')as file:
​
 data=yaml.load(file)
​

​
devices_list=['127.0.0.1:62001','127.0.0.1:62025']
​

​
def appium_desire(udid,port):
​
 desired_caps={}
​
 desired_caps['platformName']=data['platformName']
​
 desired_caps['platformVersion']=data['platformVersion']
​
 desired_caps['deviceName']=data['deviceName']
​
 desired_caps['udid']=udid
​

​
 desired_caps['app']=data['app']
​
 desired_caps['appPackage']=data['appPackage']
​
 desired_caps['appActivity']=data['appActivity']
​
 desired_caps['noReset']=data['noReset']
​

​
 print('appium port: %s start run %s at %s' **%(port,udid,ctime()))**
​
 driver=webdriver.Remote('http://'+str(data['ip'])+':'+str(port)+'/wd/hub',desired_caps)
​
 return driver
​

​
if __name__ == '__main__':
​
 appium_desire(devices_list[0],4723)
​
 appium_desire(devices_list[1],4725)
​

多进程并发启动设备****

上面的案例设备启动并不是并发进行的,而是先后执行。如何实现2台设备同时启动,并启动App呢?

测试场景****

同时启动2台设备:'127.0.0.1:62025'和'127.0.0.1:62001'并打开考研帮app

实现思路****

可以使用Python多线程或者多进程实现。这里我们推荐使用多进程( multiprocessing) 原因如下:

· 多进程中,同一个变量,各自有一份拷贝存在于每个进程中,互不影响。

· 而多线程中,所有变量都由所有线程共享,所以,任何一个变量都可以被任何一个线程修改,因此,线程之间共享数据最大的危险在于多个线程同时改一个变量,容易把内容给改乱了。

知识点补充:

· 线程与进程

· Python多进程

· python多线程

代码实现****

multi_devices_sync.py

from appium ***\*import\**** webdriver

***\*import\**** yaml

from  time ***\*import\**** ctime

***\*import\**** multiprocessing

***\*with\**** open('desired_caps.yaml','r') ***\*as\**** file:

  ***\*data\****=yaml.load(file)

devices_list=['127.0.0.1:62001','127.0.0.1:62025']

def appium_desired(udid,port):

  desired_caps={}

  desired_caps['platformName']=***\*data\****['platformName']

  desired_caps['platformVersion']=***\*data\****['platformVersion']

  desired_caps['deviceName']=***\*data\****['deviceName']

  desired_caps['udid']=udid

  desired_caps['app']=***\*data\****['app']

  desired_caps['appPackage']=***\*data\****['appPackage']

  desired_caps['appActivity']=***\*data\****['appActivity']

  desired_caps['noReset']=***\*data\****['noReset']

  print('appium port:%s start run %s at %s' %(port,udid,ctime()))

  driver=webdriver.Remote('http://'+str(***\*data\****['ip'])+':'+str(port)+'/wd/hub',desired_caps)

  driver.implicitly_wait(5)

  return driver

\#构建desired进程租

desired_process=[]

\#加载desied进程

***\*for\**** i in range(len(devices_list)):

  port = 4723 + 2 * i

  desired=multiprocessing.Process(target=appium_desired,args=(devices_list[i],port))

  desired_process.append(desired)

***\*if\**** __name__ == '__main__':

  \# 启动多设备执行测试

  ***\*for\**** desired in desired_process:

    desired.start()

  ***\*for\**** desired in desired_process:

    desired.join()

Python启动Appium 服务****

目前我们已经实现了并发启动设备,但是我们的Appium服务启动还是手动档,比如使用Dos命令或者bat批处理来手动启动appium服务,启动效率低下。如何将启动Appium服务也实现自动化呢?

方案分析****

我们可以使用python启动appium服务,这里需要使用subprocess模块,该模块可以创建新的进程,并且连接到进程的输入、输出、错误等管道信息,并且可以获取进程的返回值。

subprocess模块官方文档

测试场景****

使用Python启动2台appium服务,端口配置如下:

· Appium服务器端口:4723,bp端口为4724

· Appium服务器端口:4725,bp端口为4726

说明:bp端口( --bootstrap-port)是appium和设备之间通信的端口,如果不指定到时无法操作多台设备运行脚本。

代码实现****

首先我们使用Python脚本启动单个appium服务:

· host:127.0.0.1

· port:4723

multi_appium.py

***\*import\**** subprocess

***\*from\**** time ***\*import\**** ctime

***\*def\**** ***\*appium_start\****(host,port):

  '''启动appium server'''

  bootstrap_port = str(port + 1)

  cmd = 'start /b appium -a ' + host + ' -p ' + str(port) + ' -bp ' + str(bootstrap_port)

  print('%s at %s' %(cmd,ctime()))

  subprocess.Popen(cmd, shell=***\*True\****,stdout=open('./appium_log/'+str(port)+'.log','a'),stderr=subprocess.STDOUT)

***\*if\**** __name__ == '__main__':

  host = '127.0.0.1'

  port=4723

  appium_start(host,port)

启动校验****

启动后我们需要校验服务是否启动成功,校验方法如下:

\1. 首先查看有没有生成对应的log文件,查看log里面的内容。

\2. 使用如下命令来查看

netstat -ano |findstr 端口号

netstat 命令解释****

netstat命令是一个监控TCP/IP网络的非常有用的工具,它可以显示路由表、实际的网络连接以及每一个网络接口设备的状态信息。输入 netstat -ano 回车.可以查看本机开放的全部端口;输入命令 netstat -h可以查看全部参数含义。

C:\Users\Shuqing>netstat -ano |findstr "4723"

 TCP   127.0***\*.0.1\****:4723     0.0***\*.0.0\****:0       LISTENING    8224

关闭Appium服务****

关闭进程有2种方式,具体如下:

\1. 通过netstat命令找到对应的Appium进程pid然后可以在系统任务管理器去关闭进程;

win7系统任务管理器PID显示

\2. 使用如下命令来关闭:

taskkill -f -pid appium进程id

多个appium服务启动****

多个appium服务启动非常简单,只需在执行环境使用循环调用即可。

***\*if\**** __name__ == '__main__':

  host = '127.0.0.1'

  ***\*for\**** ***\*i\**** ***\*in\**** range(2):

    port=4723+2****\*i\****

    appium_start(host,port)

多进程并发启动appium服务****

上面的案例还不是并发执行启动appium,因此我们需要使用多进程来实现并发启动。 同样需要引入multiprocessing多进程模块。

muti_appium_sync.py

***\*import\**** multiprocessing

***\*import\**** subprocess

***\*from\**** time ***\*import\**** ctime

***\*def\**** ***\*appium_start\****(host,port):

  '''启动appium server'''

  bootstrap_port = str(port + 1)

  cmd = 'start /b appium -a ' + host + ' -p ' + str(port) + ' --bootstrap-port ' + str(bootstrap_port)

  print('%s at %s' %(cmd,ctime()))

  subprocess.Popen(cmd, shell=***\*True\****,stdout=open('./appium_log/'+str(port)+'.log','a'),stderr=subprocess.STDOUT)

**#****构建****appium****进程组**

appium_process=[]

**#****加载****appium****进程**

***\*for\**** i ***\*in\**** range(2):

  host='127.0.0.1'

  port = 4723 + 2 * i

  appium=multiprocessing.Process(target=appium_start,args=(host,port))

  appium_process.append(appium)

***\*if\**** __name__ == '__main__':

  **#****并发启动****appium****服务**

  ***\*for\**** appium ***\*in\**** appium_process:

    appium.start()

  ***\*for\**** appium ***\*in\**** appium_process:

    appium.join()

Appium端口检测****

问题思考****

经过前面学习,我们已经能够使用python启动appium服务,但是启动Appium服务之前必须保证对应的端口没有被占用,否则会出现如下报错:

C:\Users\Shuqing>appium -a 127.0.0.1 -p 4723

[Appium] Welcome ***\*to\**** Appium v1.7.2

[Appium] Non-default server args:

[Appium]  address: 127.0.0.1

[HTTP] Could ***\*not\**** start REST http interface listener. The requested port may already be ***\*in\**** use. Please make sure there is no other instance of this server running already.

uncaughtException: listen EADDRINUSE 127.0.0.1:4723

针对以上这种情况,我们在启动appium服务前该如何检测端口是否可用呢?对于被占用的端口我们又该如何释放?

需求分析****

\1. 自动检测端口是否被占用

\2. 如果端口被占用则自动关闭对应端口的进程

端口检测****

端口检测需要使用到socket模块来校验端口是否被占用。

python socket模块官方文档

什么是socket?****

网络上的两个程序通过一个双向的通信连接实现数据的交换,这个连接的一端称为一个socket。建立网络通信连接至少要一对端口号(socket)。

socket本质是编程接口(API),对TCP/IP的封装,TCP/IP也要提供可供程序员做网络开发所用的接口,这就是Socket编程接口;HTTP是轿车,提供了封装或者显示数据的具体形式;Socket是发动机,提供了网络通信的能力。

例如当你用浏览器打开我要自学网主页时,你的浏览器会创建一个socket并命令它去连接 自学网的服务器主机,服务器也对客户端的请求创建一个socket进行监听。两端使用各自的socket来发送和接收信息。在socket通信的时候,每个socket都被绑定到一个特定的IP地址和端口。

补充资料: 网络工程师视频教程

代码实现****

check_port.py

***\*import\**** socket

***\*def\**** ***\*check_port\****(host, port):

  """检测指定的端口是否被占用"""

  **#****创建****socket****对象**

  s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)

  ***\*try\****:

    s.connect((host, port))

    s.shutdown(2)

  ***\*except\**** OSError ***\*as\**** msg:

    print('port %s is available! ' %port)

    print(msg)

    ***\*return\**** ***\*True\****

  ***\*else\****:

    print('port %s already be in use !' % port)

    ***\*return\**** ***\*False\****

***\*if\**** __name__ == '__main__':

  host='127.0.0.1'

  port=4723

  check_port(host,port)

方法****

shutdown(self, flag):禁止在一个Socket上进行数据的接收与发送。利用shutdown()函数使socket双向数据传输变为单向数据传输。shutdown()需要一个单独的参数, 该参数表示了如何关闭socket

参数****

· 0表示禁止将来读;

· 1表示禁止将来写

· 2表示禁止将来读和写。

当端口可以使用时,控制台输出如下:此使说明服务端没有开启这个端口服务,所以可用。

C:\Python35\python.exe E:/AppiumScript/advance/appium_cmd/appium_multiProcess.py

port 4723 is available! 

[WinError 10061] 由于目标计算机积极拒绝,无法连接。 

Process finished with ***\*exit\**** code 0

端口释放****

如果端口被占用,则需要释放该端口。那么怎么样去释放被占用的端口呢?

代码实现****

check_port.py

***\*import\**** os

***\*def\**** ***\*release_port\****(port):

  """释放指定的端口"""

  **#****查找对应端口的****pid**

  cmd_find='netstat -aon | findstr %s' %port

  print(cmd_find)

  **#****返回命令执行后的结果**

  result = os.popen(cmd_find).read()

  print(result)

  ***\*if\**** str(port) ***\*and\**** 'LISTENING' ***\*in\**** result:

    **#****获取端口对应的****pid****进程**

    i=result.index('LISTENING')

    start=i+len('LISTENING')+7

    end=result.index('\n')

    pid=result[start:end]

    **#** **关闭被占用端口的****pid**

    cmd_kill='taskkill -f -pid %s' %pid

    print(cmd_kill)

    os.popen(cmd_kill)

  ***\*else\****:

    print('port %s is available !' %port)

***\*if\**** __name__ == '__main__':

  host='127.0.0.1'

  port=4723

  **# check_port(host,port)**

  release_port(port)

控制台显示:

C:\Python35\python.exe E:/AppiumScript/advance/appium_cmd/appium_multiProcess.py

netstat -aon | findstr "4723"

 TCP   127.0.0.1:4723     0.0.0.0:0       LISTENING    29532

taskkill -f -pid 29532

Process finished with ***\*exit\**** code 0

Appium并发测试综合实践****

测试场景****

并发启动2个appium服务,再并发启动2台设备测试考研帮App

2个appium服务,端口配置如下:

· Appium服务器端口:4723,bp端口为4724

· Appium服务器端口:4725,bp端口为4726

2台设备:

· 127.0.0.1:62025

· 127.0.0.1:62001

测试app:考研帮Andriod版

场景分析****

其实就是将前面所讲的两部分组合起来,先启动appium服务,再分配设备启动app。

代码实现****

appium_devices_sync.py

***\*from\**** appium_sync.multi_appium ***\*import\**** appium_start

***\*from\**** appium_sync.multi_devices ***\*import\**** appium_desired

***\*from\**** appium_sync.check_port ***\*import\**** *

***\*from\**** time ***\*import\**** sleep

***\*import\**** multiprocessing

devices_list=['127.0.0.1:62025','127.0.0.1:62001']

***\*def\**** ***\*start_appium_action\****(host,port):

  '''检测端口是否被占用,如果没有被占用则启动appium服务'''

  ***\*if\**** check_port(host,port):

    appium_start(host,port)

    ***\*return\**** ***\*True\****

  ***\*else\****:

    print('appium %s start failed!' %port)

    ***\*return\**** ***\*False\****

***\*def\**** ***\*start_devices_action\****(udid,port):

  '''先检测appium服务是否启动成功,启动成功则再启动App,否则释放端口'''

  host='127.0.0.1'

  ***\*if\**** start_appium_action(host,port):

    appium_desired(udid,port)

  ***\*else\****:

    release_port(port)

***\*def\**** ***\*appium_start_sync\****():

  '''并发启动appium服务'''

  print('====appium_start_sync=====')

  **#****构建****appium****进程组**

  appium_process=[]

  **#****加载****appium****进程**

  ***\*for\**** i ***\*in\**** range(len(devices_list)):

    host='127.0.0.1'

    port = 4723 + 2 * i

    appium=multiprocessing.Process(target=start_appium_action,args=(host,port))

    appium_process.append(appium)

  **#** **启动****appium****服务**

  ***\*for\**** appium ***\*in\**** appium_process:

    appium.start()

  ***\*for\**** appium ***\*in\**** appium_process:

    appium.join()

  sleep(5)

***\*def\**** ***\*devices_start_sync\****():

  '''并发启动设备'''

  print('===devices_start_sync===')

  **#****定义****desired****进程组**

  desired_process = []

  **#****加载****desired****进程**

  ***\*for\**** i ***\*in\**** range(len(devices_list)):

    port = 4723 + 2 * i

    desired = multiprocessing.Process(target=start_devices_action, args=(devices_list[i], port))

    desired_process.append(desired)

  **#****并发启动****App**

  ***\*for\**** desired ***\*in\**** desired_process:

    desired.start()

  ***\*for\**** desired ***\*in\**** desired_process:

    desired.join()

***\*if\**** __name__ == '__main__':

  appium_start_sync()

  devices_start_sync()

补充资料:谈谈TCP中的TIME_WAIT

netstat -ano |findstr 4723

 TCP   127.0***\*.0.1\****:4723     127.0***\*.0.1\****:63255    TIME_WAIT    0

 TCP   127.0***\*.0.1\****:4723     127.0***\*.0.1\****:63257    TIME_WAIT    0

 TCP   127.0***\*.0.1\****:4723     127.0***\*.0.1\****:63260    TIME_WAIT    0

 TCP   127.0***\*.0.1\****:62998    127.0***\*.0.1\****:4723     TIME_WAIT    0

port 4723 is available

并发用例执行****

测试场景****

再上面的场景基础之上,并发启动设备后然后执行跳过引导页面操作。

代码实现****

kyb_test.py

***\*from\**** selenium.common.exceptions ***\*import\**** NoSuchElementException

***\*class\**** ***\*KybTest\****(object):

  ***\*def\**** ***\*__init__\****(self,driver):

    self.driver=driver

  ***\*def\**** ***\*check_cancelBtn\****(self):

    print('check cancelBtn')

    ***\*try\****:

      cancelBtn = self.driver.find_element_by_id('android:id/button2')

    ***\*except\**** NoSuchElementException:

      print('no cancelBtn')

    ***\*else\****:

      cancelBtn.click()

  ***\*def\**** ***\*check_skipBtn\****(self):

    print('check skipBtn')

    ***\*try\****:

      skipBtn = self.driver.find_element_by_id('com.tal.kaoyan:id/tv_skip')

    ***\*except\**** NoSuchElementException:

      print('no skipBtn')

    ***\*else\****:

      skipBtn.click()

  ***\*def\**** ***\*skip_update_guide\****(self):

    self.check_cancelBtn()

    self.check_skipBtn()

将执行的用例集成到 multi_devices.py

from appium ***\*import\**** webdriver

***\*import\**** yaml

from  time ***\*import\**** ctime

from appium_sync.kyb_test ***\*import\**** KybTest

***\*with\**** open('desired_caps.yaml','r') ***\*as\**** file:

  ***\*data\****=yaml.load(file)

devices_list=['127.0.0.1:62001','127.0.0.1:62025']

def appium_desired(udid,port):

  desired_caps={}

  desired_caps['platformName']=***\*data\****['platformName']

  desired_caps['platformVersion']=***\*data\****['platformVersion']

  desired_caps['deviceName']=***\*data\****['deviceName']

  desired_caps['udid']=udid

  desired_caps['app']=***\*data\****['app']

  desired_caps['appPackage']=***\*data\****['appPackage']

  desired_caps['appActivity']=***\*data\****['appActivity']

  desired_caps['noReset']=***\*data\****['noReset']

  print('appium port:%s start run %s at %s' %(port,udid,ctime()))

  driver=webdriver.Remote('http://'+str(***\*data\****['ip'])+':'+str(port)+'/wd/hub',desired_caps)

  driver.implicitly_wait(5)

  k=KybTest(driver)

  k.skip_update_guide()

  return driver

***\*if\**** __name__ == '__main__':

  appium_desired(devices_list[0],4723)

  appium_desired(devices_list[1],4725)

基于Docker+STF Appium并发测试****

Docker

STF

实践案例:github.com/haifengrund…

参考资料****

· testerhome.com/topics/2918

· testerhome.com/topics/7917

· baike.baidu.com/item/socket…

· www.runoob.com/python/os-p…

· blog.csdn.net/jewes/artic…

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,496评论 6 501
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,407评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,632评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,180评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,198评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,165评论 1 299
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,052评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,910评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,324评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,542评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,711评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,424评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,017评论 3 326
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,668评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,823评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,722评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,611评论 2 353

推荐阅读更多精彩内容