android studio 写一个小计时器(版本二)

as版本:23.3.1patch2

例程:timer

在前一个版本的基本上改的,增加了继续的功能,实现方法稍微不同。

动画演示:

activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/main"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity">

    <Button
        android:id="@+id/btn_clear"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_marginStart="144dp"
        android:layout_marginTop="60dp"
        android:text="清零"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toBottomOf="@+id/btn_stop" />

    <TextView
        android:id="@+id/tv_timer"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="Hello World!"
        android:textSize="30dp"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintEnd_toEndOf="parent"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toTopOf="parent"
        app:layout_constraintVertical_bias="0.272" />

    <Button
        android:id="@+id/btn_start"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_marginStart="36dp"
        android:layout_marginTop="80dp"
        android:text="开始"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toBottomOf="@+id/tv_timer" />

    <Button
        android:id="@+id/btn_stop"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_marginStart="20dp"
        android:text="暂停"
        app:layout_constraintStart_toEndOf="@+id/btn_start"
        app:layout_constraintTop_toTopOf="@+id/btn_start" />

    <Button
        android:id="@+id/btn_resume"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_marginStart="20dp"
        android:text="继续"
        app:layout_constraintStart_toEndOf="@+id/btn_stop"
        app:layout_constraintTop_toTopOf="@+id/btn_stop" />

</androidx.constraintlayout.widget.ConstraintLayout>

mainactivity.java

package com.shudu.timer;
//按钮控制显示计时器。
//开始、暂停、继续、清零四个按钮
//2024.12.30

import android.os.Bundle;
import android.os.Handler;
import android.view.View;
import android.widget.Button;
import android.widget.TextView;

import androidx.activity.EdgeToEdge;
import androidx.appcompat.app.AppCompatActivity;
import androidx.core.graphics.Insets;
import androidx.core.view.ViewCompat;
import androidx.core.view.WindowInsetsCompat;

public class MainActivity extends AppCompatActivity {
    private TextView tvTimer;
    private Button btnStart, btnPause, btnResume,btnClear;

    private Handler handler = new Handler();
    private long startTime=0;
    private long elapsedTime = 0;
    private Runnable timerRunnable;
    private boolean isRunning = false;
    private boolean isPaused = false;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        EdgeToEdge.enable(this);
        setContentView(R.layout.activity_main);
        ViewCompat.setOnApplyWindowInsetsListener(findViewById(R.id.main), (v, insets) -> {
            Insets systemBars = insets.getInsets(WindowInsetsCompat.Type.systemBars());
            v.setPadding(systemBars.left, systemBars.top, systemBars.right, systemBars.bottom);
            return insets;
        });
        tvTimer = findViewById(R.id.tv_timer);
        btnStart = findViewById(R.id.btn_start);
        btnPause = findViewById(R.id.btn_stop);
        btnResume = findViewById(R.id.btn_resume);
        btnClear=findViewById(R.id.btn_clear);
        btnClear.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                clearTimer();
            }
        });
        btnStart.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                if (!isRunning) {
                    startTime = System.currentTimeMillis();
                    isRunning = true;
                    handler.postDelayed(updateTimer, 0);
                }
            }
        });

        btnPause.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                if (isRunning &&!isPaused) {
                    isPaused = true;
                    elapsedTime += System.currentTimeMillis() - startTime;
                    handler.removeCallbacks(updateTimer);
                }
            }
        });

        btnResume.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                if (isPaused) {
                    isPaused = false;
                    startTime = System.currentTimeMillis();
                    handler.postDelayed(updateTimer, 0);
                }
            }
        });

    }

    private void clearTimer() {
        isRunning = false;
//        startTime=0;
        startTime = System.currentTimeMillis();//这行代码什么意思,不写还不行。写成starttime=0也不行。
        tvTimer.setText("00:00:00");
    }

    private Runnable updateTimer = new Runnable() {
        @Override
        public void run() {
            long currentTime;
            if (isRunning) {
                if (isPaused) {
                    currentTime = elapsedTime;
                } else {
                    currentTime = System.currentTimeMillis() - startTime + elapsedTime;
                }
                int seconds = (int) (currentTime / 1000);
                int minutes = seconds / 60;
                int hours = minutes / 60;
                seconds %= 60;
                minutes %= 60;
                tvTimer.setText(String.format("%02d:%02d:%02d", hours, minutes, seconds));
                handler.postDelayed(this, 1000);
            }
        }
    };
}

 25.1.6修改。

上面代码有个bug,清零后再点开始不是从零开始而是继续暂停时间。

如图:

以下代码已修正,增加了一个是否点击“归零”的判断,修改点已备注。

package com.shudu.timer;
//按钮控制显示计时器。
//开始、暂停、继续、清零四个按钮
//2024.12.30

import android.os.Bundle;
import android.os.Handler;
import android.view.View;
import android.widget.Button;
import android.widget.TextView;

import androidx.activity.EdgeToEdge;
import androidx.appcompat.app.AppCompatActivity;
import androidx.core.graphics.Insets;
import androidx.core.view.ViewCompat;
import androidx.core.view.WindowInsetsCompat;

public class MainActivity extends AppCompatActivity {
    private TextView tvTimer;
    private Button btnStart, btnPause, btnResume,btnClear;

    private Handler handler = new Handler();
    private long startTime=0;
    private long elapsedTime = 0;
    private Runnable timerRunnable;
    private boolean isRunning = false;
    private boolean isPaused = false;
    private boolean iszero=false;//增加
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        EdgeToEdge.enable(this);
        setContentView(R.layout.activity_main);
        ViewCompat.setOnApplyWindowInsetsListener(findViewById(R.id.main), (v, insets) -> {
            Insets systemBars = insets.getInsets(WindowInsetsCompat.Type.systemBars());
            v.setPadding(systemBars.left, systemBars.top, systemBars.right, systemBars.bottom);
            return insets;
        });
        tvTimer = findViewById(R.id.tv_timer);
        btnStart = findViewById(R.id.btn_start);
        btnPause = findViewById(R.id.btn_stop);
        btnResume = findViewById(R.id.btn_resume);
        btnClear=findViewById(R.id.btn_clear);


        btnStart.setOnClickListener(new View.OnClickListener() {

            @Override
            public void onClick(View v) {
                if (!isRunning) {
                    startTime = System.currentTimeMillis();
                    isRunning = true;
                    handler.postDelayed(updateTimer, 0);
                }
            }
        });

        btnPause.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                if (isRunning &&!isPaused) {
                    isPaused = true;
                    iszero=false;//增加
                    elapsedTime += System.currentTimeMillis() - startTime;
                    handler.removeCallbacks(updateTimer);
                }
            }
        });

        btnResume.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                if (isPaused) {
                    isPaused = false;
                    startTime = System.currentTimeMillis();
                    handler.postDelayed(updateTimer, 0);
                }
            }
        });
        btnClear.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                clearTimer();
            }
        });
    }

    private void clearTimer() {
        isRunning = false;
        isPaused=false;
        iszero=true;//增加
        startTime = System.currentTimeMillis();//这行代码什么意思,不写还不行。写成starttime=0也不行。
        tvTimer.setText("00:00:00");
    }

    private Runnable updateTimer = new Runnable() {
        @Override
        public void run() {
            long currentTime;
            if (isRunning) {
                if (isPaused) {
                    currentTime = elapsedTime;
                    //下面两行为增加
                } else if(iszero){
                    currentTime = System.currentTimeMillis() - startTime ;
                }else{
                    currentTime = System.currentTimeMillis() - startTime + elapsedTime;
                }
                int seconds = (int) (currentTime / 1000);
                int minutes = seconds / 60;
                int hours = minutes / 60;
                seconds %= 60;
                minutes %= 60;
                tvTimer.setText(String.format("%02d:%02d:%02d", hours, minutes, seconds));
                handler.postDelayed(this, 1000);
            }
        }
    };
}

修改后图:

Android Studio中设计一个简单的计时器,你可以按照以下步骤操作: 1. **新建项目**:打开Android Studio,选择"Start a new Android Studio project",创建一个新的Empty Activity。 2. **布局文件**:在`activity_main.xml`中添加一个TextView用于显示时间,以及两个Button分别代表“开始”和“停止”。例如: ```xml <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="wrap_content" android:orientation="vertical" tools:context=".MainActivity"> <TextView android:id="@+id/timeTextView" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="00:00:00"/> <Button android:id="@+id/startButton" android:text="开始"/> <Button android:id="@+id/stopButton" android:text="停止"/> </LinearLayout> ``` 3. **Java/ Kotlin代码**:在`MainActivity.java`或`MainActivity.kt`中,为按钮设置点击事件监听器,并处理计时逻辑。可以使用Handler和Runnable来更新TextView的内容并持续运行定时任务。这里是一个简单的示例: ```java // Java public class MainActivity extends AppCompatActivity { private TextView timeTextView; private Button startButton, stopButton; private Handler handler = new Handler(); private Runnable timerTask = new Runnable() { @Override public void run() { updateTime(); if (!isRunning) { // 停止计时后,移除Runnable handler.removeCallbacks(this); } else { // 继续计时 handler.postDelayed(this, 1000); // 每秒更新一次 } } private boolean isRunning = false; private void updateTime() { int minutes = (int) (time / 60); int seconds = time % 60; timeTextView.setText(String.format("%02d:%02d", minutes, seconds)); } }; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); timeTextView = findViewById(R.id.timeTextView); startButton = findViewById(R.id.startButton); stopButton = findViewById(R.id.stopButton); startButton.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { if (!isRunning) { isRunning = true; handler.post(timerTask); startButton.setText("停止"); } else { isRunning = false; handler.removeCallbacks(timerTask); startButton.setText("开始"); } } }); stopButton.setVisibility(View.GONE); } } ``` 或者在Kotlin: ```kotlin class MainActivity : AppCompatActivity() { // ... (代码类似) override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) setContentView(R.layout.activity_main) // ... (代码类似) startButton.setOnClickListener { startStopTimer(isRunning = false) } stopButton.setOnClickListener { startStopTimer(isRunning = true) } stopButton.visibility = View.GONE } private fun startStopTimer(isRunning: Boolean) { this.isRunning = isRunning if (isRunning) { handler.postDelayed(object : Runnable() { override fun run() { updateTime() postDelayed(this, 1000L) // 每秒更新一次 } }, 0) startButton.text = "停止" } else { handler.removeCallbacksAndMessages(null) startButton.text = "开始" } } } ``` 4. **运行测试**:将Activity放入模拟器或实际设备上运行,点击“开始”按钮启动计时,点击“停止”按钮暂停计时。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

kim5659

你的鼓励是我创作的最大动力!

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值