分类: Android Junit测试2011-11-14 16:49 1601人阅读 评论(2) 收藏 举报

测试相关资源 
让开发自动化: 用 Eclipse 插件提高代码质量http://www.ibm.com/developerworks/cn/java/j-ap01117/index.html

代码测试覆盖率介绍:http://www.cnblogs.com/coderzh/archive/2009/03/29/1424344.html

学习android单元测试时遇到的一些问题: 
1.开始以为单元测试一定要从程序的launch Activity,一步一步的跳转到所要测试的Activity能对其进行测试。 
但实际上,我们可以从任意一个activity开始,对任意一个activity进行测试。

2.在运行单元测试之前,一定要先将要测试的程序安装到模拟器或真机上。

junit相关 
android中的测试框架是扩展的junit3,所以在学习android中的单元测试签,可以先熟悉下junit3的使用,junit3要学习的东西应该并不多,就几页纸的东西。入门可以参考这个:http://android.blog.51cto.com/268543/49994

android单元测试框架中涉及的注解 
@Suppress 可以用在类或这方法上,这样该类或者该方法就不会被执行 
@UiThreadTest 可以用在方法上,这样该方法就会在程序的ui线程上执行 
@LargeTest, @MediumTest, @SmallTest 用在方法上,标记所属的测试类型,主要是用于单独执行其中的某一类测试时使用。具体参考InstrumentationTestRunner类的文档。
@Smoke 具体用法还不清楚

android单元测试框架中涉及的一些类的uml

接下来我们以demo project来讲解如何使用android中的单元测试 
主要包括了三个activity: 
MainActivity:仅包含一个button,点击后就可以进入LoginActivity

LoginActivity:可以输入username, password,然后点击submit的话可进入HomeActivity,如果点击reset的话,输入的内容就会被清空

HomeActivity:在TextView中显示LoginActivity输入的内容

首先我们创建要测试的项目demo(使用了2.1)

MainActivity代码

Java代码

public class MainActivity extends Activity {

private static final boolean DEBUG = true;

private static final String TAG = "-- MainActivity";

@Override

protected void onCreate(Bundle savedInstanceState) {

if (DEBUG) {

Log.i(TAG, "onCreate");

}

super.onCreate(savedInstanceState);

setContentView(R.layout.act_main);

View toLoginView = findViewById(R.id.to_login);

toLoginView.setOnClickListener(new View.OnClickListener() {

public void onClick(View view) {

if (DEBUG) {

Log.i(TAG, "toLoginView clicked");

}

Intent intent = new Intent(getApplicationContext(), LoginActivity.class);

startActivity(intent);

}

});

}

}

MainActivity的布局文件

Xml代码

  1. <LinearLayout
  2. xmlns:android="http://schemas.android.com/apk/res/android"
  3. android:layout_width="fill_parent"
  4. android:layout_height="fill_parent"
  5. >
  6. <Button
  7. android:id="@+id/to_login"
  8. android:layout_width="fill_parent"
  9. android:layout_height="wrap_content"
  10. android:layout_gravity="bottom"
  11. android:text="to login" />
  12. </LinearLayout>

<LinearLayout

xmlns:android="http://schemas.android.com/apk/res/android"

android:layout_width="fill_parent"

android:layout_height="fill_parent"

>

<Button

android:id="@+id/to_login"

android:layout_width="fill_parent"

android:layout_height="wrap_content"

android:layout_gravity="bottom"

android:text="to login" />

</LinearLayout>

LoginActivity代码

Java代码

  1. public class LoginActivity extends Activity {
  2. private static final boolean DEBUG = true;
  3. private static final String TAG = "-- LoginActivity";
  4. private EditText mUsernameView;
  5. private EditText mPasswordView;
  6. @Override
  7. protected void onCreate(Bundle savedInstanceState) {
  8. if (DEBUG) {
  9. Log.i(TAG, "onCreate");
  10. }
  11. super.onCreate(savedInstanceState);
  12. setContentView(R.layout.act_login);
  13. mUsernameView = (EditText) findViewById(R.id.username);
  14. mPasswordView = (EditText) findViewById(R.id.password);
  15. View submitView = findViewById(R.id.submit);
  16. submitView.setOnClickListener(new View.OnClickListener() {
  17. public void onClick(View view) {
  18. if (DEBUG) {
  19. Log.i(TAG, "submitView clicked");
  20. }
  21. Intent intent = new Intent(getApplicationContext(), HomeActivity.class);
  22. intent.putExtra(HomeActivity.EXTRA_USERNAME, mUsernameView.getText().toString());
  23. intent.putExtra(HomeActivity.EXTRA_PASSWORD, mPasswordView.getText().toString());
  24. startActivity(intent);
  25. }
  26. });
  27. View resetView = findViewById(R.id.reset);
  28. resetView.setOnClickListener(new View.OnClickListener() {
  29. public void onClick(View view) {
  30. if (DEBUG) {
  31. Log.i(TAG, "resetView clicked");
  32. }
  33. mUsernameView.setText("");
  34. mPasswordView.setText("");
  35. mUsernameView.requestFocus();
  36. }
  37. });
  38. }
  39. }

public class LoginActivity extends Activity {

private static final boolean DEBUG = true;

private static final String TAG = "-- LoginActivity";

private EditText mUsernameView;

private EditText mPasswordView;

@Override

protected void onCreate(Bundle savedInstanceState) {

if (DEBUG) {

Log.i(TAG, "onCreate");

}

super.onCreate(savedInstanceState);

setContentView(R.layout.act_login);

mUsernameView = (EditText) findViewById(R.id.username);

mPasswordView = (EditText) findViewById(R.id.password);

View submitView = findViewById(R.id.submit);

submitView.setOnClickListener(new View.OnClickListener() {

public void onClick(View view) {

if (DEBUG) {

Log.i(TAG, "submitView clicked");

}

Intent intent = new Intent(getApplicationContext(), HomeActivity.class);

intent.putExtra(HomeActivity.EXTRA_USERNAME, mUsernameView.getText().toString());

intent.putExtra(HomeActivity.EXTRA_PASSWORD, mPasswordView.getText().toString());

startActivity(intent);

}

});

View resetView = findViewById(R.id.reset);

resetView.setOnClickListener(new View.OnClickListener() {

public void onClick(View view) {

if (DEBUG) {

Log.i(TAG, "resetView clicked");

}

mUsernameView.setText("");

mPasswordView.setText("");

mUsernameView.requestFocus();

}

});

}

}

LoginActivity的布局文件

Xml代码

  1. <LinearLayout
  2. xmlns:android="http://schemas.android.com/apk/res/android"
  3. android:layout_width="fill_parent"
  4. android:layout_height="fill_parent"
  5. android:orientation="vertical"
  6. >
  7. <TextView
  8. android:id="@+id/label_username"
  9. android:layout_width="fill_parent"
  10. android:layout_height="wrap_content"
  11. android:text="username:" />
  12. <EditText
  13. android:id="@+id/username"
  14. android:layout_width="fill_parent"
  15. android:layout_height="wrap_content"
  16. android:inputType="text" />
  17. <TextView
  18. android:id="@+id/label_password"
  19. android:layout_width="fill_parent"
  20. android:layout_height="wrap_content"
  21. android:text="password:" />
  22. <EditText
  23. android:id="@+id/password"
  24. android:layout_width="fill_parent"
  25. android:layout_height="wrap_content"
  26. android:inputType="textPassword" />
  27. <Button
  28. android:id="@+id/submit"
  29. android:layout_width="fill_parent"
  30. android:layout_height="wrap_content"
  31. android:text="submit" />
  32. <Button
  33. android:id="@+id/reset"
  34. android:layout_width="fill_parent"
  35. android:layout_height="wrap_content"
  36. android:text="reset" />
  37. </LinearLayout>

<LinearLayout

xmlns:android="http://schemas.android.com/apk/res/android"

android:layout_width="fill_parent"

android:layout_height="fill_parent"

android:orientation="vertical"

>

<TextView

android:id="@+id/label_username"

android:layout_width="fill_parent"

android:layout_height="wrap_content"

android:text="username:" />

<EditText

android:id="@+id/username"

android:layout_width="fill_parent"

android:layout_height="wrap_content"

android:inputType="text" />

<TextView

android:id="@+id/label_password"

android:layout_width="fill_parent"

android:layout_height="wrap_content"

android:text="password:" />

<EditText

android:id="@+id/password"

android:layout_width="fill_parent"

android:layout_height="wrap_content"

android:inputType="textPassword" />

<Button

android:id="@+id/submit"

android:layout_width="fill_parent"

android:layout_height="wrap_content"

android:text="submit" />

<Button

android:id="@+id/reset"

android:layout_width="fill_parent"

android:layout_height="wrap_content"

android:text="reset" />

</LinearLayout>

HomeActivity代码

Java代码

public class HomeActivity extends Activity {

private static final boolean DEBUG = true;

private static final String TAG = "-- HomeActivity";

public static final String EXTRA_USERNAME = "yuan.activity.username";

public static final String EXTRA_PASSWORD = "yuan.activity.password";

@Override

protected void onCreate(Bundle savedInstanceState) {

if (DEBUG) {

Log.i(TAG, "onCreate");

}

super.onCreate(savedInstanceState);

Intent intent = getIntent();

StringBuilder sb = new StringBuilder();

sb.append("username:").append(intent.getStringExtra(EXTRA_USERNAME)).append("\n");

sb.append("password:").append(intent.getStringExtra(EXTRA_PASSWORD));

setContentView(R.layout.act_home);

TextView loginContentView = (TextView) findViewById(R.id.login_content);

loginContentView.setText(sb.toString());

}

}

HomeActivity的布局文件

Xml代码

<LinearLayout

xmlns:android="http://schemas.android.com/apk/res/android"

android:layout_width="fill_parent"

android:layout_height="fill_parent"

>

<TextView

android:id="@+id/login_content"

android:layout_width="fill_parent"

android:layout_height="wrap_content"

android:layout_gravity="center_vertical"

android:gravity="center"

android:textColor="#EEE"

android:textStyle="bold"

android:textSize="25sp" />

</LinearLayout>

程序非常简单,接下来我们为demo创建单元测试工程demo_unittest

选择之前创建的工程demo,然后eclipse会自动帮我们设定api level,包名等。(测试用例的包名一般就是在要测试类的包名后加上test)

创建完后eclipse会自动为我们创建好所需的目录,Manifest.xml文件

接下来就是为要测试的类编写测试用例。关于要用哪个测试用例类,在第一张UML图中也做了简要的说明。 
ActivityInstrumentationTestCase2:主要是用于进行activity的功能测试,和activity的交互测试,如activity间的跳转,ui的交互等。

ActivityInstrumentationTestCase:这个类现在已deprecated了,所以不许考虑。

SingleLaunchActivityTestCase:该测试用例仅掉用setUp和tearDown一次,而不像其它测试用例类一样,每调用一次测试方法就会重新调用一次setUp和tearDown。所以主要测试activity是否能够正确处理多次调用。

ActivityUnitTestCase:主要用于测试Activity,因为它允许注入MockContext和MockApplicaton,所以可以测试Activity在不同资源和应用下的情况。

还有Application等的测试用例比较简单,看uml图。如果觉得不够详细,可以参考sdk文档的dev guide和api reference。

MainActivityTest测试用例

Java代码

  1. public class MainActivityTest extends ActivityInstrumentationTestCase2<MainActivity> {
  2. private static final String TAG = "=== MainActivityTest";
  3. private Instrumentation mInstrument;
  4. private MainActivity mActivity;
  5. private View mToLoginView;
  6. public MainActivityTest() {
  7. super("yuan.activity", MainActivity.class);
  8. }
  9. @Override
  10. public void setUp() throws Exception {
  11. super.setUp();
  12. mInstrument = getInstrumentation();
  13. // 启动被测试的Activity
  14. mActivity = getActivity();
  15. mToLoginView = mActivity.findViewById(yuan.activity.R.id.to_login);
  16. }
  17. public void testPreConditions() {
  18. // 在执行测试之前,确保程序的重要对象已被初始化
  19. assertTrue(mToLoginView != null);
  20. }
  21. //mInstrument.runOnMainSync(new Runnable() {
  22. //  public void run() {
  23. //      mToLoginView.requestFocus();
  24. //      mToLoginView.performClick();
  25. //  }
  26. //});
  27. @UiThreadTest
  28. public void testToLogin() {
  29. // @UiThreadTest注解使整个方法在UI线程上执行,等同于上面注解掉的代码
  30. mToLoginView.requestFocus();
  31. mToLoginView.performClick();
  32. }
  33. @Suppress
  34. public void testNotCalled() {
  35. // 使用了@Suppress注解的方法不会被测试
  36. Log.i(TAG, "method 'testNotCalled' is called");
  37. }
  38. @Override
  39. public void tearDown() throws Exception {
  40. super.tearDown();
  41. }
  42. }

public class MainActivityTest extends ActivityInstrumentationTestCase2<MainActivity> {

private static final String TAG = "=== MainActivityTest";

private Instrumentation mInstrument;

private MainActivity mActivity;

private View mToLoginView;

public MainActivityTest() {

super("yuan.activity", MainActivity.class);

}

@Override

public void setUp() throws Exception {

super.setUp();

mInstrument = getInstrumentation();

// 启动被测试的Activity

mActivity = getActivity();

mToLoginView = mActivity.findViewById(yuan.activity.R.id.to_login);

}

public void testPreConditions() {

// 在执行测试之前,确保程序的重要对象已被初始化

assertTrue(mToLoginView != null);

}

//mInstrument.runOnMainSync(new Runnable() {

//  public void run() {

//   mToLoginView.requestFocus();

//   mToLoginView.performClick();

//  }

//});

@UiThreadTest

public void testToLogin() {

// @UiThreadTest注解使整个方法在UI线程上执行,等同于上面注解掉的代码

mToLoginView.requestFocus();

mToLoginView.performClick();

}

@Suppress

public void testNotCalled() {

// 使用了@Suppress注解的方法不会被测试

Log.i(TAG, "method 'testNotCalled' is called");

}

@Override

public void tearDown() throws Exception {

super.tearDown();

}

}

LoginActivityTest测试用例

Java代码

  1. public class LoginActivityTest extends ActivityInstrumentationTestCase2<LoginActivity> {
  2. private static final String TAG = "=== LoginActivityTest";
  3. private Instrumentation mInstrument;
  4. private LoginActivity mActivity;
  5. private EditText mUsernameView;
  6. private EditText mPasswordView;
  7. private View mSubmitView;
  8. private View mResetView;
  9. public LoginActivityTest() {
  10. super("yuan.activity", LoginActivity.class);
  11. }
  12. @Override
  13. public void setUp() throws Exception {
  14. super.setUp();
  15. /*
  16. *  要向程序发送key事件的话,必须在getActivity之前调用该方法来关闭touch模式
  17. * 否则key事件会被忽略
  18. */
  19. setActivityInitialTouchMode(false);
  20. mInstrument = getInstrumentation();
  21. mActivity = getActivity();
  22. Log.i(TAG, "current activity: " + mActivity.getClass().getName());
  23. mUsernameView = (EditText) mActivity.findViewById(yuan.activity.R.id.username);
  24. mPasswordView = (EditText) mActivity.findViewById(yuan.activity.R.id.password);
  25. mSubmitView = mActivity.findViewById(yuan.activity.R.id.submit);
  26. mResetView = mActivity.findViewById(yuan.activity.R.id.reset);
  27. }
  28. public void testPreConditions() {
  29. assertTrue(mUsernameView != null);
  30. assertTrue(mPasswordView != null);
  31. assertTrue(mSubmitView != null);
  32. assertTrue(mResetView != null);
  33. }
  34. public void testInput() {
  35. input();
  36. assertEquals("yuan", mUsernameView.getText().toString());
  37. assertEquals("1123", mPasswordView.getText().toString());
  38. }
  39. public void testSubmit() {
  40. input();
  41. mInstrument.runOnMainSync(new Runnable() {
  42. public void run() {
  43. mSubmitView.requestFocus();
  44. mSubmitView.performClick();
  45. }
  46. });
  47. }
  48. public void testReset() {
  49. input();
  50. mInstrument.runOnMainSync(new Runnable() {
  51. public void run() {
  52. mResetView.requestFocus();
  53. mResetView.performClick();
  54. }
  55. });
  56. assertEquals("", mUsernameView.getText().toString());
  57. assertEquals("", mPasswordView.getText().toString());
  58. }
  59. @Override
  60. public void tearDown() throws Exception {
  61. super.tearDown();
  62. }
  63. private void input() {
  64. mActivity.runOnUiThread(new Runnable() {
  65. public void run() {
  66. mUsernameView.requestFocus();
  67. }
  68. });
  69. // 因为测试用例运行在单独的线程上,这里最好要
  70. // 同步application,等待其执行完后再运行
  71. mInstrument.waitForIdleSync();
  72. sendKeys(KeyEvent.KEYCODE_Y, KeyEvent.KEYCODE_U,
  73. KeyEvent.KEYCODE_A, KeyEvent.KEYCODE_N);
  74. // 效果同上面sendKeys之前的代码
  75. mInstrument.runOnMainSync(new Runnable() {
  76. public void run() {
  77. mPasswordView.requestFocus();
  78. }
  79. });
  80. sendKeys(KeyEvent.KEYCODE_1, KeyEvent.KEYCODE_1,
  81. KeyEvent.KEYCODE_2, KeyEvent.KEYCODE_3);
  82. }
  83. }

public class LoginActivityTest extends ActivityInstrumentationTestCase2<LoginActivity> {

private static final String TAG = "=== LoginActivityTest";

private Instrumentation mInstrument;

private LoginActivity mActivity;

private EditText mUsernameView;

private EditText mPasswordView;

private View mSubmitView;

private View mResetView;

public LoginActivityTest() {

super("yuan.activity", LoginActivity.class);

}

@Override

public void setUp() throws Exception {

super.setUp();

/*

*  要向程序发送key事件的话,必须在getActivity之前调用该方法来关闭touch模式

* 否则key事件会被忽略

*/

setActivityInitialTouchMode(false);

mInstrument = getInstrumentation();

mActivity = getActivity();

Log.i(TAG, "current activity: " + mActivity.getClass().getName());

mUsernameView = (EditText) mActivity.findViewById(yuan.activity.R.id.username);

mPasswordView = (EditText) mActivity.findViewById(yuan.activity.R.id.password);

mSubmitView = mActivity.findViewById(yuan.activity.R.id.submit);

mResetView = mActivity.findViewById(yuan.activity.R.id.reset);

}

public void testPreConditions() {

assertTrue(mUsernameView != null);

assertTrue(mPasswordView != null);

assertTrue(mSubmitView != null);

assertTrue(mResetView != null);

}

public void testInput() {

input();

assertEquals("yuan", mUsernameView.getText().toString());

assertEquals("1123", mPasswordView.getText().toString());

}

public void testSubmit() {

input();

mInstrument.runOnMainSync(new Runnable() {

public void run() {

mSubmitView.requestFocus();

mSubmitView.performClick();

}

});

}

public void testReset() {

input();

mInstrument.runOnMainSync(new Runnable() {

public void run() {

mResetView.requestFocus();

mResetView.performClick();

}

});

assertEquals("", mUsernameView.getText().toString());

assertEquals("", mPasswordView.getText().toString());

}

@Override

public void tearDown() throws Exception {

super.tearDown();

}

private void input() {

mActivity.runOnUiThread(new Runnable() {

public void run() {

mUsernameView.requestFocus();

}

});

// 因为测试用例运行在单独的线程上,这里最好要

// 同步application,等待其执行完后再运行

mInstrument.waitForIdleSync();

sendKeys(KeyEvent.KEYCODE_Y, KeyEvent.KEYCODE_U,

KeyEvent.KEYCODE_A, KeyEvent.KEYCODE_N);

// 效果同上面sendKeys之前的代码

mInstrument.runOnMainSync(new Runnable() {

public void run() {

mPasswordView.requestFocus();

}

});

sendKeys(KeyEvent.KEYCODE_1, KeyEvent.KEYCODE_1,

KeyEvent.KEYCODE_2, KeyEvent.KEYCODE_3);

}

}

HomeActivityTest测试用例

Java代码

  1. public class HomeActivityTest extends ActivityUnitTestCase<HomeActivity> {
  2. private static final String TAG = "=== HomeActivityTest";
  3. private static final String LOGIN_CONTENT = "username:yuan\npassword:1123";
  4. private HomeActivity mHomeActivity;
  5. private TextView mLoginContentView;
  6. public HomeActivityTest() {
  7. super(HomeActivity.class);
  8. }
  9. @Override
  10. public void setUp() throws Exception {
  11. super.setUp();
  12. Intent intent = new Intent();
  13. intent.putExtra(HomeActivity.EXTRA_USERNAME, "yuan");
  14. intent.putExtra(HomeActivity.EXTRA_PASSWORD, "1123");
  15. // HomeActivity有extra参数,所以我们需要以intent来启动它
  16. mHomeActivity = launchActivityWithIntent("yuan.activity", HomeActivity.class, intent);
  17. mLoginContentView = (TextView) mHomeActivity.findViewById(yuan.activity.R.id.login_content);
  18. }
  19. public void testLoginContent() {
  20. assertEquals(LOGIN_CONTENT, mLoginContentView.getText().toString());
  21. }
  22. @Override
  23. public void tearDown() throws Exception {
  24. super.tearDown();
  25. }
  26. }

public class HomeActivityTest extends ActivityUnitTestCase<HomeActivity> {

private static final String TAG = "=== HomeActivityTest";

private static final String LOGIN_CONTENT = "username:yuan\npassword:1123";

private HomeActivity mHomeActivity;

private TextView mLoginContentView;

public HomeActivityTest() {

super(HomeActivity.class);

}

@Override

public void setUp() throws Exception {

super.setUp();

Intent intent = new Intent();

intent.putExtra(HomeActivity.EXTRA_USERNAME, "yuan");

intent.putExtra(HomeActivity.EXTRA_PASSWORD, "1123");

// HomeActivity有extra参数,所以我们需要以intent来启动它

mHomeActivity = launchActivityWithIntent("yuan.activity", HomeActivity.class, intent);

mLoginContentView = (TextView) mHomeActivity.findViewById(yuan.activity.R.id.login_content);

}

public void testLoginContent() {

assertEquals(LOGIN_CONTENT, mLoginContentView.getText().toString());

}

@Override

public void tearDown() throws Exception {

super.tearDown();

}

}

接下来是运行测试用例,首先我们需要把要测试的程序安装到模拟器或真机上

运行测试用例,查看运行结果

这里仅仅讲了使用eclipse来进行单元测试,当然也是可以在命令行中进行单元测试的,但既然有eclipse这种图形界面的工具,就不再折腾什么命令行了。 
还有就是测试用例也可以直接创建在源程序中(即源代码和测试代码放在一个项目中),具体怎么做的话google一些吧,就是把测试时涉及的一些Manifest元素移到源码工程的Manifest中等

android单元测试 activity跳转 以及 input 输入后 测试的更多相关文章

  1. Android的Activity跳转动画各种效果整理

    Android的Activity跳转就是很生硬的切换界面.其实Android的Activity跳转可以设置各种动画,本文整理了一些,还有很多动画效果,就要靠我们发挥自己的想象力 大家使用Android ...

  2. Android——关于Activity跳转的返回(无返回值和有返回值)——有返回值

    说明: 跳转页面,并将第一页的Edittext输入的数据通过按钮Button传到第二页用Edittext显示,点击第二页的 返回按钮Button返回第一页(改变第二页的Edittext的内容会传至第一 ...

  3. Android之Activity跳转

    简述 如果把每个activity看成一个页面的话,那么activity之间的跳转和页面的之间的跳转基本上是一样的.首先需要监听一个事件,当这个事件发生的时候,就进行跳转.html中有个<a sr ...

  4. android 15 activity跳转

    从一个屏幕跳到另一个屏幕,一个activity跳转到另一个activity,Intent类用于组件之间传递数据和跳转,组件包括不仅activity. package com.sxt.day04_01; ...

  5. android入门,activity跳转,并传递message

    首先是布局文件,如下: activity_main.xml <?xml version="1.0" encoding="utf-8"?> <L ...

  6. android之 Activity跳转出现闪屏

    原文:http://blog.csdn.net/az313/article/details/17321549 同一个应用,在不同手机上测试,Activity之间跳转出现闪屏,界面来回跳转…… 查阅网上 ...

  7. Android:activity跳转过渡效果

    放在startActivity(intent);后面 overridePendingTransition(android.R.anim.fade_in,android.R.anim.fade_out) ...

  8. Android——Activity跳转

    Activity_main.xml <?xml version="1.0" encoding="utf-8"?> <RelativeLayou ...

  9. Android笔记---Intent实现Activity跳转

    学了之前的Android控件以及布局,我们就能够做一些UI的设计了,这里我结合之前的知识.以一个小的登录项目来解说下Activity之间跳转. 先看下效果图: 1.登录界面: 2.点击登录按钮跳转到另 ...

随机推荐

  1. 微信小程序路过——新手不要错过哦!

    应该算是入门篇, 从我怎么0基础然后沿着什么方向走,遇到的什么坑,如何方向解决,不过本人接触不是很多,所以也就了解有限. 小程序的前提: 1.小程序大小不允许超过2M.(也就是本地图片,大图精图不要在 ...

  2. Solidity 官方文档中文版 4_Solidity 编程实例

    Voting 投票 接下来的合约非常复杂,但展示了很多Solidity的特性.它实现了一个投票合约.当然,电子选举的主要问题是如何赋予投票权给准确的人,并防止操纵.我们不能解决所有的问题,但至少我们会 ...

  3. python 浮点数转分数

    from fractions import Fraction value = 4.2 print(Fraction(value).limit_denominator())

  4. android:点击popupwindow以外区域 popupwindow自动消失

    方法一(这种方法可以处理popupwindows dimiss的时候一些其他的操作,比如让其他控件的隐藏,消失等): 代码如下popupWindow.setFocusable(false);//foc ...

  5. 关于UDP很好的书籍和文章(整理、持续更新)

    文章 告知你不为人知的 UDP:疑难杂症和使用(必看)

  6. WPF——RenderTransform特效

    WPF: RenderTransform特效 WPF中的变形(RenderTransform)类是为了达到直接去改变某个Silverlight对象的形状(比如缩放.旋转一个元素)的目的而设计的,Ren ...

  7. 001——数组(一)数组知识及foreach函数应用

    <?php /**数组(一)数组知识及foreach函数应用*/ /*数组:在一个变量中,存储一个或多个值,每一个元素都有一个访问ID * * */ /* * //索引型数组 $arr=arra ...

  8. java应用简单递归

    毕业后就怎么学过算法,还在上学的时候学过数据结构,现在基本上都还给老师了,可惜老师学费没有还给我... 情景: 类似于给定一个数字,算他由多少个数字组成,比如:36 现在有10.5.1 ,那么最佳帅3 ...

  9. SharePoint开发 - TimerJob简单实例讲解

    博客地址 http://blog.csdn.net/foxdave SharePoint中的TimerJob类似于Windows系统的计划任务,可以实现定时执行指定操作的功能. 本篇所述的实例为在Sh ...

  10. I.MX6 Linux U-boot 环境变量解析

    /********************************************************************************** * I.MX6 Linux U- ...